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

proguard.io.ZipWriter Maven / Gradle / Ivy

Go to download

ProGuardCORE is a free library to read, analyze, modify, and write Java class files.

There is a newer version: 9.1.6
Show newest version
/*
 * ProGuardCORE -- library to process Java bytecode.
 *
 * Copyright (c) 2002-2020 Guardsquare NV
 *
 * 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 proguard.io;

import proguard.classfile.TypeConstants;
import proguard.util.StringMatcher;

import java.io.*;

/**
 * This {@link DataEntryWriter} sends data entries to the zip files specified by
 * their parents.
 *
 * @author Eric Lafortune
 */
public class ZipWriter implements DataEntryWriter
{
    private final StringMatcher   uncompressedFilter;
    private final int             uncompressedAlignment;
    private final StringMatcher   extraUncompressedAlignmentFilter;
    private final int             extraUncompressedAlignment;
    private final int             modificationTime;
    private final byte[]          header;
    private final DataEntryWriter dataEntryWriter;

    private DataEntry currentParentEntry;
    private ZipOutput currentZipOutput;


    /**
     * Creates a new ZipWriter that compresses all zip entries.
     * @param dataEntryWriter the data entry writer that can provide output
     *                        streams for the zip archives.
     */
    public ZipWriter(DataEntryWriter dataEntryWriter)
    {
        this(null,
             1,
             0,
             dataEntryWriter);
    }


    /**
     * Creates a new ZipWriter.
     * @param uncompressedFilter    an optional filter for files that should not
     *                              be compressed.
     * @param uncompressedAlignment the desired alignment for the data of
     *                              uncompressed entries.
     * @param modificationTime      the modification date and time of the zip
     *                              entries, in DOS format.
     * @param dataEntryWriter       the data entry writer that can provide
     *                              output streams for the zip archives.
     */
    public ZipWriter(StringMatcher   uncompressedFilter,
                     int             uncompressedAlignment,
                     int             modificationTime,
                     DataEntryWriter dataEntryWriter)
    {
        this(uncompressedFilter,
             uncompressedAlignment,
             null,
             1,
             modificationTime,
             dataEntryWriter);
    }


    /**
     * Creates a new ZipWriter.
     * @param uncompressedFilter                an optional filter for files that should not
     *                                          be compressed.
     * @param uncompressedAlignment             the desired alignment for the data of
     *                                          uncompressed entries.
     * @param extraUncompressedAlignmentFilter  an optional filter for files that should not
     *                                          be compressed and use a different alignment.
     * @param extraUncompressedAlignment        the desired alignment for the data of
     *                                          entries matching extraAlignmentFilter.
     * @param modificationTime                  the modification date and time of the zip
     *                                          entries, in DOS format.
     * @param dataEntryWriter                   the data entry writer that can provide
     *                                          output streams for the zip archives.
     */
    public ZipWriter(StringMatcher   uncompressedFilter,
                     int             uncompressedAlignment,
                     StringMatcher   extraUncompressedAlignmentFilter,
                     int             extraUncompressedAlignment,
                     int             modificationTime,
                     DataEntryWriter dataEntryWriter)
    {
        this.uncompressedFilter               = uncompressedFilter;
        this.uncompressedAlignment            = uncompressedAlignment;
        this.extraUncompressedAlignmentFilter = extraUncompressedAlignmentFilter;
        this.extraUncompressedAlignment       = extraUncompressedAlignment;
        this.modificationTime                 = modificationTime;
        this.header                           = null;
        this.dataEntryWriter                  = dataEntryWriter;
    }


    /**
     * Creates a new ZipWriter.
     * @param uncompressedFilter    an optional filter for files that should not
     *                              be compressed.
     * @param uncompressedAlignment the desired alignment for the data of
     *                              uncompressed entries.
     * @param modificationTime      the modification date and time of the zip
     *                              entries, in DOS format.
     * @param dataEntryWriter       the data entry writer that can provide
     *                              output streams for the zip archives.
     */
    public ZipWriter(StringMatcher   uncompressedFilter,
                     int             uncompressedAlignment,
                     int             modificationTime,
                     byte[]          header,
                     DataEntryWriter dataEntryWriter)
    {
        this(uncompressedFilter,
             uncompressedAlignment,
             null,
             1,
             modificationTime,
             header,
             dataEntryWriter);
    }


    /**
     * Creates a new ZipWriter.
     * @param uncompressedFilter                an optional filter for files that should not
     *                                          be compressed.
     * @param uncompressedAlignment             the desired alignment for the data of
     *                                          uncompressed entries.
     * @param extraUncompressedAlignmentFilter  an optional filter for files that should not
     *                                          be compressed and use a different alignment.
     * @param extraUncompressedAlignment        the desired alignment for the data of
     *                                          entries matching extraAlignmentFilter.
     * @param modificationTime                  the modification date and time of the zip
     *                                          entries, in DOS format.
     * @param dataEntryWriter                   the data entry writer that can provide
     *                                          output streams for the zip archives.
     */
    public ZipWriter(StringMatcher   uncompressedFilter,
                     int             uncompressedAlignment,
                     StringMatcher   extraUncompressedAlignmentFilter,
                     int             extraUncompressedAlignment,
                     int             modificationTime,
                     byte[]          header,
                     DataEntryWriter dataEntryWriter)
    {
        this.uncompressedFilter               = uncompressedFilter;
        this.uncompressedAlignment            = uncompressedAlignment;
        this.extraUncompressedAlignmentFilter = extraUncompressedAlignmentFilter;
        this.extraUncompressedAlignment       = extraUncompressedAlignment;
        this.modificationTime                 = modificationTime;
        this.header                           = header;
        this.dataEntryWriter                  = dataEntryWriter;
    }


    // Implementations for DataEntryWriter.

    @Override
    public boolean createDirectory(DataEntry dataEntry) throws IOException
    {
        finishIfNecessary(dataEntry);
        setUp(dataEntry);

        // Did we get a zip output?
        if (currentZipOutput == null)
        {
            return false;
        }

        // Get the directory entry name.
        String name = dataEntry.getName() + TypeConstants.PACKAGE_SEPARATOR;

        // Create a new directory entry.
        OutputStream outputStream =
            currentZipOutput.createOutputStream(name,
                                                false,
                                                modificationTime);
        outputStream.close();

        return true;
    }


    @Override
    public boolean sameOutputStream(DataEntry dataEntry1,
                                    DataEntry dataEntry2)
    throws IOException
    {
        return dataEntry1 != null &&
               dataEntry2 != null &&
               dataEntry1.getName().equals(dataEntry2.getName()) &&
               dataEntryWriter.sameOutputStream(dataEntry1.getParent(),
                                                dataEntry2.getParent());
    }


    @Override
    public OutputStream createOutputStream(DataEntry dataEntry) throws IOException
    {
        finishIfNecessary(dataEntry);
        setUp(dataEntry);

        // Did we get a zip output?
        if (currentZipOutput == null)
        {
            return null;
        }

        // Create a new zip entry.
        String name         = dataEntry.getName();
        String originalName = dataEntry.getOriginalName();

        // Only compress if both filters allow it.
        boolean compress1 = uncompressedFilter               == null || !uncompressedFilter.matches(originalName);
        boolean compress2 = extraUncompressedAlignmentFilter == null || !extraUncompressedAlignmentFilter.matches(originalName);

        // Set the alignment, or the extra alignment if the extra filter
        // matched.
        int uncompressedAlignment = extraUncompressedAlignmentFilter == null || compress2 ?
            this.uncompressedAlignment :
            this.extraUncompressedAlignment;

        return
            currentZipOutput.createOutputStream(name,
                                                compress1 && compress2,
                                                uncompressedAlignment,
                                                modificationTime);
    }


    @Override
    public void close() throws IOException
    {
        finish();

        // Close the delegate writer.
        dataEntryWriter.close();
    }


    @Override
    public void println(PrintWriter pw, String prefix)
    {
        pw.println(prefix + "ZipWriter (uncompressed filter = "+uncompressedFilter+", alignment = "+uncompressedAlignment+
                   ", extraAlignmentFilter = "+extraUncompressedAlignmentFilter+", extraAlignment = "+extraUncompressedAlignment+")");
        dataEntryWriter.println(pw, prefix + "  ");
    }


    // Small utility methods.

    /**
     * Sets up the zip output for the parent of the given entry.
     */
    private void setUp(DataEntry dataEntry) throws IOException
    {
        if (currentZipOutput == null)
        {
            // Create a new zip output.
            currentParentEntry = dataEntry.getParent();
            currentZipOutput   =
                createZipOutput(dataEntryWriter.createOutputStream(currentParentEntry),
                                header,
                                uncompressedAlignment,
                                null);
        }
    }


    /**
     * Creates a zip output with the given header and parameters.
     */
    protected ZipOutput createZipOutput(OutputStream  outputStream,
                                        byte[]        header,
                                        int           uncompressedAlignment,
                                        String        comment)
    throws IOException
    {
        return new ZipOutput(outputStream,
                             header,
                             uncompressedAlignment,
                             comment);
    }


    private void finishIfNecessary(DataEntry dataEntry) throws IOException
    {
        // Would the new data entry end up in a different zip?
        if (currentParentEntry != null &&
            !dataEntryWriter.sameOutputStream(currentParentEntry, dataEntry.getParent()))
        {
            finish();
        }
    }


    /**
     * Closes the zip output, if any.
     */
    private void finish() throws IOException
    {
        // Finish the zip output, if any.
        if (currentZipOutput != null)
        {
            // Close the zip output and its underlying output stream.
            currentZipOutput.close();

            currentParentEntry = null;
            currentZipOutput   = null;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy