/* * Copyright (C) 2010 The Android Open Source Project * * 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 libcore.java.util.zip; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.EOFException; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.Arrays; import java.util.Random; import java.util.zip.GZIPInputStream; import java.util.zip.GZIPOutputStream; import junit.framework.TestCase; import libcore.io.IoUtils; import libcore.io.Streams; public final class GZIPInputStreamTest extends TestCase { private static final byte[] HELLO_WORLD_GZIPPED = new byte[] { 31, -117, 8, 0, 0, 0, 0, 0, 0, 0, -13, 72, -51, -55, -55, 87, 8, -49, 47, -54, 73, 1, 0, 86, -79, 23, 74, 11, 0, 0, 0 }; public void testShortMessage() throws IOException { assertEquals("Hello World", new String(gunzip(HELLO_WORLD_GZIPPED), "UTF-8")); } public void testLongMessage() throws IOException { byte[] data = new byte[1024 * 1024]; new Random().nextBytes(data); assertTrue(Arrays.equals(data, gunzip(GZIPOutputStreamTest.gzip(data)))); } /** http://b/3042574 GzipInputStream.skip() causing CRC failures */ public void testSkip() throws IOException { byte[] data = new byte[1024 * 1024]; byte[] gzipped = GZIPOutputStreamTest.gzip(data); GZIPInputStream in = new GZIPInputStream(new ByteArrayInputStream(gzipped)); long totalSkipped = 0; long count; do { count = in.skip(Long.MAX_VALUE); totalSkipped += count; } while (count > 0); assertEquals(data.length, totalSkipped); in.close(); } // https://code.google.com/p/android/issues/detail?id=63873 public void testMultipleMembers() throws Exception { final int length = HELLO_WORLD_GZIPPED.length; byte[] data = new byte[length * 2]; System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, 0, length); System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, length, length); assertEquals("Hello WorldHello World", new String(gunzip(data), "UTF-8")); } // https://code.google.com/p/android/issues/detail?id=63873 public void testTrailingNonGzipData() throws Exception { final int length = HELLO_WORLD_GZIPPED.length; // 50 bytes of 0s at the end of the first message. byte[] data = new byte[length + 50]; System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, 0, length); assertEquals("Hello World", new String(gunzip(data), "UTF-8")); } // https://code.google.com/p/android/issues/detail?id=63873 // // Differences from the RI: Tests show the RI ignores *some* types of partial // data but not others and this test case fails as a result. Our implementation // will throw if it sees the gzip magic sequence at the end of a member // but malformed / invalid data after. public void testTrailingHeaderAndPartialMember() throws Exception { final int length = HELLO_WORLD_GZIPPED.length; // Copy just the header from HELLO_WORLD_GZIPPED so that our input // stream becomes one complete member + a header member. byte[] data = new byte[length + 10]; System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, 0, length); System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, length, 10); try { gunzip(data); fail(); } catch (EOFException expected) { } // Copy just the header from HELLO_WORLD_GZIPPED so that our input // stream becomes one complete member + a header member. data = new byte[length + 18]; System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, 0, length); System.arraycopy(HELLO_WORLD_GZIPPED, 0, data, length, 18); try { gunzip(data); fail(); } catch (EOFException expected) { } } // https://code.google.com/p/android/issues/detail?id=66409 public void testMultipleMembersWithCustomBufferSize() throws Exception { final int[] memberSizes = new int[] { 1000, 2000 }; // We don't care what the exact contents of this file is, as long // as the file has multiple members, and that the (compressed) size of // the second member is larger than the size of the input buffer. // // There's no way to achieve this for a GZIPOutputStream so we generate // pseudo-random sequence of bytes and assert that they don't compress // well. final Random r = new Random(10); byte[] bytes = new byte[3000]; r.nextBytes(bytes); File f = File.createTempFile("GZIPInputStreamTest", ".gzip"); int offset = 0; for (int size : memberSizes) { GZIPOutputStream gzos = null; try { FileOutputStream fos = new FileOutputStream(f, true /* append */); gzos = new GZIPOutputStream(fos, size + 1); gzos.write(bytes, offset, size); offset += size; gzos.finish(); } finally { IoUtils.closeQuietly(gzos); } } assertTrue(f.length() > 2048); FileInputStream fis = new FileInputStream(f); GZIPInputStream gzip = null; try { gzip = new GZIPInputStream(fis, memberSizes[0]); byte[] unzipped = Streams.readFully(gzip); assertTrue(Arrays.equals(bytes, unzipped)); } finally { IoUtils.closeQuietly(gzip); } } public static byte[] gunzip(byte[] bytes) throws IOException { ByteArrayInputStream bis = new ByteArrayInputStream(bytes); InputStream in = new GZIPInputStream(bis); ByteArrayOutputStream out = new ByteArrayOutputStream(); byte[] buffer = new byte[1024]; int count; while ((count = in.read(buffer)) != -1) { out.write(buffer, 0, count); } byte[] outArray = out.toByteArray(); in.close(); return outArray; } }