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

org.zeroturnaround.zip.Zips Maven / Gradle / Ivy

/**
 *    Copyright (C) 2012 ZeroTurnaround LLC 
 *
 *    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.zeroturnaround.zip;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileFilter;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipInputStream;
import java.util.zip.ZipOutputStream;

import org.zeroturnaround.zip.commons.FileUtils;
import org.zeroturnaround.zip.commons.IOUtils;
import org.zeroturnaround.zip.transform.ZipEntryTransformer;
import org.zeroturnaround.zip.transform.ZipEntryTransformerEntry;

/**
 * Fluent api for zip handling.
 *
 * @author shelajev
 *
 */
public class Zips {

  /**
   * Source archive.
   */
  private final File src;

  /**
   * Optional destination archive, if null, src will be overwritten
   */
  private File dest;

  /**
   * Charset to use for entry names
   */
  private Charset charset;

  /**
   * Flag to carry timestamps of entries on.
   */
  private boolean preserveTimestamps;

  /**
   * List
   */
  private List changedEntries = new ArrayList();

  /**
   * Set
   */
  private Set removedEntries = new HashSet();

  /**
   * List
   */
  private List transformers = new ArrayList();

  /*
   * If you want many name mappers here, you can create some compound instance that knows if
   * it wants to stop after first successfull mapping or go through all transformations, if null means
   * stop and ignore entry or that name mapper didn't know how to transform, etc.
   */
  private NameMapper nameMapper;

  /**
   * Flag to show that we want the final result to be unpacked
   */
  private boolean unpackedResult;

  private Zips(File src) {
    this.src = src;
  }

  /**
   * Static factory method to obtain an instance of Zips.
   *
   * @param src zip file to process
   * @return instance of Zips
   */
  public static Zips get(File src) {
    return new Zips(src);
  }

  /**
   * Static factory method to obtain an instance of Zips without source file.
   * See {@link #get(File src)}.
   *
   * @return instance of Zips
   */
  public static Zips create() {
    return new Zips(null);
  }

  /**
   * Specifies an entry to add or change to the output when this Zips executes.
   * Adding takes precedence over removal of entries.
   *
   * @param entry entry to add
   * @return this Zips for fluent api
   */
  public Zips addEntry(ZipEntrySource entry) {
    this.changedEntries.add(entry);
    return this;
  }

  /**
   * Specifies entries to add or change to the output when this Zips executes.
   * Adding takes precedence over removal of entries.
   *
   * @param entries entries to add
   * @return this Zips for fluent api
   */
  public Zips addEntries(ZipEntrySource[] entries) {
    this.changedEntries.addAll(Arrays.asList(entries));
    return this;
  }

  /**
   * Adds a file entry. If given file is a dir, adds it and all subfiles recursively.
   * Adding takes precedence over removal of entries.
   *
   * @param file file to add.
   * @return this Zips for fluent api
   */
  public Zips addFile(File file) {
    return addFile(file, false, null);
  }

  /**
   * Adds a file entry. If given file is a dir, adds it and all subfiles recursively.
   * Adding takes precedence over removal of entries.
   *
   * @param file file to add.
   * @param preserveRoot if file is a directory, true indicates we want to preserve this dir in the zip.
   *          otherwise children of the file are added directly under root.
   * @return this Zips for fluent api
   */
  public Zips addFile(File file, boolean preserveRoot) {
    return addFile(file, preserveRoot, null);
  }

  /**
   * Adds a file entry. If given file is a dir, adds it and all subfiles recursively.
   * Adding takes precedence over removal of entries.
   *
   * @param file file to add.
   * @param filter a filter to accept files for adding, null means all files are accepted
   * @return this Zips for fluent api
   */
  public Zips addFile(File file, FileFilter filter) {
    return this.addFile(file, false, filter);
  }

  /**
   * Adds a file entry. If given file is a dir, adds it and all subfiles recursively.
   * Adding takes precedence over removal of entries.
   *
   * @param file file to add.
   * @param preserveRoot if file is a directory, true indicates we want to preserve this dir in the zip.
   *          otherwise children of the file are added directly under root.
   * @param filter a filter to accept files for adding, null means all files are accepted
   * @return this Zips for fluent api
   */
  public Zips addFile(File file, boolean preserveRoot, FileFilter filter) {
    if (!file.isDirectory()) {
      this.changedEntries.add(new FileSource(file.getName(), file));
      return this;
    }
    
    Collection files = ZTFileUtil.listFiles(file);
    for (File entryFile : files) {
      if (filter != null && !filter.accept(entryFile)) {
        continue;
      }
      String entryPath = getRelativePath(file, entryFile);
      if (File.separatorChar == IOUtils.DIR_SEPARATOR_WINDOWS) {
        // replace directory separators on windows as at least 7zip packs zip with entries having "/" like on linux
        entryPath = entryPath.replace(IOUtils.DIR_SEPARATOR_WINDOWS, IOUtils.DIR_SEPARATOR_UNIX);
      }
      if (preserveRoot) {
        entryPath = file.getName() + entryPath;
      }
      if (entryPath.startsWith("/")) {
        entryPath = entryPath.substring(1);
      }
      this.changedEntries.add(new FileSource(entryPath, entryFile));
    }
    return this;
  }

  private String getRelativePath(File parent, File file) {
    String parentPath = parent.getPath();
    String filePath = file.getPath();
    if (!filePath.startsWith(parentPath)) {
      throw new IllegalArgumentException("File " + file + " is not a child of " + parent);
    }
    return filePath.substring(parentPath.length());
  }

  /**
   * Specifies an entry to remove to the output when this Zips executes.
   *
   * @param entry path of the entry to remove
   * @return this Zips for fluent api
   */
  public Zips removeEntry(String entry) {
    this.removedEntries.add(entry);
    return this;
  }

  /**
   * Specifies entries to remove to the output when this Zips executes.
   *
   * @param entries paths of the entry to remove
   * @return this Zips for fluent api
   */
  public Zips removeEntries(String[] entries) {
    this.removedEntries.addAll(Arrays.asList(entries));
    return this;
  }

  /**
   * Enables timestamp preserving for this Zips execution
   *
   * @return this Zips for fluent api
   */
  public Zips preserveTimestamps() {
    this.preserveTimestamps = true;
    return this;
  }

  /**
   * Specifies timestamp preserving for this Zips execution
   *
   * @param preserve flag to preserve timestamps
   * @return this Zips for fluent api
   */
  public Zips setPreserveTimestamps(boolean preserve) {
    this.preserveTimestamps = preserve;
    return this;
  }

  /**
   * Specifies charset for this Zips execution
   *
   * @param charset charset to use
   * @return this Zips for fluent api
   */
  public Zips charset(Charset charset) {
    this.charset = charset;
    return this;
  }

  /**
   * Specifies destination file for this Zips execution,
   * if destination is null (default value), then source file will be overwritten.
   * Temporary file will be used as destination and then written over the source to
   * create an illusion if inplace action.
   *
   * @param destination charset to use
   * @return this Zips for fluent api
   */
  public Zips destination(File destination) {
    this.dest = destination;
    return this;
  }

  /**
   *
   * @param nameMapper to use when processing entries
   * @return this Zips for fluent api
   */
  public Zips nameMapper(NameMapper nameMapper) {
    this.nameMapper = nameMapper;
    return this;
  }

  public Zips unpack() {
    this.unpackedResult = true;
    return this;
  }

  /**
   * @return true if destination is not specified.
   */
  private boolean isInPlace() {
    return dest == null;
  }

  /**
   * @return should the result of the processing be unpacked.
   */
  private boolean isUnpack() {
    return unpackedResult || (dest != null && dest.isDirectory());
  }

  /**
   * Registers a transformer for a given entry.
   *
   * @param path entry to transform
   * @param transformer transformer for the entry
   * @return this Zips for fluent api
   */
  public Zips addTransformer(String path, ZipEntryTransformer transformer) {
    this.transformers.add(new ZipEntryTransformerEntry(path, transformer));
    return this;
  }

  /**
   * Iterates through source Zip entries removing or changing them according to
   * set parameters.
   */
  public void process() {
    if (src == null && dest == null) {
      throw new IllegalArgumentException("Source and destination shouldn't be null together");
    }

    File destinationFile = null;
    try {
      destinationFile = getDestinationFile();
      ZipOutputStream out = null;
      ZipEntryOrInfoAdapter zipEntryAdapter = null;

      if (destinationFile.isFile()) {
        out = ZipFileUtil.createZipOutputStream(new BufferedOutputStream(new FileOutputStream(destinationFile)), charset);
        zipEntryAdapter = new ZipEntryOrInfoAdapter(new CopyingCallback(transformers, out, preserveTimestamps), null);
      }
      else { // directory
        zipEntryAdapter = new ZipEntryOrInfoAdapter(new UnpackingCallback(transformers, destinationFile), null);
      }
      try {
        processAllEntries(zipEntryAdapter);
      }
      finally {
        IOUtils.closeQuietly(out);
      }
        handleInPlaceActions(destinationFile);
    }
    catch (IOException e) {
      ZipExceptionUtil.rethrow(e);
    }
    finally {
      if (isInPlace()) {
        // destinationZip is a temporary file
        FileUtils.deleteQuietly(destinationFile);
      }
    }
  }

  private void processAllEntries(ZipEntryOrInfoAdapter zipEntryAdapter) {
    iterateChangedAndAdded(zipEntryAdapter);
    iterateExistingExceptRemoved(zipEntryAdapter);
  }

  private File getDestinationFile() throws IOException {
    if(isUnpack()) {
      if(isInPlace()) {
        File tempFile = File.createTempFile("zips", null);
        FileUtils.deleteQuietly(tempFile);
        tempFile.mkdirs(); // temp dir created
        return tempFile;
      }
      else {
        if (!dest.isDirectory()) {
          // destination is a directory, actually we shouldn't be here, because this should mean we want an unpacked result.
          FileUtils.deleteQuietly(dest);
          File result = new File(dest.getAbsolutePath());
          result.mkdirs(); // create a directory instead of dest file
          return result;
        }
        return dest;
      }
    }
    else {
      // we need a file
      if(isInPlace()) { // no destination specified, temp file
        return File.createTempFile("zips", ".zip");
      }
      else {
        if(dest.isDirectory()) {
          // destination is a directory, actually we shouldn't be here, because this should mean we want an unpacked result.
          FileUtils.deleteQuietly(dest);
          return new File(dest.getAbsolutePath());
        }
        return dest;
      }
    }
  }

  /**
   * Reads the source ZIP file and executes the given callback for each entry.
   * 

* For each entry the corresponding input stream is also passed to the callback. If you want to stop the loop then throw a ZipBreakException. * * This method is charset aware and uses Zips.charset. * * @param zipEntryCallback * callback to be called for each entry. * * @see ZipEntryCallback * */ public void iterate(ZipEntryCallback zipEntryCallback) { ZipEntryOrInfoAdapter zipEntryAdapter = new ZipEntryOrInfoAdapter(zipEntryCallback, null); processAllEntries(zipEntryAdapter); } /** * Scans the source ZIP file and executes the given callback for each entry. *

* Only the meta-data without the actual data is read. If you want to stop the loop then throw a ZipBreakException. * * This method is charset aware and uses Zips.charset. * * @param callback * callback to be called for each entry. * * @see ZipInfoCallback * @see #iterate(ZipEntryCallback) */ public void iterate(ZipInfoCallback callback) { ZipEntryOrInfoAdapter zipEntryAdapter = new ZipEntryOrInfoAdapter(null, callback); processAllEntries(zipEntryAdapter); } /** * Alias to ZipUtil.getEntry() * * @param name * name of the entry to fetch bytes from * @return byte[] * contents of the entry by given name */ public byte[] getEntry(String name) { if (src == null) { throw new IllegalStateException("Source is not given"); } return ZipUtil.unpackEntry(src, name); } /** * Alias to ZipUtil.containsEntry() * * @param name * entry to check existence of * @return true if zip archive we're processing contains entry by given name, false otherwise */ public boolean containsEntry(String name) { if (src == null) { throw new IllegalStateException("Source is not given"); } return ZipUtil.containsEntry(src, name); } // ///////////// private api /////////////// /** * Iterate through source for not removed entries with a given callback * * @param zipEntryCallback callback to execute on entries or their info. */ private void iterateExistingExceptRemoved(ZipEntryOrInfoAdapter zipEntryCallback) { if (src == null) { // if we don't have source specified, then we have nothing to iterate. return; } final Set removedDirs = ZipUtil.filterDirEntries(src, removedEntries); ZipFile zf = null; try { zf = getZipFile(); // manage existing entries Enumeration en = zf.entries(); while (en.hasMoreElements()) { ZipEntry entry = en.nextElement(); String entryName = entry.getName(); if (removedEntries.contains(entryName) || isEntryInDir(removedDirs, entryName)) { // removed entries are continue; } if (nameMapper != null) { String mappedName = nameMapper.map(entry.getName()); if (mappedName == null) { continue; // we should ignore this entry } else if (!mappedName.equals(entry.getName())) { // if name is different, do nothing entry = ZipEntryUtil.copy(entry, mappedName); } } InputStream is = zf.getInputStream(entry); try { zipEntryCallback.process(is, entry); } catch (ZipBreakException ex) { break; } finally { IOUtils.closeQuietly(is); } } } catch (IOException e) { ZipExceptionUtil.rethrow(e); } finally { ZipUtil.closeQuietly(zf); } } /** * Iterate through ZipEntrySources for added or changed entries with a given callback * * @param zipEntryCallback callback to execute on entries or their info */ private void iterateChangedAndAdded(ZipEntryOrInfoAdapter zipEntryCallback) { for (ZipEntrySource entrySource : changedEntries) { InputStream entrySourceStream = null; try { ZipEntry entry = entrySource.getEntry(); if (nameMapper != null) { String mappedName = nameMapper.map(entry.getName()); if (mappedName == null) { continue; // we should ignore this entry } else if (!mappedName.equals(entry.getName())) { // if name is different, do nothing entry = ZipEntryUtil.copy(entry, mappedName); } } entrySourceStream = entrySource.getInputStream(); zipEntryCallback.process(entrySourceStream, entry); } catch (ZipBreakException ex) { break; } catch (IOException e) { ZipExceptionUtil.rethrow(e); } finally { IOUtils.closeQuietly(entrySourceStream); } } } /** * if we are doing something in place, move result file into src. * * @param result destination zip file */ private void handleInPlaceActions(File result) throws IOException { if (isInPlace()) { // we operate in-place FileUtils.forceDelete(src); if (result.isFile()) { FileUtils.moveFile(result, src); } else { FileUtils.moveDirectory(result, src); } } } /** * Checks if entry given by name resides inside of one of the dirs. * * @param dirNames dirs * @param entryName entryPath */ private boolean isEntryInDir(Set dirNames, String entryName) { // this should be done with a trie, put dirNames in a trie and check if entryName leads to // some node or not. for(String dirName : dirNames) { if (entryName.startsWith(dirName)) { return true; } } return false; } /** * Creates a ZipFile from src and charset of this object. If a constructor with charset is * not available, throws an exception. * * @return ZipFile * @throws IOException if ZipFile cannot be constructed * @throws IllegalArgumentException if accessing constructor ZipFile(File, Charset) * */ private ZipFile getZipFile() throws IOException { return ZipFileUtil.getZipFile(src, charset); } private static class CopyingCallback implements ZipEntryCallback { private final Map entryByPath; private final ZipOutputStream out; private final Set visitedNames; private final boolean preserveTimestapms; private CopyingCallback(List transformerEntries, ZipOutputStream out, boolean preserveTimestapms) { this.out = out; this.preserveTimestapms = preserveTimestapms; entryByPath = ZipUtil.transformersByPath(transformerEntries); visitedNames = new HashSet(); } public void process(InputStream in, ZipEntry zipEntry) throws IOException { String entryName = zipEntry.getName(); if (visitedNames.contains(entryName)) { return; } visitedNames.add(entryName); ZipEntryTransformer transformer = (ZipEntryTransformer) entryByPath.remove(entryName); if (transformer == null) { // no transformer ZipEntryUtil.copyEntry(zipEntry, in, out, preserveTimestapms); } else { // still transfom entry transformer.transform(in, zipEntry, out); } } } private static class UnpackingCallback implements ZipEntryCallback { private final Map entryByPath; private final Set visitedNames; private final File destination; private UnpackingCallback(List entries, File destination) { this.destination = destination; this.entryByPath = ZipUtil.transformersByPath(entries); visitedNames = new HashSet(); } public void process(InputStream in, ZipEntry zipEntry) throws IOException { String entryName = zipEntry.getName(); if (visitedNames.contains(entryName)) { return; } visitedNames.add(entryName); File file = new File(destination, entryName); if (zipEntry.isDirectory()) { FileUtils.forceMkdir(file); return; } else { FileUtils.forceMkdir(file.getParentFile()); file.createNewFile(); } ZipEntryTransformer transformer = (ZipEntryTransformer) entryByPath.remove(entryName); if (transformer == null) { // no transformer FileUtils.copy(in, file); } else { // still transform entry transformIntoFile(transformer, in, zipEntry, file); } } private void transformIntoFile(final ZipEntryTransformer transformer, final InputStream entryIn, final ZipEntry zipEntry, final File destination) throws IOException { final PipedInputStream pipedIn = new PipedInputStream(); final PipedOutputStream pipedOut = new PipedOutputStream(pipedIn); final ZipOutputStream zipOut = new ZipOutputStream(pipedOut); final ZipInputStream zipIn = new ZipInputStream(pipedIn); ExecutorService newFixedThreadPool = Executors.newFixedThreadPool(1); try { newFixedThreadPool.execute(new Runnable() { public void run() { try { transformer.transform(entryIn, zipEntry, zipOut); } catch (IOException e) { ZipExceptionUtil.rethrow(e); } } }); zipIn.getNextEntry(); FileUtils.copy(zipIn, destination); } finally { try { zipIn.closeEntry(); } catch (IOException e) { // closing quietly } newFixedThreadPool.shutdown(); IOUtils.closeQuietly(pipedIn); IOUtils.closeQuietly(zipIn); IOUtils.closeQuietly(pipedOut); IOUtils.closeQuietly(zipOut); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy