diff options
Diffstat (limited to 'src/test/java/org/apache/commons/compress/archivers/zip/X7875_NewUnixTest.java')
-rw-r--r-- | src/test/java/org/apache/commons/compress/archivers/zip/X7875_NewUnixTest.java | 247 |
1 files changed, 247 insertions, 0 deletions
diff --git a/src/test/java/org/apache/commons/compress/archivers/zip/X7875_NewUnixTest.java b/src/test/java/org/apache/commons/compress/archivers/zip/X7875_NewUnixTest.java new file mode 100644 index 000000000..b1488091d --- /dev/null +++ b/src/test/java/org/apache/commons/compress/archivers/zip/X7875_NewUnixTest.java @@ -0,0 +1,247 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +package org.apache.commons.compress.archivers.zip; + +import org.junit.Before; +import org.junit.Test; + +import java.io.File; +import java.util.Arrays; +import java.util.Enumeration; +import java.util.zip.ZipException; + +import static org.apache.commons.compress.AbstractTestCase.getFile; +import static org.junit.Assert.assertArrayEquals; +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +public class X7875_NewUnixTest { + + private final static ZipShort X7875 = new ZipShort(0x7875); + + private X7875_NewUnix xf; + + @Before + public void before() { + xf = new X7875_NewUnix(); + } + + + @Test + public void testSampleFile() throws Exception { + final File archive = getFile("COMPRESS-211_uid_gid_zip_test.zip"); + ZipFile zf = null; + + try { + zf = new ZipFile(archive); + final Enumeration<ZipArchiveEntry> en = zf.getEntries(); + + // We expect EVERY entry of this zip file (dir & file) to + // contain extra field 0x7875. + while (en.hasMoreElements()) { + + final ZipArchiveEntry zae = en.nextElement(); + final String name = zae.getName(); + final X7875_NewUnix xf = (X7875_NewUnix) zae.getExtraField(X7875); + + // The directory entry in the test zip file is uid/gid 1000. + long expected = 1000; + if (name.contains("uid555_gid555")) { + expected = 555; + } else if (name.contains("uid5555_gid5555")) { + expected = 5555; + } else if (name.contains("uid55555_gid55555")) { + expected = 55555; + } else if (name.contains("uid555555_gid555555")) { + expected = 555555; + } else if (name.contains("min_unix")) { + expected = 0; + } else if (name.contains("max_unix")) { + // 2^32-2 was the biggest UID/GID I could create on my linux! + // (December 2012, linux kernel 3.4) + expected = 0x100000000L - 2; + } + assertEquals(expected, xf.getUID()); + assertEquals(expected, xf.getGID()); + } + } finally { + if (zf != null) { + zf.close(); + } + } + } + + @Test + public void testGetHeaderId() { + assertEquals(X7875, xf.getHeaderId()); + } + + @Test + public void testMisc() throws Exception { + assertFalse(xf.equals(new Object())); + assertTrue(xf.toString().startsWith("0x7875 Zip Extra Field")); + final Object o = xf.clone(); + assertEquals(o.hashCode(), xf.hashCode()); + assertTrue(xf.equals(o)); + xf.setUID(12345); + assertFalse(xf.equals(o)); + } + + @Test + public void testTrimLeadingZeroesForceMinLength4() { + final byte[] NULL = null; + final byte[] EMPTY = new byte[0]; + final byte[] ONE_ZERO = {0}; + final byte[] TWO_ZEROES = {0, 0}; + final byte[] FOUR_ZEROES = {0, 0, 0, 0}; + final byte[] SEQUENCE = {1, 2, 3}; + final byte[] SEQUENCE_LEADING_ZERO = {0, 1, 2, 3}; + final byte[] SEQUENCE_LEADING_ZEROES = {0, 0, 0, 0, 0, 0, 0, 1, 2, 3}; + final byte[] TRAILING_ZERO = {1, 2, 3, 0}; + final byte[] PADDING_ZERO = {0, 1, 2, 3, 0}; + final byte[] SEQUENCE6 = {1, 2, 3, 4, 5, 6}; + final byte[] SEQUENCE6_LEADING_ZERO = {0, 1, 2, 3, 4, 5, 6}; + + assertTrue(NULL == trimTest(NULL)); + assertTrue(Arrays.equals(ONE_ZERO, trimTest(EMPTY))); + assertTrue(Arrays.equals(ONE_ZERO, trimTest(ONE_ZERO))); + assertTrue(Arrays.equals(ONE_ZERO, trimTest(TWO_ZEROES))); + assertTrue(Arrays.equals(ONE_ZERO, trimTest(FOUR_ZEROES))); + assertTrue(Arrays.equals(SEQUENCE, trimTest(SEQUENCE))); + assertTrue(Arrays.equals(SEQUENCE, trimTest(SEQUENCE_LEADING_ZERO))); + assertTrue(Arrays.equals(SEQUENCE, trimTest(SEQUENCE_LEADING_ZEROES))); + assertTrue(Arrays.equals(TRAILING_ZERO, trimTest(TRAILING_ZERO))); + assertTrue(Arrays.equals(TRAILING_ZERO, trimTest(PADDING_ZERO))); + assertTrue(Arrays.equals(SEQUENCE6, trimTest(SEQUENCE6))); + assertTrue(Arrays.equals(SEQUENCE6, trimTest(SEQUENCE6_LEADING_ZERO))); + } + + private static byte[] trimTest(final byte[] b) { return X7875_NewUnix.trimLeadingZeroesForceMinLength(b); } + + @Test + public void testParseReparse() throws ZipException { + + // Version=1, Len=0, Len=0. + final byte[] ZERO_LEN = {1, 0, 0}; + + // Version=1, Len=1, zero, Len=1, zero. + final byte[] ZERO_UID_GID = {1, 1, 0, 1, 0}; + + // Version=1, Len=1, one, Len=1, one + final byte[] ONE_UID_GID = {1, 1, 1, 1, 1}; + + // Version=1, Len=2, one thousand, Len=2, one thousand + final byte[] ONE_THOUSAND_UID_GID = {1, 2, -24, 3, 2, -24, 3}; + + // (2^32 - 2). I guess they avoid (2^32 - 1) since it's identical to -1 in + // two's complement, and -1 often has a special meaning. + final byte[] UNIX_MAX_UID_GID = {1, 4, -2, -1, -1, -1, 4, -2, -1, -1, -1}; + + // Version=1, Len=5, 2^32, Len=5, 2^32 + 1 + // Esoteric test: can we handle 40 bit numbers? + final byte[] LENGTH_5 = {1, 5, 0, 0, 0, 0, 1, 5, 1, 0, 0, 0, 1}; + + // Version=1, Len=8, 2^63 - 2, Len=8, 2^63 - 1 + // Esoteric test: can we handle 64 bit numbers? + final byte[] LENGTH_8 = {1, 8, -2, -1, -1, -1, -1, -1, -1, 127, 8, -1, -1, -1, -1, -1, -1, -1, 127}; + + final long TWO_TO_32 = 0x100000000L; + final long MAX = TWO_TO_32 - 2; + + parseReparse(0, 0, ZERO_LEN, 0, 0); + parseReparse(0, 0, ZERO_UID_GID, 0, 0); + parseReparse(1, 1, ONE_UID_GID, 1, 1); + parseReparse(1000, 1000, ONE_THOUSAND_UID_GID, 1000, 1000); + parseReparse(MAX, MAX, UNIX_MAX_UID_GID, MAX, MAX); + parseReparse(-2, -2, UNIX_MAX_UID_GID, MAX, MAX); + parseReparse(TWO_TO_32, TWO_TO_32 + 1, LENGTH_5, TWO_TO_32, TWO_TO_32 + 1); + parseReparse(Long.MAX_VALUE - 1, Long.MAX_VALUE, LENGTH_8, Long.MAX_VALUE - 1, Long.MAX_VALUE); + + // We never emit this, but we should be able to parse it: + final byte[] SPURIOUS_ZEROES_1 = {1, 4, -1, 0, 0, 0, 4, -128, 0, 0, 0}; + final byte[] EXPECTED_1 = {1, 1, -1, 1, -128}; + xf.parseFromLocalFileData(SPURIOUS_ZEROES_1, 0, SPURIOUS_ZEROES_1.length); + + assertEquals(255, xf.getUID()); + assertEquals(128, xf.getGID()); + assertTrue(Arrays.equals(EXPECTED_1, xf.getLocalFileDataData())); + + final byte[] SPURIOUS_ZEROES_2 = {1, 4, -1, -1, 0, 0, 4, 1, 2, 0, 0}; + final byte[] EXPECTED_2 = {1, 2, -1, -1, 2, 1, 2}; + xf.parseFromLocalFileData(SPURIOUS_ZEROES_2, 0, SPURIOUS_ZEROES_2.length); + + assertEquals(65535, xf.getUID()); + assertEquals(513, xf.getGID()); + assertTrue(Arrays.equals(EXPECTED_2, xf.getLocalFileDataData())); + } + + + private void parseReparse( + final long uid, + final long gid, + final byte[] expected, + final long expectedUID, + final long expectedGID + ) throws ZipException { + + // Initial local parse (init with garbage to avoid defaults causing test to pass). + xf.setUID(54321); + xf.setGID(12345); + xf.parseFromLocalFileData(expected, 0, expected.length); + assertEquals(expectedUID, xf.getUID()); + assertEquals(expectedGID, xf.getGID()); + + xf.setUID(uid); + xf.setGID(gid); + if (expected.length < 5) { + // We never emit zero-length entries. + assertEquals(5, xf.getLocalFileDataLength().getValue()); + } else { + assertEquals(expected.length, xf.getLocalFileDataLength().getValue()); + } + byte[] result = xf.getLocalFileDataData(); + if (expected.length < 5) { + // We never emit zero-length entries. + assertTrue(Arrays.equals(new byte[]{1,1,0,1,0}, result)); + } else { + assertTrue(Arrays.equals(expected, result)); + } + + + + // And now we re-parse: + xf.parseFromLocalFileData(result, 0, result.length); + + // Did uid/gid change from re-parse? They shouldn't! + assertEquals(expectedUID, xf.getUID()); + assertEquals(expectedGID, xf.getGID()); + + assertEquals(0, xf.getCentralDirectoryLength().getValue()); + result = xf.getCentralDirectoryData(); + assertArrayEquals(new byte[0], result); + + // And now we re-parse: + xf.parseFromCentralDirectoryData(result, 0, result.length); + + // Did uid/gid change from 2nd re-parse? They shouldn't! + assertEquals(expectedUID, xf.getUID()); + assertEquals(expectedGID, xf.getGID()); + } +} |