proguard.io.ZipWriter Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of proguard-core Show documentation
Show all versions of proguard-core Show documentation
ProGuardCORE is a free library to read, analyze, modify, and write Java class files.
/*
* 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 boolean useZip64;
private final StringMatcher extraUncompressedAlignmentFilter;
private final int extraUncompressedAlignment;
private final int modificationTime;
private final byte[] header;
private final DataEntryWriter dataEntryWriter;
private DataEntry currentParentEntry;
public 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,
false,
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 useZip64 Whether to write out the archive in zip64 format.
* @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,
boolean useZip64,
int modificationTime,
DataEntryWriter dataEntryWriter)
{
this(uncompressedFilter,
uncompressedAlignment,
useZip64,
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 useZip64 Whether to write out the archive in zip64 format.
* @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,
boolean useZip64,
StringMatcher extraUncompressedAlignmentFilter,
int extraUncompressedAlignment,
int modificationTime,
DataEntryWriter dataEntryWriter)
{
this.uncompressedFilter = uncompressedFilter;
this.uncompressedAlignment = uncompressedAlignment;
this.useZip64 = useZip64;
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 useZip64 Whether to write out the archive in zip64 format.
* @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,
boolean useZip64,
int modificationTime,
byte[] header,
DataEntryWriter dataEntryWriter)
{
this(uncompressedFilter,
uncompressedAlignment,
useZip64,
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.useZip64 = false;
this.extraUncompressedAlignmentFilter = extraUncompressedAlignmentFilter;
this.extraUncompressedAlignment = extraUncompressedAlignment;
this.modificationTime = modificationTime;
this.header = header;
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 useZip64 Whether to write out the archive in zip64 format.
* @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,
boolean useZip64,
StringMatcher extraUncompressedAlignmentFilter,
int extraUncompressedAlignment,
int modificationTime,
byte[] header,
DataEntryWriter dataEntryWriter)
{
this.uncompressedFilter = uncompressedFilter;
this.uncompressedAlignment = uncompressedAlignment;
this.useZip64 = useZip64;
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,
useZip64,
null);
}
}
/**
* Creates a zip output with the given header and parameters.
*/
protected ZipOutput createZipOutput(OutputStream outputStream,
byte[] header,
int uncompressedAlignment,
boolean useZip64,
String comment)
throws IOException
{
return new ZipOutput(outputStream,
header,
uncompressedAlignment,
useZip64,
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 - 2025 Weber Informatics LLC | Privacy Policy