1
1
/*
2
- * Copyright (c) 2017, 2018 , Oracle and/or its affiliates. All rights reserved.
2
+ * Copyright (c) 2017, 2020 , Oracle and/or its affiliates. All rights reserved.
3
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
4
*
5
5
* This code is free software; you can redistribute it and/or modify it
29
29
* java.net.http/jdk.internal.net.http.frame
30
30
* java.net.http/jdk.internal.net.http.hpack
31
31
* @library /test/lib server
32
+ * @compile ../ReferenceTracker.java
32
33
* @build Http2TestServer
33
34
* @build jdk.test.lib.net.SimpleSSLContext
34
35
* @run testng/othervm ContinuationFrameTest
@@ -72,6 +73,9 @@ public class ContinuationFrameTest {
72
73
Http2TestServer https2TestServer ; // HTTP/2 ( h2 )
73
74
String http2URI ;
74
75
String https2URI ;
76
+ String noBodyhttp2URI ;
77
+ String noBodyhttps2URI ;
78
+ final ReferenceTracker TRACKER = ReferenceTracker .INSTANCE ;
75
79
76
80
/**
77
81
* A function that returns a list of 1) a HEADERS frame ( with an empty
@@ -87,6 +91,23 @@ public class ContinuationFrameTest {
87
91
return List .of (hf , cf );
88
92
};
89
93
94
+ /**
95
+ * A function that returns a list of 1) a HEADERS frame with END_STREAM
96
+ * ( and with an empty payload ), and 2) two CONTINUATION frames,the first
97
+ * is empty and the second contains headers and the END_HEADERS flag
98
+ */
99
+ static BiFunction <Integer ,List <ByteBuffer >,List <Http2Frame >> twoContinuation =
100
+ (Integer streamid , List <ByteBuffer > encodedHeaders ) -> {
101
+ List <ByteBuffer > empty = List .of (ByteBuffer .wrap (new byte [0 ]));
102
+ HeadersFrame hf = new HeadersFrame (streamid , HeaderFrame .END_STREAM , empty );
103
+ ContinuationFrame cf = new ContinuationFrame (streamid , 0 ,empty );
104
+ ContinuationFrame cf1 = new ContinuationFrame (streamid ,
105
+ HeaderFrame .END_HEADERS ,
106
+ encodedHeaders );
107
+
108
+ return List .of (hf , cf , cf1 );
109
+ };
110
+
90
111
/**
91
112
* A function that returns a list of a HEADERS frame followed by a number of
92
113
* CONTINUATION frames. Each frame contains just a single byte of payload.
@@ -112,15 +133,20 @@ public class ContinuationFrameTest {
112
133
@ DataProvider (name = "variants" )
113
134
public Object [][] variants () {
114
135
return new Object [][] {
115
- { http2URI , false , oneContinuation },
116
- { https2URI , false , oneContinuation },
117
- { http2URI , true , oneContinuation },
118
- { https2URI , true , oneContinuation },
119
-
120
- { http2URI , false , byteAtATime },
121
- { https2URI , false , byteAtATime },
122
- { http2URI , true , byteAtATime },
123
- { https2URI , true , byteAtATime },
136
+ { http2URI , false , oneContinuation },
137
+ { https2URI , false , oneContinuation },
138
+ { http2URI , true , oneContinuation },
139
+ { https2URI , true , oneContinuation },
140
+
141
+ { noBodyhttp2URI , false , twoContinuation },
142
+ { noBodyhttp2URI , true , twoContinuation },
143
+ { noBodyhttps2URI , false , twoContinuation },
144
+ { noBodyhttps2URI , true , twoContinuation },
145
+
146
+ { http2URI , false , byteAtATime },
147
+ { https2URI , false , byteAtATime },
148
+ { http2URI , true , byteAtATime },
149
+ { https2URI , true , byteAtATime },
124
150
};
125
151
}
126
152
@@ -136,8 +162,13 @@ void test(String uri,
136
162
137
163
HttpClient client = null ;
138
164
for (int i =0 ; i < ITERATION_COUNT ; i ++) {
139
- if (!sameClient || client == null )
140
- client = HttpClient .newBuilder ().sslContext (sslContext ).build ();
165
+ if (!sameClient || client == null ) {
166
+ client = HttpClient .newBuilder ()
167
+ .proxy (HttpClient .Builder .NO_PROXY )
168
+ .sslContext (sslContext )
169
+ .build ();
170
+ TRACKER .track (client );
171
+ }
141
172
142
173
HttpRequest request = HttpRequest .newBuilder (URI .create (uri ))
143
174
.POST (BodyPublishers .ofString ("Hello there!" ))
@@ -149,6 +180,13 @@ void test(String uri,
149
180
resp = client .sendAsync (request , BodyHandlers .ofString ()).join ();
150
181
}
151
182
183
+ if (uri .contains ("nobody" )) {
184
+ out .println ("Got response: " + resp );
185
+ assertTrue (resp .statusCode () == 204 ,
186
+ "Expected 204, got:" + resp .statusCode ());
187
+ assertEquals (resp .version (), HTTP_2 );
188
+ continue ;
189
+ }
152
190
out .println ("Got response: " + resp );
153
191
out .println ("Got body: " + resp .body ());
154
192
assertTrue (resp .statusCode () == 200 ,
@@ -166,13 +204,17 @@ public void setup() throws Exception {
166
204
167
205
http2TestServer = new Http2TestServer ("localhost" , false , 0 );
168
206
http2TestServer .addHandler (new Http2EchoHandler (), "/http2/echo" );
207
+ http2TestServer .addHandler (new Http2NoBodyHandler (), "/http2/nobody" );
169
208
int port = http2TestServer .getAddress ().getPort ();
170
209
http2URI = "http://localhost:" + port + "/http2/echo" ;
210
+ noBodyhttp2URI = "http://localhost:" + port + "/http2/nobody" ;
171
211
172
212
https2TestServer = new Http2TestServer ("localhost" , true , sslContext );
173
213
https2TestServer .addHandler (new Http2EchoHandler (), "/https2/echo" );
214
+ https2TestServer .addHandler (new Http2NoBodyHandler (), "/https2/nobody" );
174
215
port = https2TestServer .getAddress ().getPort ();
175
216
https2URI = "https://localhost:" + port + "/https2/echo" ;
217
+ noBodyhttps2URI = "https://localhost:" + port + "/https2/nobody" ;
176
218
177
219
// Override the default exchange supplier with a custom one to enable
178
220
// particular test scenarios
@@ -185,8 +227,15 @@ public void setup() throws Exception {
185
227
186
228
@ AfterTest
187
229
public void teardown () throws Exception {
188
- http2TestServer .stop ();
189
- https2TestServer .stop ();
230
+ AssertionError fail = TRACKER .check (500 );
231
+ try {
232
+ http2TestServer .stop ();
233
+ https2TestServer .stop ();
234
+ } finally {
235
+ if (fail != null ) {
236
+ throw fail ;
237
+ }
238
+ }
190
239
}
191
240
192
241
static class Http2EchoHandler implements Http2Handler {
@@ -204,6 +253,17 @@ public void handle(Http2TestExchange t) throws IOException {
204
253
}
205
254
}
206
255
256
+ static class Http2NoBodyHandler implements Http2Handler {
257
+ @ Override
258
+ public void handle (Http2TestExchange t ) throws IOException {
259
+ try (InputStream is = t .getRequestBody ();
260
+ OutputStream os = t .getResponseBody ()) {
261
+ byte [] bytes = is .readAllBytes ();
262
+ t .sendResponseHeaders (204 , -1 );
263
+ }
264
+ }
265
+ }
266
+
207
267
// A custom Http2TestExchangeImpl that overrides sendResponseHeaders to
208
268
// allow headers to be sent with a number of CONTINUATION frames.
209
269
static class CFTHttp2TestExchange extends Http2TestExchangeImpl {
@@ -225,7 +285,7 @@ static void setHeaderFrameSupplier(BiFunction<Integer,List<ByteBuffer>,List<Http
225
285
@ Override
226
286
public void sendResponseHeaders (int rCode , long responseLength ) throws IOException {
227
287
this .responseLength = responseLength ;
228
- if (responseLength > 0 || responseLength < 0 ) {
288
+ if (responseLength != 0 && rCode != 204 ) {
229
289
long clen = responseLength > 0 ? responseLength : 0 ;
230
290
rspheadersBuilder .setHeader ("Content-length" , Long .toString (clen ));
231
291
}
@@ -236,10 +296,8 @@ public void sendResponseHeaders(int rCode, long responseLength) throws IOExcepti
236
296
List <Http2Frame > headerFrames = headerFrameSupplier .apply (streamid , encodeHeaders );
237
297
assert headerFrames .size () > 0 ; // there must always be at least 1
238
298
239
- if (responseLength < 0 ) {
240
- headerFrames .get (headerFrames .size () -1 ).setFlag (HeadersFrame .END_STREAM );
299
+ if (headerFrames .get (0 ).getFlag (HeaderFrame .END_STREAM ))
241
300
os .closeInternal ();
242
- }
243
301
244
302
for (Http2Frame f : headerFrames )
245
303
conn .outputQ .put (f );
0 commit comments