JdkZlibDecoder may corrupt data when GZIP footer is fragmented (#11521)
Motivation: Due a bug in the statemachine we produced an decoding error when the GZIP footer was fragmented in some cases Modifications: - Fix statemachine - Add testcase Result: Correctly decode GZIP in all cases
This commit is contained in:
parent
412af40f5d
commit
c83c5b7d7c
@ -204,13 +204,16 @@ public class JdkZlibDecoder extends ZlibDecoder {
|
|||||||
if (gzipState != GzipState.HEADER_END) {
|
if (gzipState != GzipState.HEADER_END) {
|
||||||
if (gzipState == GzipState.FOOTER_START) {
|
if (gzipState == GzipState.FOOTER_START) {
|
||||||
if (!handleGzipFooter(in)) {
|
if (!handleGzipFooter(in)) {
|
||||||
|
// Either there was not enough data or the input is finished.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else {
|
// If we consumed the footer we will start with the header again.
|
||||||
|
assert gzipState == GzipState.HEADER_START;
|
||||||
|
}
|
||||||
if (!readGZIPHeader(in)) {
|
if (!readGZIPHeader(in)) {
|
||||||
|
// There was not enough data readable to read the GZIP header.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
// Some bytes may have been consumed, and so we must re-set the number of readable bytes.
|
// Some bytes may have been consumed, and so we must re-set the number of readable bytes.
|
||||||
readableBytes = in.readableBytes();
|
readableBytes = in.readableBytes();
|
||||||
if (readableBytes == 0) {
|
if (readableBytes == 0) {
|
||||||
|
@ -20,16 +20,20 @@ import io.netty.buffer.Unpooled;
|
|||||||
import io.netty.channel.embedded.EmbeddedChannel;
|
import io.netty.channel.embedded.EmbeddedChannel;
|
||||||
import io.netty.util.CharsetUtil;
|
import io.netty.util.CharsetUtil;
|
||||||
import io.netty.util.ReferenceCountUtil;
|
import io.netty.util.ReferenceCountUtil;
|
||||||
|
import io.netty.util.internal.PlatformDependent;
|
||||||
import org.apache.commons.compress.utils.IOUtils;
|
import org.apache.commons.compress.utils.IOUtils;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.api.function.Executable;
|
import org.junit.jupiter.api.function.Executable;
|
||||||
|
|
||||||
|
import java.io.ByteArrayOutputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Queue;
|
import java.util.Queue;
|
||||||
|
import java.util.zip.GZIPOutputStream;
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
import static org.junit.jupiter.api.Assertions.assertFalse;
|
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertNull;
|
||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
|
||||||
@ -129,4 +133,35 @@ public class JdkZlibTest extends ZlibTest {
|
|||||||
chDecoderGZip.close();
|
chDecoderGZip.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDecodeWithHeaderFollowingFooter() throws Exception {
|
||||||
|
byte[] bytes = new byte[1024];
|
||||||
|
PlatformDependent.threadLocalRandom().nextBytes(bytes);
|
||||||
|
ByteArrayOutputStream bytesOut = new ByteArrayOutputStream();
|
||||||
|
GZIPOutputStream out = new GZIPOutputStream(bytesOut);
|
||||||
|
out.write(bytes);
|
||||||
|
out.close();
|
||||||
|
|
||||||
|
byte[] compressed = bytesOut.toByteArray();
|
||||||
|
ByteBuf buffer = Unpooled.buffer().writeBytes(compressed).writeBytes(compressed);
|
||||||
|
EmbeddedChannel channel = new EmbeddedChannel(new JdkZlibDecoder(ZlibWrapper.GZIP, true));
|
||||||
|
// Write it into the Channel in a way that we were able to decompress the first data completely but not the
|
||||||
|
// whole footer.
|
||||||
|
assertTrue(channel.writeInbound(buffer.readRetainedSlice(compressed.length - 1)));
|
||||||
|
assertTrue(channel.writeInbound(buffer));
|
||||||
|
assertTrue(channel.finish());
|
||||||
|
|
||||||
|
ByteBuf uncompressedBuffer = Unpooled.wrappedBuffer(bytes);
|
||||||
|
ByteBuf read = channel.readInbound();
|
||||||
|
assertEquals(uncompressedBuffer, read);
|
||||||
|
read.release();
|
||||||
|
|
||||||
|
read = channel.readInbound();
|
||||||
|
assertEquals(uncompressedBuffer, read);
|
||||||
|
read.release();
|
||||||
|
|
||||||
|
assertNull(channel.readInbound());
|
||||||
|
uncompressedBuffer.release();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user