Skip to content

Commit 54a744b

Browse files
author
Lance Andersen
committedOct 21, 2024
8340553: ZipEntry field validation does not take into account the size of a CEN header
Reviewed-by: jpai, redestad, eirbjo
1 parent 18b55ce commit 54a744b

File tree

4 files changed

+236
-195
lines changed

4 files changed

+236
-195
lines changed
 

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

+39-13
Original file line numberDiff line numberDiff line change
@@ -91,20 +91,23 @@ public class ZipEntry implements ZipConstants, Cloneable {
9191
*/
9292
private static final long UPPER_DOSTIME_BOUND =
9393
128L * 365 * 24 * 60 * 60 * 1000;
94-
94+
// CEN header size + name length + comment length + extra length
95+
// should not exceed 65,535 bytes per the PKWare APP.NOTE
96+
// 4.4.10, 4.4.11, & 4.4.12.
97+
private static final int MAX_COMBINED_CEN_HEADER_SIZE = 0xFFFF;
9598
/**
9699
* Creates a new ZIP entry with the specified name.
97100
*
98101
* @param name
99102
* The entry name
100103
*
101104
* @throws NullPointerException if the entry name is null
102-
* @throws IllegalArgumentException if the entry name is longer than
103-
* 0xFFFF bytes
105+
* @throws IllegalArgumentException if the combined length of the entry name
106+
* and the {@linkplain #CENHDR CEN Header size} exceeds 65,535 bytes.
104107
*/
105108
public ZipEntry(String name) {
106109
Objects.requireNonNull(name, "name");
107-
if (name.length() > 0xFFFF) {
110+
if (!isCENHeaderValid(name, null, null)) {
108111
throw new IllegalArgumentException("entry name too long");
109112
}
110113
this.name = name;
@@ -519,8 +522,10 @@ public int getMethod() {
519522
* @param extra
520523
* The extra field data bytes
521524
*
522-
* @throws IllegalArgumentException if the length of the specified
523-
* extra field data is greater than 0xFFFF bytes
525+
* @throws IllegalArgumentException if the combined length of the specified
526+
* extra field data, the {@linkplain #getName() entry name},
527+
* the {@linkplain #getComment() entry comment}, and the
528+
* {@linkplain #CENHDR CEN Header size} exceeds 65,535 bytes.
524529
*
525530
* @see #getExtra()
526531
*/
@@ -541,7 +546,7 @@ public void setExtra(byte[] extra) {
541546
*/
542547
void setExtra0(byte[] extra, boolean doZIP64, boolean isLOC) {
543548
if (extra != null) {
544-
if (extra.length > 0xFFFF) {
549+
if (!isCENHeaderValid(name, extra, comment)) {
545550
throw new IllegalArgumentException("invalid extra field length");
546551
}
547552
// extra fields are in "HeaderID(2)DataSize(2)Data... format
@@ -642,16 +647,19 @@ public byte[] getExtra() {
642647

643648
/**
644649
* Sets the optional comment string for the entry.
645-
*
646-
* <p>ZIP entry comments have maximum length of 0xffff. If the length of the
647-
* specified comment string is greater than 0xFFFF bytes after encoding, only
648-
* the first 0xFFFF bytes are output to the ZIP file entry.
649-
*
650650
* @param comment the comment string
651-
*
651+
* @throws IllegalArgumentException if the combined length
652+
* of the specified entry comment, the {@linkplain #getName() entry name},
653+
* the {@linkplain #getExtra() extra field data}, and the
654+
* {@linkplain #CENHDR CEN Header size} exceeds 65,535 bytes.
652655
* @see #getComment()
653656
*/
654657
public void setComment(String comment) {
658+
if (comment != null) {
659+
if (!isCENHeaderValid(name, extra, comment)) {
660+
throw new IllegalArgumentException("entry comment too long");
661+
}
662+
}
655663
this.comment = comment;
656664
}
657665

@@ -702,4 +710,22 @@ public Object clone() {
702710
throw new InternalError(e);
703711
}
704712
}
713+
714+
/**
715+
* Initial validation that the CEN header size + name length + comment length
716+
* + extra length do not exceed 65,535 bytes per the PKWare APP.NOTE
717+
* 4.4.10, 4.4.11, & 4.4.12. Prior to writing out the CEN Header,
718+
* ZipOutputStream::writeCEN will do an additional validation of the combined
719+
* length of the fields after encoding the name and comment to a byte array.
720+
* @param name Zip entry name
721+
* @param extra Zip extra data
722+
* @param comment Zip entry comment
723+
* @return true if valid CEN Header size; false otherwise
724+
*/
725+
static boolean isCENHeaderValid(String name, byte[] extra, String comment) {
726+
int clen = comment == null ? 0 : comment.length();
727+
int elen = extra == null ? 0 : extra.length;
728+
long headerSize = (long)CENHDR + name.length() + clen + elen;
729+
return headerSize <= MAX_COMBINED_CEN_HEADER_SIZE;
730+
}
705731
}

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

+23-13
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
import java.util.Vector;
3333
import java.util.HashSet;
3434
import static java.util.zip.ZipConstants64.*;
35+
import static java.util.zip.ZipEntry.isCENHeaderValid;
3536
import static java.util.zip.ZipUtils.*;
3637
import sun.nio.cs.UTF_8;
3738
import sun.security.action.GetBooleanAction;
@@ -262,6 +263,12 @@ public void putNextEntry(ZipEntry e) throws IOException {
262263
}
263264
if (zc.isUTF8())
264265
e.flag |= USE_UTF8;
266+
// CEN header size + name length + comment length + extra length
267+
// should not exceed 65,535 bytes per the PKWare APP.NOTE
268+
// 4.4.10, 4.4.11, & 4.4.12.
269+
if (!isCENHeaderValid(e.name, e.extra, e.comment) ) {
270+
throw new ZipException("invalid CEN header (bad header size)");
271+
}
265272
current = new XEntry(e, written);
266273
xentries.add(current);
267274
writeLOC(current);
@@ -602,6 +609,22 @@ private void writeCEN(XEntry xentry) throws IOException {
602609
if (hasZip64) {
603610
elen += (elenZIP64 + 4);// + headid(2) + datasize(2)
604611
}
612+
613+
int clen = 0;
614+
byte[] commentBytes = null;
615+
if (e.comment != null) {
616+
commentBytes = zc.getBytes(e.comment);
617+
clen = commentBytes.length;
618+
}
619+
620+
// CEN header size + name length + comment length + extra length
621+
// should not exceed 65,535 bytes per the PKWare APP.NOTE
622+
// 4.4.10, 4.4.11, & 4.4.12.
623+
long headerSize = (long)CENHDR + nlen + clen + elen;
624+
if (headerSize > 0xFFFF ) {
625+
throw new ZipException("invalid CEN header (bad header size)");
626+
}
627+
605628
// cen info-zip extended timestamp only outputs mtime
606629
// but set the flag for a/ctime, if present in loc
607630
int flagEXTT = 0;
@@ -633,12 +656,6 @@ private void writeCEN(XEntry xentry) throws IOException {
633656
}
634657
}
635658
writeShort(elen);
636-
byte[] commentBytes = null;
637-
int clen = 0;
638-
if (e.comment != null) {
639-
commentBytes = zc.getBytes(e.comment);
640-
clen = Math.min(commentBytes.length, 0xffff);
641-
}
642659
writeShort(clen); // file comment length
643660
writeShort(0); // starting disk number
644661
writeShort(0); // internal file attributes (unused)
@@ -686,13 +703,6 @@ private void writeCEN(XEntry xentry) throws IOException {
686703
}
687704
}
688705

689-
// CEN header size + name length + comment length + extra length
690-
// should not exceed 65,535 bytes per the PKWare APP.NOTE
691-
// 4.4.10, 4.4.11, & 4.4.12.
692-
long headerSize = (long)CENHDR + nlen + clen + elen;
693-
if (headerSize > 0xFFFF ) {
694-
throw new ZipException("invalid CEN header (bad header size)");
695-
}
696706
writeExtra(e.extra);
697707
if (commentBytes != null) {
698708
writeBytes(commentBytes, 0, clen);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,174 @@
1+
/*
2+
* Copyright (c) 2024, 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 8340553
26+
* @summary Verify that ZipEntry(String), ZipEntry::setComment, and
27+
* ZipEntry::setExtra throws a IllegalArgumentException when the
28+
* combined length of the fields, including the size of the CEN Header,
29+
* exceeds 65,535 bytes
30+
* @run junit MaxZipEntryFieldSizeTest
31+
*/
32+
33+
import org.junit.jupiter.api.BeforeEach;
34+
import org.junit.jupiter.api.Test;
35+
import org.junit.jupiter.params.ParameterizedTest;
36+
import org.junit.jupiter.params.provider.ValueSource;
37+
38+
import java.io.IOException;
39+
import java.nio.ByteBuffer;
40+
import java.nio.ByteOrder;
41+
import java.nio.file.Files;
42+
import java.nio.file.Path;
43+
import java.util.zip.ZipEntry;
44+
import java.util.zip.ZipFile;
45+
46+
import static org.junit.jupiter.api.Assertions.assertThrows;
47+
48+
public class MaxZipEntryFieldSizeTest {
49+
50+
// CEN header size + name length + comment length + extra length
51+
// should not exceed 65,535 bytes per the PKWare APP.NOTE
52+
// 4.4.10, 4.4.11, & 4.4.12.
53+
static final int MAX_COMBINED_CEN_HEADER_SIZE = 0xFFFF;
54+
// Maximum possible size of name length + comment length + extra length
55+
// for entries in order to not exceed 65,489 bytes
56+
static final int MAX_NAME_COMMENT_EXTRA_SIZE =
57+
MAX_COMBINED_CEN_HEADER_SIZE - ZipFile.CENHDR;
58+
// Tag for the 'unknown' field type, specified in APPNOTE.txt 'Third party mappings'
59+
static final short UNKNOWN_ZIP_TAG = (short) 0x9902;
60+
// Zip Entry name used by tests
61+
static final String ENTRY_NAME = "EntryName";
62+
// Max length minus the size of the ENTRY_NAME or ENTRY_COMMENT
63+
static final int MAX_FIELD_LEN_MINUS_ENTRY_NAME =
64+
MAX_NAME_COMMENT_EXTRA_SIZE - 9;
65+
66+
/**
67+
* Validate an IllegalArgumentException is thrown when the
68+
* combined length of the entry name, entry comment, entry extra data,
69+
* and CEN Header size exceeds 65,535 bytes.
70+
*/
71+
@ParameterizedTest
72+
@ValueSource(ints = {30000, 35000})
73+
void combinedLengthTest(int length) {
74+
String comment = "a".repeat(length);
75+
byte[] bytes = creatExtraData(length);
76+
int combinedLength = ENTRY_NAME.length() + comment.length() + bytes.length;
77+
boolean expectException = combinedLength > MAX_COMBINED_CEN_HEADER_SIZE;
78+
System.out.printf("Combined Len= %s, exception: %s%n", combinedLength, expectException);
79+
ZipEntry zipEntry = new ZipEntry(ENTRY_NAME);
80+
zipEntry.setComment(comment);
81+
// The extra data length will trigger the IllegalArgumentException
82+
if (expectException) {
83+
assertThrows(IllegalArgumentException.class, () ->
84+
zipEntry.setExtra(bytes));
85+
} else {
86+
zipEntry.setExtra(bytes);
87+
}
88+
}
89+
90+
/**
91+
* Validate an IllegalArgumentException is thrown when the comment
92+
* length exceeds 65,489 bytes.
93+
*/
94+
@ParameterizedTest
95+
@ValueSource(ints = {MAX_COMBINED_CEN_HEADER_SIZE,
96+
MAX_NAME_COMMENT_EXTRA_SIZE,
97+
MAX_NAME_COMMENT_EXTRA_SIZE + 1,
98+
MAX_FIELD_LEN_MINUS_ENTRY_NAME,
99+
MAX_FIELD_LEN_MINUS_ENTRY_NAME - 1})
100+
void setCommentLengthTest(int length) {
101+
boolean expectException = length >= MAX_NAME_COMMENT_EXTRA_SIZE;
102+
ZipEntry zipEntry = new ZipEntry(ENTRY_NAME);
103+
String comment = "a".repeat(length);
104+
System.out.printf("Comment Len= %s, exception: %s%n", comment.length(), expectException);
105+
// The comment length will trigger the IllegalArgumentException
106+
if (expectException) {
107+
assertThrows(IllegalArgumentException.class, () ->
108+
zipEntry.setComment(comment));
109+
} else {
110+
zipEntry.setComment(comment);
111+
}
112+
}
113+
114+
/**
115+
* Validate an IllegalArgumentException is thrown when the name
116+
* length exceeds 65,489 bytes.
117+
*/
118+
@ParameterizedTest
119+
@ValueSource(ints = {MAX_COMBINED_CEN_HEADER_SIZE,
120+
MAX_NAME_COMMENT_EXTRA_SIZE,
121+
MAX_NAME_COMMENT_EXTRA_SIZE + 1,
122+
MAX_FIELD_LEN_MINUS_ENTRY_NAME,
123+
MAX_FIELD_LEN_MINUS_ENTRY_NAME - 1})
124+
void nameLengthTest(int length) {
125+
boolean expectException = length > MAX_NAME_COMMENT_EXTRA_SIZE;
126+
String name = "a".repeat(length);
127+
System.out.printf("name Len= %s, exception: %s%n", name.length(), expectException);
128+
// The name length will trigger the IllegalArgumentException
129+
if (expectException) {
130+
assertThrows(IllegalArgumentException.class, () -> new ZipEntry(name));
131+
} else {
132+
new ZipEntry(name);
133+
}
134+
}
135+
136+
/**
137+
* Validate an IllegalArgumentException is thrown when the extra data
138+
* length exceeds 65,489 bytes.
139+
*/
140+
@ParameterizedTest
141+
@ValueSource(ints = {MAX_COMBINED_CEN_HEADER_SIZE,
142+
MAX_NAME_COMMENT_EXTRA_SIZE,
143+
MAX_NAME_COMMENT_EXTRA_SIZE + 1,
144+
MAX_FIELD_LEN_MINUS_ENTRY_NAME,
145+
MAX_FIELD_LEN_MINUS_ENTRY_NAME - 1})
146+
void setExtraLengthTest(int length) {
147+
boolean expectException = length >= MAX_NAME_COMMENT_EXTRA_SIZE;
148+
byte[] bytes = creatExtraData(length);
149+
ZipEntry zipEntry = new ZipEntry(ENTRY_NAME);
150+
System.out.printf("extra Len= %s, exception: %s%n", bytes.length, expectException);
151+
// The extra data length will trigger the IllegalArgumentException
152+
if (expectException) {
153+
assertThrows(IllegalArgumentException.class, () -> zipEntry.setExtra(bytes));
154+
} else {
155+
zipEntry.setExtra(bytes);
156+
}
157+
}
158+
159+
/**
160+
* Create the extra field data which will be passed to ZipEntry::setExtra
161+
* @param length size of the extra data
162+
* @return byte array containing the extra data
163+
*/
164+
private static byte[] creatExtraData(int length) {
165+
byte[] bytes = new byte[length];
166+
// Little-endian ByteBuffer for updating the header fields
167+
ByteBuffer buffer = ByteBuffer.wrap(bytes).order(ByteOrder.LITTLE_ENDIAN);
168+
// We use the 'unknown' tag, specified in APPNOTE.TXT, 4.6.1 Third party mappings'
169+
buffer.putShort(UNKNOWN_ZIP_TAG);
170+
// Size of the actual (empty) data
171+
buffer.putShort((short) (length - 2 * Short.BYTES));
172+
return bytes;
173+
}
174+
}

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

-169
This file was deleted.

0 commit comments

Comments
 (0)
Please sign in to comment.