// Copyright 2015 The Bazel Authors. All rights reserved. // // Licensed 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 com.google.devtools.build.android.ziputils; import static com.google.common.truth.Truth.assertWithMessage; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCCRC; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCFLG; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCHOW; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCLEN; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCSIG; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCSIZ; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCTIM; import static com.google.devtools.build.android.ziputils.LocalFileHeader.LOCVER; import static java.nio.charset.StandardCharsets.UTF_8; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.zip.ZipInputStream; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; /** Unit tests for {@link LocalFileHeader}. */ @RunWith(JUnit4.class) public class LocalFileHeaderTest { @Test public void testViewOf() { ByteBuffer buffer = ByteBuffer.allocate(100).order(ByteOrder.LITTLE_ENDIAN); for (int i = 0; i < 100; i++) { buffer.put((byte) i); } int offset = 20; int filenameLength = 10; int extraLength = 25; int marker = LocalFileHeader.SIGNATURE; buffer.putShort(offset + ZipInputStream.LOCNAM, (short) filenameLength); // filename length buffer.putShort(offset + ZipInputStream.LOCEXT, (short) extraLength); // extra data length buffer.putInt(offset, marker); // need to zero filename length to have predictable size buffer.position(offset); LocalFileHeader view = LocalFileHeader.viewOf(buffer); int expMark = (int) ZipInputStream.LOCSIG; int expSize = ZipInputStream.LOCHDR + filenameLength + extraLength; // fixed + comment int expPos = 0; assertWithMessage("not based at current position").that(view.get(LOCSIG)).isEqualTo(expMark); assertWithMessage("Not slice with position 0").that(view.buffer.position()).isEqualTo(expPos); assertWithMessage("Not sized with comment").that(view.getSize()).isEqualTo(expSize); assertWithMessage("Not limited to size").that(view.buffer.limit()).isEqualTo(expSize); } @Test public void testView_String_byteArr() { String filename = "pkg/foo.class"; byte[] extraData = { 1, 2, 3, 4, 5, 6, 7, 8}; int expSize = ZipInputStream.LOCHDR + filename.getBytes(UTF_8).length + extraData.length; int expPos = 0; LocalFileHeader view = LocalFileHeader.allocate(filename, extraData); assertWithMessage("Incorrect filename").that(view.getFilename()).isEqualTo(filename); assertWithMessage("Incorrect extra data").that(view.getExtraData()).isEqualTo(extraData); assertWithMessage("Not at position 0").that(view.buffer.position()).isEqualTo(expPos); assertWithMessage("Not sized correctly").that(view.getSize()).isEqualTo(expSize); assertWithMessage("Not limited to size").that(view.buffer.limit()).isEqualTo(expSize); } @Test public void testView_3Args() { ByteBuffer buffer = ByteBuffer.allocate(100).order(ByteOrder.LITTLE_ENDIAN); for (int i = 0; i < 100; i++) { buffer.put((byte) i); } int offset = 20; buffer.position(offset); String filename = "pkg/foo.class"; int expMark = LocalFileHeader.SIGNATURE; int expSize = ZipInputStream.LOCHDR + filename.getBytes(UTF_8).length; int expPos = 0; LocalFileHeader view = LocalFileHeader.view(buffer, filename, null); assertWithMessage("not based at current position").that(view.get(LOCSIG)).isEqualTo(expMark); assertWithMessage("Not slice with position 0").that(view.buffer.position()).isEqualTo(expPos); assertWithMessage("Not sized with filename").that(view.getSize()).isEqualTo(expSize); assertWithMessage("Not limited to size").that(view.buffer.limit()).isEqualTo(expSize); assertWithMessage("Incorrect filename").that(view.getFilename()).isEqualTo(filename); } @Test public void testCopy() { ByteBuffer buffer = ByteBuffer.allocate(100).order(ByteOrder.LITTLE_ENDIAN); LocalFileHeader view = LocalFileHeader.allocate("pkg/foo.class", null); view.copy(buffer); int expSize = view.getSize(); assertWithMessage("buffer not advanced as expected").that(buffer.position()).isEqualTo(expSize); buffer.position(0); LocalFileHeader clone = LocalFileHeader.viewOf(buffer); assertWithMessage("Fail to copy mark").that(view.get(LOCSIG)).isEqualTo(view.get(LOCSIG)); assertWithMessage("Fail to copy comment") .that(clone.getFilename()) .isEqualTo(view.getFilename()); } @Test public void testWithAndGetMethods() { int crc = 0x12345678; int compressed = 0x357f1d5; int uncompressed = 0x74813159; short flags = 0x7a61; short method = 0x3b29; int time = 0x12c673e1; short version = 0x1234; LocalFileHeader view = LocalFileHeader.allocate("pkg/foo.class", null) .set(LOCCRC, crc) .set(LOCSIZ, compressed) .set(LOCLEN, uncompressed) .set(LOCFLG, flags) .set(LOCHOW, method) .set(LOCTIM, time) .set(LOCVER, version); assertWithMessage("CRC").that(view.get(LOCCRC)).isEqualTo(crc); assertWithMessage("Compressed size").that(view.get(LOCSIZ)).isEqualTo(compressed); assertWithMessage("Uncompressed size").that(view.get(LOCLEN)).isEqualTo(uncompressed); assertWithMessage("Flags").that(view.get(LOCFLG)).isEqualTo(flags); assertWithMessage("Method").that(view.get(LOCHOW)).isEqualTo(method); assertWithMessage("Modified time").that(view.get(LOCTIM)).isEqualTo(time); assertWithMessage("Version needed").that(view.get(LOCVER)).isEqualTo(version); } }