mirror of
https://github.com/corda/corda.git
synced 2025-01-09 22:42:40 +00:00
094af1e794
We were decrementing the "remaining" field twice for each byte read using the no-arg read method, which resulted in available() returning a value that was too small.
390 lines
10 KiB
Java
390 lines
10 KiB
Java
/* Copyright (c) 2008-2013, Avian Contributors
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
for any purpose with or without fee is hereby granted, provided
|
|
that the above copyright notice and this permission notice appear
|
|
in all copies.
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
details. */
|
|
|
|
package java.util.zip;
|
|
|
|
import java.io.File;
|
|
import java.io.RandomAccessFile;
|
|
import java.io.InputStream;
|
|
import java.io.IOException;
|
|
import java.util.Enumeration;
|
|
import java.util.Iterator;
|
|
import java.util.Map;
|
|
import java.util.HashMap;
|
|
|
|
public class ZipFile {
|
|
private final RandomAccessFile file;
|
|
private final Window window;
|
|
private final Map<String,Integer> index = new HashMap();
|
|
|
|
public ZipFile(String name) throws IOException {
|
|
file = new RandomAccessFile(name, "r");
|
|
window = new Window(file, 4096);
|
|
|
|
int fileLength = (int) file.length();
|
|
int pointer = fileLength - 22;
|
|
byte[] magic = new byte[] { 0x50, 0x4B, 0x05, 0x06 };
|
|
while (pointer > 0) {
|
|
if (equal(window.data, window.seek(pointer, magic.length),
|
|
magic, 0, magic.length))
|
|
{
|
|
pointer = directoryOffset(window, pointer);
|
|
|
|
magic = new byte[] { 0x50, 0x4B, 0x01, 0x02 };
|
|
while (pointer < fileLength) {
|
|
if (equal(window.data, window.seek(pointer, magic.length),
|
|
magic, 0, magic.length))
|
|
{
|
|
index.put(entryName(window, pointer), pointer);
|
|
pointer = entryEnd(window, pointer);
|
|
} else {
|
|
pointer = fileLength;
|
|
}
|
|
}
|
|
pointer = 0;
|
|
} else {
|
|
-- pointer;
|
|
}
|
|
}
|
|
}
|
|
|
|
public ZipFile(File file) throws IOException {
|
|
this(file.getAbsolutePath());
|
|
}
|
|
|
|
public int size() {
|
|
return index.size();
|
|
}
|
|
|
|
protected Enumeration<? extends ZipEntry> makeEnumeration
|
|
(EntryFactory factory)
|
|
{
|
|
return new MyEnumeration(factory, window, index.values().iterator());
|
|
}
|
|
|
|
public Enumeration<? extends ZipEntry> entries() {
|
|
return makeEnumeration(ZipEntryFactory.Instance);
|
|
}
|
|
|
|
protected ZipEntry getEntry(EntryFactory factory, String name) {
|
|
while (name.startsWith("/")) {
|
|
name = name.substring(1);
|
|
}
|
|
Integer pointer = index.get(name);
|
|
return (pointer == null ? null : factory.makeEntry(window, pointer));
|
|
}
|
|
|
|
public ZipEntry getEntry(String name) {
|
|
return getEntry(ZipEntryFactory.Instance, name);
|
|
}
|
|
|
|
public InputStream getInputStream(ZipEntry entry) throws IOException {
|
|
final int pointer = ((MyEntry) entry).pointer();
|
|
int method = compressionMethod(window, pointer);
|
|
int size = compressedSize(window, pointer);
|
|
InputStream in = new MyInputStream(file, fileData(window, pointer), size);
|
|
|
|
final int Stored = 0;
|
|
final int Deflated = 8;
|
|
|
|
switch (method) {
|
|
case Stored:
|
|
return in;
|
|
|
|
case Deflated:
|
|
return new InflaterInputStream(in, new Inflater(true)) {
|
|
int remaining = uncompressedSize(window, pointer);
|
|
|
|
public int read() throws IOException {
|
|
byte[] buffer = new byte[1];
|
|
int c = read(buffer);
|
|
return (c < 0 ? c : (buffer[0] & 0xFF));
|
|
}
|
|
|
|
public int read(byte[] buffer) throws IOException {
|
|
return read(buffer, 0, buffer.length);
|
|
}
|
|
|
|
public int read(byte[] buffer, int offset, int length)
|
|
throws IOException
|
|
{
|
|
int c = super.read(buffer, offset, length);
|
|
if (c > 0) {
|
|
remaining -= c;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
public int available() {
|
|
return remaining;
|
|
}
|
|
};
|
|
|
|
default:
|
|
throw new IOException();
|
|
}
|
|
}
|
|
|
|
private static boolean equal(byte[] a, int aOffset, byte[] b, int bOffset,
|
|
int size)
|
|
{
|
|
for (int i = 0; i < size; ++i) {
|
|
if (a[aOffset + i] != b[bOffset + i]) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private static int get2(Window w, int p) throws IOException {
|
|
int offset = w.seek(p, 2);
|
|
return
|
|
((w.data[offset + 1] & 0xFF) << 8) |
|
|
((w.data[offset ] & 0xFF) );
|
|
}
|
|
|
|
private static int get4(Window w, int p) throws IOException {
|
|
int offset = w.seek(p, 4);
|
|
return
|
|
((w.data[offset + 3] & 0xFF) << 24) |
|
|
((w.data[offset + 2] & 0xFF) << 16) |
|
|
((w.data[offset + 1] & 0xFF) << 8) |
|
|
((w.data[offset ] & 0xFF) );
|
|
}
|
|
|
|
private static int directoryOffset(Window w, int p) throws IOException {
|
|
return get4(w, p + 16);
|
|
}
|
|
|
|
private static int entryNameLength(Window w, int p) throws IOException {
|
|
return get2(w, p + 28);
|
|
}
|
|
|
|
protected static String entryName(Window w, int p) throws IOException {
|
|
int length = entryNameLength(w, p);
|
|
return new String(w.data, w.seek(p + 46, length), length);
|
|
}
|
|
|
|
private static int compressionMethod(Window w, int p) throws IOException {
|
|
return get2(w, p + 10);
|
|
}
|
|
|
|
protected static int compressedSize(Window w, int p) throws IOException {
|
|
return get4(w, p + 20);
|
|
}
|
|
|
|
protected static int uncompressedSize(Window w, int p) throws IOException {
|
|
return get4(w, p + 24);
|
|
}
|
|
|
|
private static int fileNameLength(Window w, int p) throws IOException {
|
|
return get2(w, p + 28);
|
|
}
|
|
|
|
private static int extraFieldLength(Window w, int p) throws IOException {
|
|
return get2(w, p + 30);
|
|
}
|
|
|
|
private static int commentFieldLength(Window w, int p) throws IOException {
|
|
return get2(w, p + 32);
|
|
}
|
|
|
|
private static int entryEnd(Window w, int p) throws IOException {
|
|
final int HeaderSize = 46;
|
|
return p + HeaderSize
|
|
+ fileNameLength(w, p)
|
|
+ extraFieldLength(w, p)
|
|
+ commentFieldLength(w, p);
|
|
}
|
|
|
|
private static int fileData(Window w, int p) throws IOException {
|
|
int localHeader = localHeader(w, p);
|
|
final int LocalHeaderSize = 30;
|
|
return localHeader
|
|
+ LocalHeaderSize
|
|
+ localFileNameLength(w, localHeader)
|
|
+ localExtraFieldLength(w, localHeader);
|
|
}
|
|
|
|
private static int localHeader(Window w, int p) throws IOException {
|
|
return get4(w, p + 42);
|
|
}
|
|
|
|
private static int localFileNameLength(Window w, int p) throws IOException {
|
|
return get2(w, p + 26);
|
|
}
|
|
|
|
private static int localExtraFieldLength(Window w, int p)
|
|
throws IOException
|
|
{
|
|
return get2(w, p + 28);
|
|
}
|
|
|
|
public void close() throws IOException {
|
|
file.close();
|
|
}
|
|
|
|
protected static class Window {
|
|
private final RandomAccessFile file;
|
|
public final byte[] data;
|
|
public int start;
|
|
public int length;
|
|
|
|
public Window(RandomAccessFile file, int size) {
|
|
this.file = file;
|
|
data = new byte[size];
|
|
}
|
|
|
|
public int seek(int start, int length) throws IOException {
|
|
int fileLength = (int) file.length();
|
|
|
|
if (length > data.length) {
|
|
throw new IllegalArgumentException
|
|
("length " + length + " greater than buffer length " + data.length);
|
|
}
|
|
|
|
if (start < 0) {
|
|
throw new IllegalArgumentException("negative start " + start);
|
|
}
|
|
|
|
if (start + length > fileLength) {
|
|
throw new IllegalArgumentException
|
|
("end " + (start + length) + " greater than file length " +
|
|
fileLength);
|
|
}
|
|
|
|
if (start < this.start || start + length > this.start + this.length) {
|
|
this.length = Math.min(data.length, fileLength);
|
|
this.start = start - ((this.length - length) / 2);
|
|
if (this.start < 0) {
|
|
this.start = 0;
|
|
} else if (this.start + this.length > fileLength) {
|
|
this.start = fileLength - this.length;
|
|
}
|
|
file.seek(this.start);
|
|
file.readFully(data, 0, this.length);
|
|
}
|
|
|
|
return start - this.start;
|
|
}
|
|
}
|
|
|
|
protected interface MyEntry {
|
|
public int pointer();
|
|
}
|
|
|
|
private static class MyZipEntry extends ZipEntry implements MyEntry {
|
|
public final Window window;
|
|
public final int pointer;
|
|
|
|
public MyZipEntry(Window window, int pointer) {
|
|
super(null);
|
|
this.window = window;
|
|
this.pointer = pointer;
|
|
}
|
|
|
|
public String getName() {
|
|
try {
|
|
return entryName(window, pointer);
|
|
} catch (IOException e) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public long getCompressedSize() {
|
|
try {
|
|
return compressedSize(window, pointer);
|
|
} catch (IOException e) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public long getSize() {
|
|
try {
|
|
return uncompressedSize(window, pointer);
|
|
} catch (IOException e) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public int pointer() {
|
|
return pointer;
|
|
}
|
|
}
|
|
|
|
protected interface EntryFactory {
|
|
public ZipEntry makeEntry(Window window, int pointer);
|
|
}
|
|
|
|
private static class ZipEntryFactory implements EntryFactory {
|
|
public static final ZipEntryFactory Instance = new ZipEntryFactory();
|
|
|
|
public ZipEntry makeEntry(Window window, int pointer) {
|
|
return new MyZipEntry(window, pointer);
|
|
}
|
|
}
|
|
|
|
private static class MyEnumeration implements Enumeration<ZipEntry> {
|
|
private final EntryFactory factory;
|
|
private final Window window;
|
|
private final Iterator<Integer> iterator;
|
|
|
|
public MyEnumeration(EntryFactory factory, Window window,
|
|
Iterator<Integer> iterator)
|
|
{
|
|
this.factory = factory;
|
|
this.window = window;
|
|
this.iterator = iterator;
|
|
}
|
|
|
|
public boolean hasMoreElements() {
|
|
return iterator.hasNext();
|
|
}
|
|
|
|
public ZipEntry nextElement() {
|
|
return factory.makeEntry(window, iterator.next());
|
|
}
|
|
}
|
|
|
|
private static class MyInputStream extends InputStream {
|
|
private RandomAccessFile file;
|
|
private int offset;
|
|
private int length;
|
|
|
|
public MyInputStream(RandomAccessFile file, int start, int length) {
|
|
this.file = file;
|
|
this.offset = start;
|
|
this.length = length;
|
|
}
|
|
|
|
public int read() throws IOException {
|
|
byte[] b = new byte[1];
|
|
int c = read(b);
|
|
return (c == -1 ? -1 : b[0] & 0xFF);
|
|
}
|
|
|
|
public int read(byte[] b, int offset, int length) throws IOException {
|
|
if (this.length == 0) return -1;
|
|
|
|
if (length > this.length) length = this.length;
|
|
|
|
file.seek(this.offset);
|
|
file.readFully(b, offset, length);
|
|
|
|
this.offset += length;
|
|
this.length -= length;
|
|
|
|
return length;
|
|
}
|
|
|
|
public void close() throws IOException {
|
|
file = null;
|
|
}
|
|
}
|
|
}
|