Skip to content

Commit 63cb1f8

Browse files
author
Eirik Bjørsnøs
committedFeb 2, 2024
8321396: Retire test/jdk/java/util/zip/NoExtensionSignature.java
Reviewed-by: lancea
1 parent f613e13 commit 63cb1f8

File tree

3 files changed

+180
-42
lines changed

3 files changed

+180
-42
lines changed
 
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,180 @@
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+
/**
25+
* @test
26+
* @bug 8321396
27+
* @summary Verify that ZipInputStream ignores non-zero, incorrect 'crc',
28+
* 'compressed size' and 'uncompressed size' values when in streaming mode.
29+
* @run junit DataDescriptorIgnoreCrcAndSizeFields
30+
*/
31+
32+
import org.junit.jupiter.api.Test;
33+
34+
import java.io.ByteArrayInputStream;
35+
import java.io.ByteArrayOutputStream;
36+
import java.io.IOException;
37+
import java.io.OutputStream;
38+
import java.nio.ByteBuffer;
39+
import java.nio.ByteOrder;
40+
import java.nio.charset.StandardCharsets;
41+
import java.util.zip.*;
42+
43+
import static org.junit.jupiter.api.Assertions.*;
44+
45+
public class DataDescriptorIgnoreCrcAndSizeFields {
46+
47+
/**
48+
* Verify that ZipInputStream correctly ignores values from a LOC header's
49+
* 'crc', 'compressed size' and 'uncompressed size' fields, when in
50+
* streaming mode and these fields are incorrectly set to non-zero values.
51+
*/
52+
@Test
53+
public void shouldIgnoreCrcAndSizeValuesInStreamingMode() throws IOException {
54+
// ZIP with incorrect 'CRC', 'compressed size' and 'uncompressed size' values
55+
byte[] zip = zipWithIncorrectCrcAndSizeValuesInLocalHeader();
56+
57+
// ZipInputStream should ignore the incorrect field values
58+
try (ZipInputStream in = new ZipInputStream(new ByteArrayInputStream(zip))) {
59+
ZipEntry first = in.getNextEntry();
60+
assertNotNull(first, "Zip file is unexpectedly missing first entry");
61+
62+
// CRC, compressed size and size should be uninitialized at this point
63+
assertCrcAndSize(first, -1, -1, -1);
64+
// Check that name and contents is as expected
65+
assertNameAndContents("first", first, in);
66+
// At this point, ZipInputStream should have read correct values from the data descriptor
67+
assertCrcAndSize(first, crc32("first"), compressedSize("first"), uncompressedSize("first"));
68+
69+
// For extra caution, also read and validate the second entry
70+
ZipEntry second = in.getNextEntry();
71+
assertNotNull(second, "Zip file is unexpectedly missing second entry");
72+
73+
// CRC, compressed size and size should be uninitialized at this point
74+
assertCrcAndSize(second, -1, -1, -1);
75+
// Check that name and contents is as expected
76+
assertNameAndContents("second", second, in);
77+
// At this point, ZipInputStream should have read correct values from the data descriptor
78+
assertCrcAndSize(second, crc32("second"), compressedSize("second"), uncompressedSize("second"));
79+
}
80+
81+
}
82+
83+
/**
84+
* Assert that the given ZipEntry has the expected name and that
85+
* the expected content can be read from the ZipInputStream
86+
* @param expected the expected name and content
87+
* @param entry the entry to check the name of
88+
* @param in the ZipInputStream to check the entry content of
89+
* @throws IOException if an IO exception occurs
90+
*/
91+
private static void assertNameAndContents(String expected, ZipEntry entry, ZipInputStream in) throws IOException {
92+
assertEquals(expected, entry.getName());
93+
assertArrayEquals(expected.getBytes(StandardCharsets.UTF_8), in.readAllBytes());
94+
}
95+
96+
/**
97+
* Assert that a ZipEntry has the expected CRC-32, compressed size and uncompressed size values
98+
* @param entry the ZipEntry to validate
99+
* @param expectedCrc the expected CRC-32 value
100+
* @param expectedCompressedSize the exprected compressed size value
101+
* @param expectedSize the expected size value
102+
*/
103+
private static void assertCrcAndSize(ZipEntry entry, long expectedCrc, long expectedCompressedSize, long expectedSize) {
104+
assertEquals(expectedCrc, entry.getCrc());
105+
assertEquals(expectedCompressedSize, entry.getCompressedSize());
106+
assertEquals(expectedSize, entry.getSize());
107+
}
108+
109+
/**
110+
* Return the CRC-32 value for the given string encoded in UTF-8
111+
* @param content the string to produce a CRC-32 checksum for
112+
* @return the CRC-value of the encoded string
113+
*/
114+
private long crc32(String content) {
115+
CRC32 crc32 = new CRC32();
116+
crc32.update(content.getBytes(StandardCharsets.UTF_8));
117+
return crc32.getValue();
118+
}
119+
120+
/**
121+
* Return the length of the given content encoded in UTF-8
122+
* @param content the content to return the encoded length for
123+
* @return the uncompressed size of the encoded content
124+
*/
125+
private long uncompressedSize(String content) {
126+
return content.getBytes(StandardCharsets.UTF_8).length;
127+
}
128+
129+
/**
130+
* Returns the size of the given content, as if it was encoded in UTF-8 and then deflated
131+
* @param content the content to get the compressed size of
132+
* @return the compressed size of the content
133+
* @throws IOException if an IO exception occurs
134+
*/
135+
private long compressedSize(String content) throws IOException {
136+
ByteArrayOutputStream bao = new ByteArrayOutputStream();
137+
try (OutputStream o = new DeflaterOutputStream(bao, new Deflater(Deflater.DEFAULT_COMPRESSION, true))) {
138+
o.write(content.getBytes(StandardCharsets.UTF_8));
139+
}
140+
return bao.size();
141+
}
142+
143+
/**
144+
* When a ZIP entry is created in 'streaming' mode, the 'general purpose bit flag' 3
145+
* is set, and the fields crc-32, compressed size and uncompressed size are set to
146+
* zero in the local header.
147+
*
148+
* Certain legacy ZIP tools incorrectly set non-zero values for one or more of these
149+
* three fields when in streaming mode.
150+
*
151+
* This method creates a ZIP where the first entry has a local header where the
152+
* mentioned fields are set to a non-zero, incorrect values. The second entry
153+
* has the correct zero values for these fields.
154+
*/
155+
private static byte[] zipWithIncorrectCrcAndSizeValuesInLocalHeader() throws IOException {
156+
ByteArrayOutputStream out = new ByteArrayOutputStream();
157+
try (ZipOutputStream zo = new ZipOutputStream(out)) {
158+
// Write a first entry
159+
zo.putNextEntry(new ZipEntry("first"));
160+
zo.write("first".getBytes(StandardCharsets.UTF_8));
161+
// Add a second entry
162+
zo.putNextEntry(new ZipEntry("second"));
163+
zo.write("second".getBytes(StandardCharsets.UTF_8));
164+
}
165+
166+
// ZipOutputStream correctly produces local headers with zero crc and sizes values
167+
byte[] zip = out.toByteArray();
168+
169+
// Buffer for updating the local header values
170+
ByteBuffer buffer = ByteBuffer.wrap(zip).order(ByteOrder.LITTLE_ENDIAN);
171+
// Set the CRC-32 field to an incorrect value
172+
buffer.putShort(ZipEntry.LOCCRC, (short) 42);
173+
// Set the compressed size to an incorrect value
174+
buffer.putShort(ZipEntry.LOCSIZ, (short) 42);
175+
// Set the uncompressed size to an incorrect value
176+
buffer.putShort(ZipEntry.LOCLEN, (short) 42);
177+
178+
return zip;
179+
}
180+
}

‎test/jdk/java/util/zip/NoExtensionSignature.java

-42
This file was deleted.

‎test/jdk/java/util/zip/test.zip

-130 Bytes
Binary file not shown.

0 commit comments

Comments
 (0)
Please sign in to comment.