// 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.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 static org.junit.Assert.assertEquals;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.zip.ZipInputStream;
/**
* 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;
assertEquals("not based at current position", expMark, view.get(LOCSIG));
assertEquals("Not slice with position 0", expPos, view.buffer.position());
assertEquals("Not sized with comment", expSize, view.getSize());
assertEquals("Not limited to size", expSize, view.buffer.limit());
}
@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);
assertEquals("Incorrect filename", filename, view.getFilename());
Assert.assertArrayEquals("Incorrect extra data", extraData, view.getExtraData());
assertEquals("Not at position 0", expPos, view.buffer.position());
assertEquals("Not sized correctly", expSize, view.getSize());
assertEquals("Not limited to size", expSize, view.buffer.limit());
}
@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);
assertEquals("not based at current position", expMark, view.get(LOCSIG));
assertEquals("Not slice with position 0", expPos, view.buffer.position());
assertEquals("Not sized with filename", expSize, view.getSize());
assertEquals("Not limited to size", expSize, view.buffer.limit());
assertEquals("Incorrect filename", filename, view.getFilename());
}
@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();
assertEquals("buffer not advanced as expected", expSize, buffer.position());
buffer.position(0);
LocalFileHeader clone = LocalFileHeader.viewOf(buffer);
assertEquals("Fail to copy mark", view.get(LOCSIG), view.get(LOCSIG));
assertEquals("Fail to copy comment", view.getFilename(), clone.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);
assertEquals("CRC", crc, view.get(LOCCRC));
assertEquals("Compressed size", compressed, view.get(LOCSIZ));
assertEquals("Uncompressed size", uncompressed, view.get(LOCLEN));
assertEquals("Flags", flags, view.get(LOCFLG));
assertEquals("Method", method, view.get(LOCHOW));
assertEquals("Modified time", time, view.get(LOCTIM));
assertEquals("Version needed", version, view.get(LOCVER));
}
}