Skip to content

Commit d3f3011

Browse files
archiecobbsjaikiran
authored andcommittedMar 20, 2024
7036144: GZIPInputStream readTrailer uses faulty available() test for end-of-stream
Reviewed-by: jpai
1 parent e5e7cd2 commit d3f3011

File tree

2 files changed

+102
-15
lines changed

2 files changed

+102
-15
lines changed
 

‎src/java.base/share/classes/java/util/zip/GZIPInputStream.java

+9-15
Original file line numberDiff line numberDiff line change
@@ -237,23 +237,17 @@ public void close() throws IOException {}
237237
(readUInt(in) != (inf.getBytesWritten() & 0xffffffffL)))
238238
throw new ZipException("Corrupt GZIP trailer");
239239

240-
// If there are more bytes available in "in" or
241-
// the leftover in the "inf" is > 26 bytes:
242-
// this.trailer(8) + next.header.min(10) + next.trailer(8)
243240
// try concatenated case
244-
if (this.in.available() > 0 || n > 26) {
245-
int m = 8; // this.trailer
246-
try {
247-
m += readHeader(in); // next.header
248-
} catch (IOException ze) {
249-
return true; // ignore any malformed, do nothing
250-
}
251-
inf.reset();
252-
if (n > m)
253-
inf.setInput(buf, len - n + m, n - m);
254-
return false;
241+
int m = 8; // this.trailer
242+
try {
243+
m += readHeader(in); // next.header
244+
} catch (IOException ze) {
245+
return true; // ignore any malformed, do nothing
255246
}
256-
return true;
247+
inf.reset();
248+
if (n > m)
249+
inf.setInput(buf, len - n + m, n - m);
250+
return false;
257251
}
258252

259253
/*
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
/*
2+
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/* @test
25+
* @bug 7036144
26+
* @summary Test concatenated gz streams when available() returns zero
27+
* @run junit GZIPInputStreamAvailable
28+
*/
29+
30+
import org.junit.jupiter.api.Test;
31+
32+
import java.io.*;
33+
import java.util.*;
34+
import java.util.zip.*;
35+
36+
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
37+
38+
public class GZIPInputStreamAvailable {
39+
40+
public static final int NUM_COPIES = 100;
41+
42+
@Test
43+
public void testZeroAvailable() throws IOException {
44+
45+
// Create some uncompressed data and then repeat it NUM_COPIES times
46+
byte[] uncompressed1 = "this is a test".getBytes("ASCII");
47+
byte[] uncompressedN = repeat(uncompressed1, NUM_COPIES);
48+
49+
// Compress the original data and then repeat that NUM_COPIES times
50+
byte[] compressed1 = deflate(uncompressed1);
51+
byte[] compressedN = repeat(compressed1, NUM_COPIES);
52+
53+
// (a) Read back inflated data from a stream where available() is accurate and verify
54+
byte[] readback1 = inflate(new ByteArrayInputStream(compressedN));
55+
assertArrayEquals(uncompressedN, readback1);
56+
57+
// (b) Read back inflated data from a stream where available() always returns zero and verify
58+
byte[] readback2 = inflate(new ZeroAvailableStream(new ByteArrayInputStream(compressedN)));
59+
assertArrayEquals(uncompressedN, readback2);
60+
}
61+
62+
public static byte[] repeat(byte[] data, int count) {
63+
byte[] repeat = new byte[data.length * count];
64+
int off = 0;
65+
for (int i = 0; i < count; i++) {
66+
System.arraycopy(data, 0, repeat, off, data.length);
67+
off += data.length;
68+
}
69+
return repeat;
70+
}
71+
72+
public static byte[] deflate(byte[] data) throws IOException {
73+
ByteArrayOutputStream buf = new ByteArrayOutputStream();
74+
try (GZIPOutputStream out = new GZIPOutputStream(buf)) {
75+
out.write(data);
76+
}
77+
return buf.toByteArray();
78+
}
79+
80+
public static byte[] inflate(InputStream in) throws IOException {
81+
return new GZIPInputStream(in).readAllBytes();
82+
}
83+
84+
public static class ZeroAvailableStream extends FilterInputStream {
85+
public ZeroAvailableStream(InputStream in) {
86+
super(in);
87+
}
88+
@Override
89+
public int available() {
90+
return 0;
91+
}
92+
}
93+
}

0 commit comments

Comments
 (0)
Please sign in to comment.