Skip to content

Commit 184b433

Browse files
Rahul YadavJulia Boes
Rahul Yadav
authored and
Julia Boes
committedApr 28, 2020
8242999: HTTP/2 client may not handle CONTINUATION frames correctly
Updated jdk.internal.net.http.Stream.incoming(Http2Frame frame) to handle continuation frame with END_HEADER flag Reviewed-by: chegar, dfuchs
1 parent 7a937e0 commit 184b433

File tree

2 files changed

+80
-22
lines changed

2 files changed

+80
-22
lines changed
 

‎src/java.net.http/share/classes/jdk/internal/net/http/Stream.java

+4-4
Original file line numberDiff line numberDiff line change
@@ -389,10 +389,10 @@ void incoming(Http2Frame frame) throws IOException {
389389
if (hframe.endHeaders()) {
390390
Log.logTrace("handling response (streamid={0})", streamid);
391391
handleResponse();
392-
if (hframe.getFlag(HeaderFrame.END_STREAM)) {
393-
if (debug.on()) debug.log("handling END_STREAM: %d", streamid);
394-
receiveDataFrame(new DataFrame(streamid, DataFrame.END_STREAM, List.of()));
395-
}
392+
}
393+
if (hframe.getFlag(HeaderFrame.END_STREAM)) {
394+
if (debug.on()) debug.log("handling END_STREAM: %d", streamid);
395+
receiveDataFrame(new DataFrame(streamid, DataFrame.END_STREAM, List.of()));
396396
}
397397
} else if (frame instanceof DataFrame) {
398398
receiveDataFrame((DataFrame)frame);

‎test/jdk/java/net/httpclient/http2/ContinuationFrameTest.java

+76-18
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
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.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -29,6 +29,7 @@
2929
* java.net.http/jdk.internal.net.http.frame
3030
* java.net.http/jdk.internal.net.http.hpack
3131
* @library /test/lib server
32+
* @compile ../ReferenceTracker.java
3233
* @build Http2TestServer
3334
* @build jdk.test.lib.net.SimpleSSLContext
3435
* @run testng/othervm ContinuationFrameTest
@@ -72,6 +73,9 @@ public class ContinuationFrameTest {
7273
Http2TestServer https2TestServer; // HTTP/2 ( h2 )
7374
String http2URI;
7475
String https2URI;
76+
String noBodyhttp2URI;
77+
String noBodyhttps2URI;
78+
final ReferenceTracker TRACKER = ReferenceTracker.INSTANCE;
7579

7680
/**
7781
* A function that returns a list of 1) a HEADERS frame ( with an empty
@@ -87,6 +91,23 @@ public class ContinuationFrameTest {
8791
return List.of(hf, cf);
8892
};
8993

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+
90111
/**
91112
* A function that returns a list of a HEADERS frame followed by a number of
92113
* CONTINUATION frames. Each frame contains just a single byte of payload.
@@ -112,15 +133,20 @@ public class ContinuationFrameTest {
112133
@DataProvider(name = "variants")
113134
public Object[][] variants() {
114135
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 },
124150
};
125151
}
126152

@@ -136,8 +162,13 @@ void test(String uri,
136162

137163
HttpClient client = null;
138164
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+
}
141172

142173
HttpRequest request = HttpRequest.newBuilder(URI.create(uri))
143174
.POST(BodyPublishers.ofString("Hello there!"))
@@ -149,6 +180,13 @@ void test(String uri,
149180
resp = client.sendAsync(request, BodyHandlers.ofString()).join();
150181
}
151182

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+
}
152190
out.println("Got response: " + resp);
153191
out.println("Got body: " + resp.body());
154192
assertTrue(resp.statusCode() == 200,
@@ -166,13 +204,17 @@ public void setup() throws Exception {
166204

167205
http2TestServer = new Http2TestServer("localhost", false, 0);
168206
http2TestServer.addHandler(new Http2EchoHandler(), "/http2/echo");
207+
http2TestServer.addHandler(new Http2NoBodyHandler(), "/http2/nobody");
169208
int port = http2TestServer.getAddress().getPort();
170209
http2URI = "http://localhost:" + port + "/http2/echo";
210+
noBodyhttp2URI = "http://localhost:" + port + "/http2/nobody";
171211

172212
https2TestServer = new Http2TestServer("localhost", true, sslContext);
173213
https2TestServer.addHandler(new Http2EchoHandler(), "/https2/echo");
214+
https2TestServer.addHandler(new Http2NoBodyHandler(), "/https2/nobody");
174215
port = https2TestServer.getAddress().getPort();
175216
https2URI = "https://localhost:" + port + "/https2/echo";
217+
noBodyhttps2URI = "https://localhost:" + port + "/https2/nobody";
176218

177219
// Override the default exchange supplier with a custom one to enable
178220
// particular test scenarios
@@ -185,8 +227,15 @@ public void setup() throws Exception {
185227

186228
@AfterTest
187229
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+
}
190239
}
191240

192241
static class Http2EchoHandler implements Http2Handler {
@@ -204,6 +253,17 @@ public void handle(Http2TestExchange t) throws IOException {
204253
}
205254
}
206255

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+
207267
// A custom Http2TestExchangeImpl that overrides sendResponseHeaders to
208268
// allow headers to be sent with a number of CONTINUATION frames.
209269
static class CFTHttp2TestExchange extends Http2TestExchangeImpl {
@@ -225,7 +285,7 @@ static void setHeaderFrameSupplier(BiFunction<Integer,List<ByteBuffer>,List<Http
225285
@Override
226286
public void sendResponseHeaders(int rCode, long responseLength) throws IOException {
227287
this.responseLength = responseLength;
228-
if (responseLength > 0 || responseLength < 0) {
288+
if (responseLength != 0 && rCode != 204) {
229289
long clen = responseLength > 0 ? responseLength : 0;
230290
rspheadersBuilder.setHeader("Content-length", Long.toString(clen));
231291
}
@@ -236,10 +296,8 @@ public void sendResponseHeaders(int rCode, long responseLength) throws IOExcepti
236296
List<Http2Frame> headerFrames = headerFrameSupplier.apply(streamid, encodeHeaders);
237297
assert headerFrames.size() > 0; // there must always be at least 1
238298

239-
if (responseLength < 0) {
240-
headerFrames.get(headerFrames.size() -1).setFlag(HeadersFrame.END_STREAM);
299+
if(headerFrames.get(0).getFlag(HeaderFrame.END_STREAM))
241300
os.closeInternal();
242-
}
243301

244302
for (Http2Frame f : headerFrames)
245303
conn.outputQ.put(f);

0 commit comments

Comments
 (0)
Please sign in to comment.