Support empty http responses when using compression
Motivation: Found a bug in that netty would generate a 20 byte body when returing a response to an HTTP HEAD. the 20 bytes seems to be related to the compression footer. RFC2616, section 9.4 states that responses to an HTTP HEAD MUST not return a message body in the response. Netty's own client implementation expected an empty response. The extra bytes lead to a 2nd response with an error decoder result: java.lang.IllegalArgumentException: invalid version format: 14 Modifications: Track the HTTP request method. When processing the response we determine if the response is passthru unnchanged. This decision now takes into account the request method and passthru responses related to HTTP HEAD requests. Result: Netty's http client works and better RFC conformance.
This commit is contained in:
parent
a6887b550e
commit
89f8dace60
@ -56,6 +56,9 @@ public abstract class HttpContentEncoder extends MessageToMessageCodec<HttpReque
|
||||
AWAIT_CONTENT
|
||||
}
|
||||
|
||||
private static final CharSequence ZERO_LENGTH_HEAD = "HEAD";
|
||||
private static final CharSequence ZERO_LENGTH_CONNECT = "CONNECT";
|
||||
|
||||
private final Queue<CharSequence> acceptEncodingQueue = new ArrayDeque<CharSequence>();
|
||||
private CharSequence acceptEncoding;
|
||||
private EmbeddedChannel encoder;
|
||||
@ -73,6 +76,14 @@ public abstract class HttpContentEncoder extends MessageToMessageCodec<HttpReque
|
||||
if (acceptedEncoding == null) {
|
||||
acceptedEncoding = HttpHeaderValues.IDENTITY;
|
||||
}
|
||||
|
||||
HttpMethod meth = msg.method();
|
||||
if (meth == HttpMethod.HEAD) {
|
||||
acceptedEncoding = ZERO_LENGTH_HEAD;
|
||||
} else if (meth == HttpMethod.CONNECT) {
|
||||
acceptedEncoding = ZERO_LENGTH_CONNECT;
|
||||
}
|
||||
|
||||
acceptEncodingQueue.add(acceptedEncoding);
|
||||
out.add(ReferenceCountUtil.retain(msg));
|
||||
}
|
||||
@ -87,12 +98,24 @@ public abstract class HttpContentEncoder extends MessageToMessageCodec<HttpReque
|
||||
|
||||
final HttpResponse res = (HttpResponse) msg;
|
||||
|
||||
// Get the list of encodings accepted by the peer.
|
||||
acceptEncoding = acceptEncodingQueue.poll();
|
||||
if (acceptEncoding == null) {
|
||||
throw new IllegalStateException("cannot send more responses than requests");
|
||||
}
|
||||
|
||||
/*
|
||||
* per rfc2616 4.3 Message Body
|
||||
* All 1xx (informational), 204 (no content), and 304 (not modified) responses MUST NOT include a
|
||||
* message-body. All other responses do include a message-body, although it MAY be of zero length.
|
||||
*
|
||||
* 9.4 HEAD
|
||||
* The HEAD method is identical to GET except that the server MUST NOT return a message-body
|
||||
* in the response.
|
||||
*
|
||||
* This code is now inline with HttpClientDecoder.Decoder
|
||||
*/
|
||||
if (isPassthru(res)) {
|
||||
if (isPassthru(res, acceptEncoding)) {
|
||||
if (isFull) {
|
||||
out.add(ReferenceCountUtil.retain(res));
|
||||
} else {
|
||||
@ -103,12 +126,6 @@ public abstract class HttpContentEncoder extends MessageToMessageCodec<HttpReque
|
||||
break;
|
||||
}
|
||||
|
||||
// Get the list of encodings accepted by the peer.
|
||||
acceptEncoding = acceptEncodingQueue.poll();
|
||||
if (acceptEncoding == null) {
|
||||
throw new IllegalStateException("cannot send more responses than requests");
|
||||
}
|
||||
|
||||
if (isFull) {
|
||||
// Pass through the full response with empty content and continue waiting for the the next resp.
|
||||
if (!((ByteBufHolder) res).content().isReadable()) {
|
||||
@ -179,9 +196,10 @@ public abstract class HttpContentEncoder extends MessageToMessageCodec<HttpReque
|
||||
}
|
||||
}
|
||||
|
||||
private static boolean isPassthru(HttpResponse res) {
|
||||
private static boolean isPassthru(HttpResponse res, CharSequence httpMethod) {
|
||||
final int code = res.status().code();
|
||||
return code < 200 || code == 204 || code == 304;
|
||||
boolean expectEmptyBody = httpMethod == ZERO_LENGTH_HEAD || (httpMethod == ZERO_LENGTH_CONNECT && code == 200);
|
||||
return code < 200 || code == 204 || code == 304 || expectEmptyBody;
|
||||
}
|
||||
|
||||
private static void ensureHeaders(HttpObject msg) {
|
||||
|
@ -252,6 +252,96 @@ public class HttpContentEncoderTest {
|
||||
assertThat(ch.readOutbound(), is(nullValue()));
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testEmptyHeadResponse() throws Exception {
|
||||
EmbeddedChannel ch = new EmbeddedChannel(new TestEncoder());
|
||||
HttpRequest req = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.HEAD, "/");
|
||||
ch.writeInbound(req);
|
||||
|
||||
HttpResponse res = new DefaultHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK);
|
||||
res.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED);
|
||||
ch.writeOutbound(res);
|
||||
ch.writeOutbound(LastHttpContent.EMPTY_LAST_CONTENT);
|
||||
|
||||
assertEmptyResponse(ch);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testHttp304Response() throws Exception {
|
||||
EmbeddedChannel ch = new EmbeddedChannel(new TestEncoder());
|
||||
HttpRequest req = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.GET, "/");
|
||||
req.headers().set(HttpHeaderNames.ACCEPT_ENCODING, HttpHeaderValues.GZIP);
|
||||
ch.writeInbound(req);
|
||||
|
||||
HttpResponse res = new DefaultHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.NOT_MODIFIED);
|
||||
res.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED);
|
||||
ch.writeOutbound(res);
|
||||
ch.writeOutbound(LastHttpContent.EMPTY_LAST_CONTENT);
|
||||
|
||||
assertEmptyResponse(ch);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testConnect200Response() throws Exception {
|
||||
EmbeddedChannel ch = new EmbeddedChannel(new TestEncoder());
|
||||
HttpRequest req = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.CONNECT, "google.com:80");
|
||||
ch.writeInbound(req);
|
||||
|
||||
HttpResponse res = new DefaultHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.OK);
|
||||
res.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED);
|
||||
ch.writeOutbound(res);
|
||||
ch.writeOutbound(LastHttpContent.EMPTY_LAST_CONTENT);
|
||||
|
||||
assertEmptyResponse(ch);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testConnectFailureResponse() throws Exception {
|
||||
String content = "Not allowed by configuration";
|
||||
|
||||
EmbeddedChannel ch = new EmbeddedChannel(new TestEncoder());
|
||||
HttpRequest req = new DefaultFullHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.CONNECT, "google.com:80");
|
||||
ch.writeInbound(req);
|
||||
|
||||
HttpResponse res = new DefaultHttpResponse(HttpVersion.HTTP_1_1, HttpResponseStatus.METHOD_NOT_ALLOWED);
|
||||
res.headers().set(HttpHeaderNames.TRANSFER_ENCODING, HttpHeaderValues.CHUNKED);
|
||||
ch.writeOutbound(res);
|
||||
ch.writeOutbound(new DefaultHttpContent(Unpooled.wrappedBuffer(content.getBytes(CharsetUtil.UTF_8))));
|
||||
ch.writeOutbound(LastHttpContent.EMPTY_LAST_CONTENT);
|
||||
|
||||
assertEncodedResponse(ch);
|
||||
Object o = ch.readOutbound();
|
||||
assertThat(o, is(instanceOf(HttpContent.class)));
|
||||
HttpContent chunk = (HttpContent) o;
|
||||
assertThat(chunk.content().toString(CharsetUtil.US_ASCII), is("28"));
|
||||
chunk.release();
|
||||
|
||||
chunk = ch.readOutbound();
|
||||
assertThat(chunk.content().isReadable(), is(true));
|
||||
assertThat(chunk.content().toString(CharsetUtil.US_ASCII), is("0"));
|
||||
chunk.release();
|
||||
|
||||
chunk = ch.readOutbound();
|
||||
assertThat(chunk, is(instanceOf(LastHttpContent.class)));
|
||||
chunk.release();
|
||||
assertThat(ch.readOutbound(), is(nullValue()));
|
||||
}
|
||||
|
||||
private static void assertEmptyResponse(EmbeddedChannel ch) {
|
||||
Object o = ch.readOutbound();
|
||||
assertThat(o, is(instanceOf(HttpResponse.class)));
|
||||
|
||||
HttpResponse res = (HttpResponse) o;
|
||||
assertThat(res, is(not(instanceOf(HttpContent.class))));
|
||||
assertThat(res.headers().getAndConvert(HttpHeaderNames.TRANSFER_ENCODING), is("chunked"));
|
||||
assertThat(res.headers().get(HttpHeaderNames.CONTENT_LENGTH), is(nullValue()));
|
||||
|
||||
HttpContent chunk = ch.readOutbound();
|
||||
assertThat(chunk, is(instanceOf(LastHttpContent.class)));
|
||||
chunk.release();
|
||||
assertThat(ch.readOutbound(), is(nullValue()));
|
||||
}
|
||||
|
||||
private static void assertEncodedResponse(EmbeddedChannel ch) {
|
||||
Object o = ch.readOutbound();
|
||||
assertThat(o, is(instanceOf(HttpResponse.class)));
|
||||
|
Loading…
x
Reference in New Issue
Block a user