Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: openjdk/jdk
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: 5671f836
Choose a base ref
...
head repository: openjdk/jdk
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: fc2b57e6
Choose a head ref

Commits on Mar 19, 2024

  1. Copy the full SHA
    53a1d52 View commit details
  2. Javadoc wording tweaks.

    archiecobbs committed Mar 19, 2024
    Copy the full SHA
    04d0c13 View commit details

Commits on Mar 21, 2024

  1. Copy the full SHA
    a6f5535 View commit details
  2. Copy the full SHA
    76a4d84 View commit details

Commits on Apr 2, 2024

  1. Copy the full SHA
    96049db View commit details
  2. Javadoc wording tweaks.

    archiecobbs committed Apr 2, 2024
    Copy the full SHA
    3be0b71 View commit details

Commits on Apr 12, 2024

  1. Copy the full SHA
    4493866 View commit details
  2. Copy the full SHA
    48b3804 View commit details
  3. Copy the full SHA
    32ff1ab View commit details

Commits on Apr 15, 2024

  1. Copy the full SHA
    df302a6 View commit details

Commits on Jun 6, 2024

  1. Copy the full SHA
    edba950 View commit details
  2. Bump @SInCE from 23 to 24.

    archiecobbs committed Jun 6, 2024
    Copy the full SHA
    f845a75 View commit details

Commits on Jul 22, 2024

  1. Copy the full SHA
    72c6229 View commit details
  2. Copy the full SHA
    bb78bd0 View commit details

Commits on Jul 24, 2024

  1. Wording tweak.

    archiecobbs committed Jul 24, 2024
    Copy the full SHA
    cbdefcc View commit details
  2. Copy the full SHA
    72b6009 View commit details

Commits on Jul 27, 2024

  1. Copy the full SHA
    55db236 View commit details
  2. Copy the full SHA
    f3939c0 View commit details

Commits on Aug 11, 2024

  1. Copy the full SHA
    a69bf2a View commit details

Commits on Aug 28, 2024

  1. Copy the full SHA
    34f2a21 View commit details

Commits on Aug 30, 2024

  1. Copy the full SHA
    2b0e103 View commit details
  2. Copy the full SHA
    58fc9b2 View commit details
  3. Copy the full SHA
    6b65c21 View commit details
  4. Copy the full SHA
    3a76c02 View commit details
  5. Copy the full SHA
    d577f6e View commit details
  6. Adjusting line breaks

    eirbjo committed Aug 30, 2024
    Copy the full SHA
    fc2b57e View commit details
Original file line number Diff line number Diff line change
@@ -37,6 +37,14 @@
* This class implements a stream filter for reading compressed data in
* the GZIP file format.
*
* <p>
* The InputStream passed to the constructor of this class may represent a
* single GZIP file or multiple consecutive GZIP files. When the end of a
* GZIP file is immediately followed by a new GZIP file, this class continues
* to decode compressed data into a single, concatenated stream of uncompressed
* data. Otherwise, any additional trailing bytes following a GZIP file are
* discarded as if the end of stream is reached.
*
* @see InflaterInputStream
* @author David Connelly
* @since 1.1
159 changes: 159 additions & 0 deletions test/jdk/java/util/zip/GZIP/GZIPInputStreamConcat.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,159 @@
/*
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/

/* @test
* @bug 8322256
* @summary Test support for concatenated GZIP streams
* @run junit GZIPInputStreamConcat
*/

import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;

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

import static org.junit.jupiter.api.Assertions.assertArrayEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class GZIPInputStreamConcat {

// The buffer size passed to GZIPInputStream
private int bufSize;

public static Stream<Arguments> testScenarios() throws IOException {
// Test concat vs. non-concat, garbage vs. no-garbage, and various buffer sizes on random data
Random random = new Random();
List<Arguments> scenarios = new ArrayList<>();
for (int bufSize = 1; bufSize < 1024; bufSize += random.nextInt(32) + 1) {
scenarios.add(Arguments.of(randomData(0, 100), bufSize));
}
return scenarios.stream();
}

@ParameterizedTest
@MethodSource("testScenarios")
public void testScenario(byte[] uncompressed, int bufSize) throws IOException {
this.bufSize = bufSize;
runTests(uncompressed);
}

public void runTests(byte[] uncompressed) throws IOException {

// Compress the test data
byte[] compressed = gzip(uncompressed);

// Decompress a single stream with no extra garbage - should always work
decomp(compressed, uncompressed);

// Decompress a truncated GZIP header
decompFail(oneByteShort(gzipHeader()), EOFException.class);

// Decompress a single stream that is one byte short - should always fail
decompFail(oneByteShort(compressed), EOFException.class);

// Decompress a single stream with one byte of extra garbage (trying all 256 possible values)
for (int extra = Byte.MIN_VALUE; extra <= Byte.MAX_VALUE ; extra++) {
decomp(oneByteLong(compressed, (byte) extra), uncompressed);
}

// Decompress a single stream followed by a truncated GZIP header
decomp(concat(compressed, oneByteShort(gzipHeader())), uncompressed);

// Decompress a single stream followed by another stream that is one byte short
decompFail(concat(compressed, oneByteShort(compressed)), IOException.class);

// Decompress two streams concatenated
decomp(concat(compressed, compressed), concat(uncompressed, uncompressed));

// Decompress three streams concatenated
decomp(concat(compressed, compressed, compressed),
concat(uncompressed, uncompressed, uncompressed));

// Decompress three streams concatenated followed by a truncated GZIP header
decomp(concat(compressed, compressed, compressed, oneByteShort(gzipHeader())),
concat(uncompressed, uncompressed, uncompressed));
}

// Do decompression and check result
public void decomp(byte[] compressed, byte[] uncompressed) throws IOException {
byte[] readback = gunzip(new ByteArrayInputStream(compressed));
assertArrayEquals(uncompressed, readback);
}

// Do decompression, asserting an execption
public void decompFail(byte[] compressed, Class<? extends IOException> exceptionType) {
assertThrows(exceptionType, () -> {
byte[] readback = gunzip(new ByteArrayInputStream(compressed));
});
}

// Create a GZIP header
public static byte[] gzipHeader() throws IOException {
byte[] compressed = gzip(new byte[0]);
return Arrays.copyOfRange(compressed, 0, 10);
}

// Add one extra byte to the given array
public static byte[] oneByteLong(byte[] array, byte value) {
return concat(array, new byte[] {value});
}

// Chop off the last byte of the given array
public static byte[] oneByteShort(byte[] array) {
return Arrays.copyOfRange(array, 0, array.length - 1);
}

// Create some random data
public static byte[] randomData(int min, int max) {
Random random = new Random();
byte[] data = new byte[min + random.nextInt(max - min)];
random.nextBytes(data);
return data;
}

// Concatenate byte arrays
public static byte[] concat(byte[]... arrays) {
final ByteArrayOutputStream buf = new ByteArrayOutputStream();
for (byte[] array : arrays)
buf.writeBytes(array);
return buf.toByteArray();
}

// GZIP compress data
public static byte[] gzip(byte[] data) throws IOException {
ByteArrayOutputStream buf = new ByteArrayOutputStream();
try (GZIPOutputStream out = new GZIPOutputStream(buf)) {
out.write(data);
}
return buf.toByteArray();
}

// GZIP decompress data
public byte[] gunzip(InputStream in) throws IOException {
return new GZIPInputStream(in, bufSize).readAllBytes();
}
}
139 changes: 139 additions & 0 deletions test/jdk/java/util/zip/GZIP/GZIPInputStreamGzipCommand.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
/*
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/

/* @test
* @bug 8322256
* @summary Test decompression of streams created by the gzip(1) command
* @run junit GZIPInputStreamGzipCommand
*/

import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;

import java.io.*;
import java.nio.charset.StandardCharsets;
import java.util.*;
import java.util.stream.*;
import java.util.zip.*;

import static org.junit.jupiter.api.Assertions.assertArrayEquals;

public class GZIPInputStreamGzipCommand {

public static Stream<Arguments> gzipScenarios() throws IOException {
List<Arguments> scenarios = new ArrayList();

/*
# To regenerate the hex data below:
(
for i in 1 2 3 4 5 6 7 8 9; do
printf 'this is compression level #%d\n' "$i" | gzip -"${i}"
done
) | hexdump -e '32/1 "%02x" "\n"'
*/
scenarios.add(Arguments.of("""
this is compression level #1
this is compression level #2
this is compression level #3
this is compression level #4
this is compression level #5
this is compression level #6
this is compression level #7
this is compression level #8
this is compression level #9
""", """
1f8b0800d42ea46604032bc9c82c5600a2e4fcdc82a2d4e2e2ccfc3c859cd4b2
d41c0565432e0092bb84691d0000001f8b0800d42ea46600032bc9c82c5600a2
e4fcdc82a2d4e2e2ccfc3c859cd4b2d41c0565232e0051e8a9421d0000001f8b
0800d42ea46600032bc9c82c5600a2e4fcdc82a2d4e2e2ccfc3c859cd4b2d41c
0565632e0010d9b25b1d0000001f8b0800d42ea46600032bc9c82c5600a2e4fc
dc82a2d4e2e2ccfc3c859cd4b2d41c0565132e00d74ff3141d0000001f8b0800
d42ea46600032bc9c82c5600a2e4fcdc82a2d4e2e2ccfc3c859cd4b2d41c0565
532e00967ee80d1d0000001f8b0800d42ea46600032bc9c82c5600a2e4fcdc82
a2d4e2e2ccfc3c859cd4b2d41c0565332e00552dc5261d0000001f8b0800d42e
a46600032bc9c82c5600a2e4fcdc82a2d4e2e2ccfc3c859cd4b2d41c0565732e
00141cde3f1d0000001f8b0800d42ea46600032bc9c82c5600a2e4fcdc82a2d4
e2e2ccfc3c859cd4b2d41c05650b2e00db0046b81d0000001f8b0800d42ea466
02032bc9c82c5600a2e4fcdc82a2d4e2e2ccfc3c859cd4b2d41c05654b2e009a
315da11d000000"""));

/*
# To regenerate the hex data below:
(
printf 'this one has a name\n' > file1 && gzip file1
printf 'this one has no name\n' > file2 && gzip --no-name file2
cat file1.gz file2.gz
rm file1.gz file2.gz
) | hexdump -e '32/1 "%02x" "\n"'
*/
scenarios.add(Arguments.of("""
this one has a name
this one has no name
""", """
1f8b08082230a466000366696c6531002bc9c82c56c8cf4b55c8482c564854c8
4bcc4de50200d7ccdc5a140000001f8b08000000000000032bc9c82c56c8cf4b
55c8482c56c8cb57c84bcc4de50200b1effb5015000000""" ));

/*
# To regenerate the hex data below:
(
i=0
while [ "${i}" -lt 1000 ]; do
printf 'aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'
i=`expr "${i}" + 1`
done
) | gzip --best | hexdump -e '32/1 "%02x" "\n"'
*/
scenarios.add(Arguments.of(
"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa".repeat(1000),
"""
1f8b08002a35a4660203edc18100000000c320d6f94b1ce45501000000000000
0000000000000000000000000000000000000000000000000000000000000000
c08f01492d182728a00000""" ));

return scenarios.stream();
}

@ParameterizedTest
@MethodSource("gzipScenarios")
public void testScenario(String input, String hexData) throws IOException {

// Get expected result
byte[] expected = input.getBytes(StandardCharsets.UTF_8);

// Get actual result
HexFormat hexFormat = HexFormat.of();
byte[] data = hexFormat.parseHex(hexData.replaceAll("\\s", ""));
ByteArrayOutputStream buf = new ByteArrayOutputStream();
try (GZIPInputStream gunzip = new GZIPInputStream(new ByteArrayInputStream(data))) {
gunzip.transferTo(buf);
}
byte[] actual = buf.toByteArray();

// Compare
System.out.println(" ACTUAL: " + hexFormat.formatHex(actual));
System.out.println("EXPECTED: " + hexFormat.formatHex(expected));
assertArrayEquals(actual, expected);
}
}