Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

7036144: GZIPInputStream readTrailer uses faulty available() test for end-of-stream #17113

Closed
wants to merge 10 commits into from
61 changes: 34 additions & 27 deletions test/jdk/java/util/zip/GZIP/GZIPInputStreamAvailable.java
Original file line number Diff line number Diff line change
@@ -28,54 +28,61 @@
*/

import org.junit.Test;
import org.junit.Assert;

import java.io.*;
import java.util.*;
import java.util.zip.*;

import static org.junit.Assert.assertArrayEquals;

public class GZIPInputStreamAvailable {

public static final int NUM_COPIES = 100;

@Test
public void testZeroAvailable() throws IOException {

// Create some compressed data
ByteArrayOutputStream buf = new ByteArrayOutputStream();
try (GZIPOutputStream out = new GZIPOutputStream(buf)) {
out.write("boo".getBytes("ASCII"));
}
byte[] gz = buf.toByteArray();
// Create some uncompressed data and then repeat it NUM_COPIES times
byte[] uncompressed1 = "this is a test".getBytes("ASCII");
byte[] uncompressedN = repeat(uncompressed1, NUM_COPIES);

// Repeat to build a sequence of concatenated compressed streams
buf.reset();
for(int i = 0; i < 100; i++)
buf.write(gz);
final byte[] gz32 = buf.toByteArray();
// Compress the original data and then repeat that NUM_COPIES times
byte[] compressed1 = gzip(uncompressed1);
byte[] compressedN = repeat(compressed1, NUM_COPIES);

// (a) Read it back from a stream where available() is accurate
long count1 = countBytes(new GZIPInputStream(new ByteArrayInputStream(gz32)));
// (a) Read back copied compressed data from a stream where available() is accurate and verify
byte[] readback1 = new GZIPInputStream(new ByteArrayInputStream(compressedN)).readAllBytes();
assertArrayEquals(uncompressedN, readback1);

// (b) Read it back from a stream where available() always returns zero
long count2 = countBytes(new GZIPInputStream(new ZeroAvailableInputStream(new ByteArrayInputStream(gz32))));
// (b) Read back copied compressed data from a stream where available() always returns zero and verify
byte[] readback2 = new GZIPInputStream(new ZeroAvailableStream(new ByteArrayInputStream(compressedN))).readAllBytes();
assertArrayEquals(uncompressedN, readback2);
}

// They should be the same
Assert.assertEquals(count2, count1);
public static byte[] repeat(byte[] data, int count) {
byte[] repeat = new byte[data.length * count];
int off = 0;
for (int i = 0; i < count; i++) {
System.arraycopy(data, 0, repeat, off, data.length);
off += data.length;
}
return repeat;
}

public long countBytes(InputStream in) throws IOException {
long count = 0;
while (in.read() != -1)
count++;
in.close();
return count;
public static byte[] gzip(byte[] data) throws IOException {
ByteArrayOutputStream buf = new ByteArrayOutputStream();
try (GZIPOutputStream out = new GZIPOutputStream(buf)) {
out.write(data);
}
return buf.toByteArray();
}

public static class ZeroAvailableInputStream extends FilterInputStream {
public ZeroAvailableInputStream(InputStream in) {
public static class ZeroAvailableStream extends FilterInputStream {
public ZeroAvailableStream(InputStream in) {
super(in);
}
@Override
public int available() throws IOException {
public int available() {
return 0;
}
}