All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jboss.vfs.VirtualJarFileInputStream Maven / Gradle / Ivy

There is a newer version: 3.3.2.Final
Show newest version
/*
 * JBoss, Home of Professional Open Source
 * Copyright 2009, JBoss Inc., and individual contributors as indicated
 * by the @authors tag.
 *
 * 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 org.jboss.vfs;

import java.io.IOException;
import java.io.InputStream;
import java.util.LinkedList;
import java.util.List;
import java.util.jar.JarEntry;
import java.util.zip.CRC32;
import java.util.zip.ZipEntry;

/**
 * An input stream that can be used to wrap an VirtualJarInputStream (so any VFS dir)
 * and produce a byte stream following the Zip standard.
 *
 * @author John Bailey
 */
public class VirtualJarFileInputStream extends InputStream {
    // Needs to be sufficiently sized to allow local and central file headers with a single entry name
    private static final int MINIMUM_BUFFER_LENGTH = 1024;

    private final VirtualJarInputStream virtualJarInputStream;

    private State currentState = State.NOT_STARTED;

    private final List processedEntries = new LinkedList();
    private ProcessedEntry currentEntry;

    private final ByteBuffer buffer;
    private final CRC32 crc = new CRC32();

    private int currentCentralEntryIdx;
    private long centralOffset;
    private long totalRead;

    /**
     * Create with the minimum put length
     *
     * @param virtualJarInputStream The virtual jar input stream to base the stream off of
     */
    public VirtualJarFileInputStream(final VirtualJarInputStream virtualJarInputStream) {
        this(virtualJarInputStream, MINIMUM_BUFFER_LENGTH);
    }

    /**
     * Create with the a specified put size
     *
     * @param virtualJarInputStream The virtual jar input stream to base the stream off of
     * @param bufferLength          The length of put to use
     */
    public VirtualJarFileInputStream(final VirtualJarInputStream virtualJarInputStream, int bufferLength) {
        if (virtualJarInputStream == null) {
            throw VFSMessages.MESSAGES.nullArgument("virtualJarInputStream");
        }
        if (bufferLength < MINIMUM_BUFFER_LENGTH) {
            throw VFSMessages.MESSAGES.bufferMustBeLargerThan(MINIMUM_BUFFER_LENGTH);
        }

        this.virtualJarInputStream = virtualJarInputStream;

        buffer = new ByteBuffer(bufferLength);
    }


    /**
     * {@inheritDoc}
     */
    @Override
    public int read() throws IOException {
        int readByte = -1;
        while (currentState != null && (readByte = currentState.read(this)) == -1) {
            currentState = currentState.getNextState(this);
        }
        totalRead++;
        return readByte;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public void close() throws IOException {
        VFSUtils.safeClose(virtualJarInputStream);
        super.close();
    }

    /**
     * Close the current entry, and calculate the crc value.
     *
     * @throws IOException if any problems occur
     */
    private void closeCurrent() throws IOException {
        virtualJarInputStream.closeEntry();
        currentEntry.crc = crc.getValue();
        crc.reset();
    }

    /**
     * Buffer the content of the local file header for a single entry.
     *
     * @return true if the next local file header was buffered
     * @throws IOException if any problems occur
     */
    private boolean bufferLocalFileHeader() throws IOException {
        buffer.reset();
        JarEntry jarEntry = virtualJarInputStream.getNextJarEntry();

        if (jarEntry == null) { return false; }

        currentEntry = new ProcessedEntry(jarEntry, totalRead);
        processedEntries.add(currentEntry);


        bufferInt(ZipEntry.LOCSIG);    // Local file header signature
        bufferShort(10);               // Extraction version
        bufferShort(0);                // Flags
        bufferShort(ZipEntry.STORED);  // Compression type
        bufferInt(jarEntry.getTime()); // Entry time
        bufferInt(0);                  // CRC
        bufferInt(0);                  // Compressed size
        bufferInt(0);                  // Uncompressed size
        byte[] nameBytes = jarEntry.getName().getBytes("UTF8");
        bufferShort(nameBytes.length); // Entry name length
        bufferShort(0);                // Extra length
        buffer(nameBytes);
        return true;
    }

    /**
     * Buffer the central file header record for a single entry.
     *
     * @return true if the next central file header was buffered
     * @throws IOException if any problems occur
     */
    private boolean bufferNextCentralFileHeader() throws IOException {
        buffer.reset();

        if (currentCentralEntryIdx == processedEntries.size()) { return false; }

        ProcessedEntry entry = processedEntries.get(currentCentralEntryIdx++);

        JarEntry jarEntry = entry.jarEntry;
        bufferInt(ZipEntry.CENSIG);       // Central file header signature
        bufferShort(10);                  // Version made by
        bufferShort(10);                  // Extraction version
        bufferShort(0);                   // Flags
        bufferShort(ZipEntry.STORED);     // Compression type
        bufferInt(jarEntry.getTime());    // Entry time
        bufferInt(entry.crc);             // CRC
        bufferInt(jarEntry.getSize());    // Compressed size
        bufferInt(jarEntry.getSize());    // Uncompressed size
        byte[] nameBytes = jarEntry.getName().getBytes("UTF8");
        bufferShort(nameBytes.length);    // Entry name length
        bufferShort(0);                   // Extra field length
        bufferShort(0);                   // File comment length
        bufferShort(0);                   // Disk number start
        bufferShort(0);                   // Internal file attributes
        bufferInt(0);                     // External file attributes
        bufferInt(entry.offset);          // Relative offset of local header
        buffer(nameBytes);
        return true;
    }

    /**
     * Write the central file header records.  This is repeated
     * until all entries have been added to the central file header.
     *
     * @throws IOException if any problem occur
     */
    private void bufferCentralDirectoryEnd() throws IOException {
        buffer.reset();
        long lengthOfCentral = totalRead - centralOffset;

        int count = processedEntries.size();
        bufferInt(JarEntry.ENDSIG);      // End of central directory signature
        bufferShort(0);                  // Number of this disk
        bufferShort(0);                  // Start of central directory disk
        bufferShort(count);              // Number of processedEntries on disk
        bufferShort(count);              // Total number of processedEntries
        bufferInt(lengthOfCentral);      // Size of central directory
        bufferInt(centralOffset);        // Offset of start of central directory
        bufferShort(0);                  // Comment Length
    }

    /**
     * Buffer a 32-bit integer in little-endian
     *
     * @param i A long representation of a 32 bit int
     */
    private void bufferInt(long i) {
        buffer((byte) (i & 0xff));
        buffer((byte) ((i >>> 8) & 0xff));
        buffer((byte) ((i >>> 16) & 0xff));
        buffer((byte) ((i >>> 24) & 0xff));
    }

    /**
     * Buffer a 16-bit short in little-endian
     *
     * @param i An int representation of a 16 bit short
     */
    private void bufferShort(int i) {
        buffer((byte) (i & 0xff));
        buffer((byte) ((i >>> 8) & 0xff));
    }

    /**
     * Buffer a single byte
     *
     * @param b The byte
     */
    private void buffer(byte b) {
        if (buffer.hasCapacity()) {
            buffer.put(b);
        } else {
            throw VFSMessages.MESSAGES.bufferDoesntHaveEnoughCapacity();
        }
    }

    /**
     * Buffer a byte array
     *
     * @param bytes The bytes
     */
    private void buffer(byte[] bytes) {
        for (byte b : bytes) { buffer(b); }
    }


    private class ProcessedEntry {
        private final JarEntry jarEntry;
        private final long offset;
        private long crc;

        private ProcessedEntry(final JarEntry jarEntry, final long offset) {
            this.jarEntry = jarEntry;
            this.offset = offset;
        }
    }

    /**
     * Basic state machine that will allow the process to transition between the different process states.
     * 

* The following describes the process flow: * [NOT_STARTED] - Initial state * - Does not provide content * - Transitions [LOCAL_ENTRY_HEADER] * [LOCAL_ENTRY_HEADER] - The phase for reading the Local Directory Header * - Provides content of the local directory header by populating and feeding off a buffer * - Transitions to [ENTRY_CONTENT] if the header was written * - Transitions to [START_CENTRAL_DIRECTORY] if this is the last local entry header * [ENTRY_CONTENT] - The phase for reading the content of an entry * - Provides content of the entry using the VirtualJarInputStream * - Transitions to [LOCAL_ENTRY_HEADER] * [START_CENTRAL_DIRECTORY] - Phased used to transition into the central directory * - Does not provide content * - Transitions to [CENTRAL_ENTRY_HEADER] * [CENTRAL_ENTRY_HEADER] - The phase for reading the content of a single central directory header * - Provides content for the central directory header by feeding off a buffer * - Transitions to [CENTRAL_ENTRY_HEADER] * - Transitions to [CENTRAL_END] if there are no more entries * [CENTRAL_END] - The phase for reading the contents of the central directory end * - Provides content for central directory end by feeing off a buffer * - Transitions to NULL to terminate the processing */ private enum State { NOT_STARTED { @Override State transition(VirtualJarFileInputStream jarFileInputStream) throws IOException { return LOCAL_ENTRY_HEADER; } }, LOCAL_ENTRY_HEADER { boolean buffered; @Override void init(final VirtualJarFileInputStream jarFileInputStream) throws IOException { buffered = jarFileInputStream.bufferLocalFileHeader(); } @Override int read(VirtualJarFileInputStream jarFileInputStream) throws IOException { final ByteBuffer buffer = jarFileInputStream.buffer; if (buffered && buffer.hasRemaining()) { return buffer.get(); } return -1; } @Override State transition(final VirtualJarFileInputStream virtualJarFileInputStream) throws IOException { if (buffered) { return ENTRY_CONTENT; } return START_CENTRAL_DIRECTORY; } }, ENTRY_CONTENT { @Override int read(final VirtualJarFileInputStream jarFileInputStream) throws IOException { final VirtualJarInputStream virtualJarInputStream = jarFileInputStream.virtualJarInputStream; return virtualJarInputStream.read(); } @Override State transition(final VirtualJarFileInputStream virtualJarFileInputStream) throws IOException { virtualJarFileInputStream.closeCurrent(); return LOCAL_ENTRY_HEADER; } }, START_CENTRAL_DIRECTORY { @Override void init(final VirtualJarFileInputStream jarFileInputStream) throws IOException { jarFileInputStream.centralOffset = jarFileInputStream.totalRead; } @Override State transition(final VirtualJarFileInputStream virtualJarFileInputStream) throws IOException { return CENTRAL_ENTRY_HEADER; } }, CENTRAL_ENTRY_HEADER { boolean buffered; @Override void init(final VirtualJarFileInputStream jarFileInputStream) throws IOException { buffered = jarFileInputStream.bufferNextCentralFileHeader(); } @Override int read(final VirtualJarFileInputStream jarFileInputStream) throws IOException { final ByteBuffer buffer = jarFileInputStream.buffer; if (buffered && buffer.hasRemaining()) { return buffer.get(); } return -1; } @Override State transition(final VirtualJarFileInputStream virtualJarFileInputStream) throws IOException { if (buffered) { return CENTRAL_ENTRY_HEADER; } return CENTRAL_END; } }, CENTRAL_END { @Override void init(final VirtualJarFileInputStream jarFileInputStream) throws IOException { jarFileInputStream.bufferCentralDirectoryEnd(); } @Override int read(final VirtualJarFileInputStream jarFileInputStream) throws IOException { final ByteBuffer buffer = jarFileInputStream.buffer; if (buffer.hasRemaining()) { return buffer.get(); } return -1; } @Override State transition(final VirtualJarFileInputStream virtualJarFileInputStream) throws IOException { return null; } }; void init(VirtualJarFileInputStream jarFileInputStream) throws IOException { } abstract State transition(VirtualJarFileInputStream virtualJarFileInputStream) throws IOException; int read(VirtualJarFileInputStream jarFileInputStream) throws IOException { return -1; } State getNextState(VirtualJarFileInputStream jarFileInputStream) throws IOException { State nextState = transition(jarFileInputStream); if (nextState != null) { nextState.init(jarFileInputStream); } return nextState; } } private static class ByteBuffer { private final int bufferLength; private final byte[] buffer; private int bufferPosition; private int bufferDepth; private ByteBuffer(final int bufferLength) { this.buffer = new byte[bufferLength]; this.bufferLength = bufferLength; } private boolean hasRemaining() { return bufferPosition < bufferDepth; } private boolean hasCapacity() { return bufferDepth < bufferLength; } private byte get() { return buffer[bufferPosition++]; } private void put(byte b) { buffer[bufferDepth++] = b; } private void reset() { bufferPosition = 0; bufferDepth = 0; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy