HttpContentDecompressor should change decompressed requests to chunked encoding. Fixes issue #5428
`HttpContentDecoder` was removing `Content-Length` header but not adding a `Transfer-Encoding` header which goes against the HTTP spec. Added `Transfer-Encoding` header with value `chunked` when `Content-Length` is removed. Modified existing unit test to also check for this condition. Compliance with HTTP spec.
This commit is contained in:
parent
8ae0b4530b
commit
b033f730f8
@ -97,6 +97,7 @@ public abstract class HttpContentDecoder extends MessageToMessageDecoder<HttpObj
|
||||
// If buffering is not an issue, add HttpObjectAggregator down the chain, it will set the header.
|
||||
// Otherwise, rely on LastHttpContent message.
|
||||
headers.remove(HttpHeaders.Names.CONTENT_LENGTH);
|
||||
headers.set(HttpHeaders.Names.TRANSFER_ENCODING, HttpHeaders.Values.CHUNKED);
|
||||
|
||||
// set new content encoding,
|
||||
CharSequence targetContentEncoding = getTargetContentEncoding(contentEncoding);
|
||||
|
@ -301,9 +301,12 @@ public class HttpContentDecoderTest {
|
||||
Object o = resp.peek();
|
||||
assertThat(o, is(instanceOf(HttpResponse.class)));
|
||||
HttpResponse r = (HttpResponse) o;
|
||||
String v = r.headers().get(HttpHeaders.Names.CONTENT_LENGTH);
|
||||
Long value = v == null ? null : Long.parseLong(v);
|
||||
assertTrue(value == null || value.longValue() == HELLO_WORLD.length());
|
||||
|
||||
assertFalse("Content-Length header not removed.", r.headers().contains(HttpHeaders.Names.CONTENT_LENGTH));
|
||||
|
||||
String transferEncoding = r.headers().get(HttpHeaders.Names.TRANSFER_ENCODING);
|
||||
assertNotNull("Content-length as well as transfer-encoding not set.", transferEncoding);
|
||||
assertEquals("Unexpected transfer-encoding value.", HttpHeaders.Values.CHUNKED.toString(), transferEncoding);
|
||||
|
||||
assertHasInboundMessages(channel, true);
|
||||
assertHasOutboundMessages(channel, false);
|
||||
@ -356,24 +359,9 @@ public class HttpContentDecoderTest {
|
||||
Queue<Object> req = channel.inboundMessages();
|
||||
assertTrue(req.size() > 1);
|
||||
int contentLength = 0;
|
||||
for (Object o : req) {
|
||||
if (o instanceof HttpContent) {
|
||||
assertTrue(((HttpContent) o).refCnt() > 0);
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
contentLength += b.readableBytes();
|
||||
}
|
||||
}
|
||||
contentLength = calculateContentLength(req, contentLength);
|
||||
|
||||
int readCount = 0;
|
||||
byte[] receivedContent = new byte[contentLength];
|
||||
for (Object o : req) {
|
||||
if (o instanceof HttpContent) {
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
int readableBytes = b.readableBytes();
|
||||
b.readBytes(receivedContent, readCount, readableBytes);
|
||||
readCount += readableBytes;
|
||||
}
|
||||
}
|
||||
byte[] receivedContent = readContent(req, contentLength);
|
||||
|
||||
assertEquals(HELLO_WORLD, new String(receivedContent, CharsetUtil.US_ASCII));
|
||||
|
||||
@ -398,24 +386,9 @@ public class HttpContentDecoderTest {
|
||||
Queue<Object> resp = channel.inboundMessages();
|
||||
assertTrue(resp.size() > 1);
|
||||
int contentLength = 0;
|
||||
for (Object o : resp) {
|
||||
if (o instanceof HttpContent) {
|
||||
assertTrue(((HttpContent) o).refCnt() > 0);
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
contentLength += b.readableBytes();
|
||||
}
|
||||
}
|
||||
contentLength = calculateContentLength(resp, contentLength);
|
||||
|
||||
int readCount = 0;
|
||||
byte[] receivedContent = new byte[contentLength];
|
||||
for (Object o : resp) {
|
||||
if (o instanceof HttpContent) {
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
int readableBytes = b.readableBytes();
|
||||
b.readBytes(receivedContent, readCount, readableBytes);
|
||||
readCount += readableBytes;
|
||||
}
|
||||
}
|
||||
byte[] receivedContent = readContent(resp, contentLength);
|
||||
|
||||
assertEquals(HELLO_WORLD, new String(receivedContent, CharsetUtil.US_ASCII));
|
||||
|
||||
@ -424,7 +397,7 @@ public class HttpContentDecoderTest {
|
||||
assertFalse(channel.finish());
|
||||
}
|
||||
|
||||
private byte[] gzDecompress(byte[] input) {
|
||||
private static byte[] gzDecompress(byte[] input) {
|
||||
ZlibDecoder decoder = ZlibCodecFactory.newZlibDecoder(ZlibWrapper.GZIP);
|
||||
EmbeddedChannel channel = new EmbeddedChannel(decoder);
|
||||
assertTrue(channel.writeInbound(Unpooled.wrappedBuffer(input)));
|
||||
@ -450,7 +423,32 @@ public class HttpContentDecoderTest {
|
||||
return output;
|
||||
}
|
||||
|
||||
private byte[] gzCompress(byte[] input) {
|
||||
private static byte[] readContent(Queue<Object> req, int contentLength) {
|
||||
byte[] receivedContent = new byte[contentLength];
|
||||
int readCount = 0;
|
||||
for (Object o : req) {
|
||||
if (o instanceof HttpContent) {
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
int readableBytes = b.readableBytes();
|
||||
b.readBytes(receivedContent, readCount, readableBytes);
|
||||
readCount += readableBytes;
|
||||
}
|
||||
}
|
||||
return receivedContent;
|
||||
}
|
||||
|
||||
private static int calculateContentLength(Queue<Object> req, int contentLength) {
|
||||
for (Object o : req) {
|
||||
if (o instanceof HttpContent) {
|
||||
assertTrue(((HttpContent) o).refCnt() > 0);
|
||||
ByteBuf b = ((HttpContent) o).content();
|
||||
contentLength += b.readableBytes();
|
||||
}
|
||||
}
|
||||
return contentLength;
|
||||
}
|
||||
|
||||
private static byte[] gzCompress(byte[] input) {
|
||||
ZlibEncoder encoder = ZlibCodecFactory.newZlibEncoder(ZlibWrapper.GZIP);
|
||||
EmbeddedChannel channel = new EmbeddedChannel(encoder);
|
||||
assertTrue(channel.writeOutbound(Unpooled.wrappedBuffer(input)));
|
||||
@ -476,7 +474,7 @@ public class HttpContentDecoderTest {
|
||||
return output;
|
||||
}
|
||||
|
||||
private void assertHasInboundMessages(EmbeddedChannel channel, boolean hasMessages) {
|
||||
private static void assertHasInboundMessages(EmbeddedChannel channel, boolean hasMessages) {
|
||||
Object o;
|
||||
if (hasMessages) {
|
||||
while (true) {
|
||||
@ -493,7 +491,7 @@ public class HttpContentDecoderTest {
|
||||
}
|
||||
}
|
||||
|
||||
private void assertHasOutboundMessages(EmbeddedChannel channel, boolean hasMessages) {
|
||||
private static void assertHasOutboundMessages(EmbeddedChannel channel, boolean hasMessages) {
|
||||
Object o;
|
||||
if (hasMessages) {
|
||||
while (true) {
|
||||
|
Loading…
Reference in New Issue
Block a user