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

org.opencastproject.util.ZipUtil Maven / Gradle / Ivy

There is a newer version: 16.7
Show newest version
/*
 * Licensed to The Apereo Foundation under one or more contributor license
 * agreements. See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 *
 * The Apereo Foundation licenses this file to you under the Educational
 * Community 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://opensource.org/licenses/ecl2.txt
 *
 * 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.opencastproject.util;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Vector;
import java.util.zip.Deflater;

import de.schlichtherle.io.ArchiveDetector;
import de.schlichtherle.io.ArchiveException;
import de.schlichtherle.io.ArchiveWarningException;
import de.schlichtherle.io.DefaultArchiveDetector;
import de.schlichtherle.io.File;
import de.schlichtherle.io.archive.zip.ZipDriver;

/*
 * WARNING:
 * Some archivers, such as file-roller in Ubuntu, seem not to be able to uncompress zip archives containg files with special characters.
 * The pure zip standard uses the CP437 encoding which CAN'T represent special characters, but applications have implemented their own methods
 * to overcome this inconvenience. TrueZip also. However, it seems file-roller (and probably others) doesn't "understand" how to restore the
 * original filenames and shows strange (and un-readable) characters in the unzipped filenames
 * However, the inverse process (unzipping zip archives made with file-roller and containing files with special characters) seems to work fine
 *
 * N.B. By "special characters" I mean those not present in the original ASCII specification, such as accents, special letters, etc
 *
 * ruben.perez
 */

/**
 * Provides static methods for compressing and extracting zip files using zip64 extensions when necessary.
 */
public final class ZipUtil {

  private static final Logger logger = LoggerFactory.getLogger(ZipUtil.class);

  public static final int BEST_SPEED = Deflater.BEST_SPEED;
  public static final int BEST_COMPRESSION = Deflater.BEST_COMPRESSION;
  public static final int DEFAULT_COMPRESSION = Deflater.DEFAULT_COMPRESSION;
  public static final int NO_COMPRESSION = Deflater.NO_COMPRESSION;

  /** Disable construction of this utility class */
  private ZipUtil() {
  }

  /**
   * Utility class to ease the process of umounting a zip file
   *
   * @param zipFile
   *          The file to umount
   * @throws IOException
   *           If some problem occurs on unmounting
   */
  private static void umount(File zipFile) throws IOException {
    try {
      File.umount(zipFile);
    } catch (ArchiveWarningException awe) {
      logger.warn("Umounting {} threw the following warning: {}", zipFile.getCanonicalPath(), awe.getMessage());
    } catch (ArchiveException ae) {
      logger.error("Unable to umount zip file: {}", zipFile.getCanonicalPath());
      throw new IOException("Unable to umount zip file: " + zipFile.getCanonicalPath(), ae);
    }
  }

  /***********************************************************************************/
  /* SERVICE CLASSES - The two following classes are the ones actually doing the job */
  /***********************************************************************************/

  /**
   * Compresses source files into a zip archive
   *
   * @param sourceFiles
   *          A {@link java.io.File} array with the files to include in the root of the archive
   * @param destination
   *          A {@link java.io.File} descriptor to the location where the zip file should be created
   * @param recursive
   *          Indicate whether or not recursively zipping nested directories
   * @param level
   *          The zip algorithm compression level. Ranges between 0 (no compression) and 9 (max. compression)
   * @return A {@link java.io.File} descriptor of the zip archive file
   * @throws IOException
   *           If the zip file can not be created, or the input files names can not be correctly parsed
   */
  public static java.io.File zip(java.io.File[] sourceFiles, java.io.File destination, boolean recursive, int level)
          throws IOException {

    if (sourceFiles == null) {
      logger.error("The array with files to zip cannot be null");
      throw new IllegalArgumentException("The array with files to zip cannot be null");
    }

    if (sourceFiles.length <= 0) {
      logger.error("The array with files to zip cannot be empty");
      throw new IllegalArgumentException("The array with files to zip cannot be empty");
    }

    if (destination == null) {
      logger.error("The destination file cannot be null");
      throw new IllegalArgumentException("The destination file cannot be null");
    }

    if (destination.exists()) {
      logger.error("The destination file {} already exists", destination.getCanonicalPath());
      throw new IllegalArgumentException("The destination file already exists");
    }

    if (level < -1) {
      logger.warn("Compression level cannot be less than 0 (or -1 for default)");
      logger.warn("Reverting to default...");
      level = -1;
    } else if (level > 9) {
      logger.warn("Compression level cannot be greater than 9");
      logger.warn("Reverting to default...");
      level = -1;
    }

    // Limits the compression support to ZIP only and sets the compression level
    ZipDriver zd = new ZipDriver(level);
    ArchiveDetector ad = new DefaultArchiveDetector(ArchiveDetector.NULL, "zip", zd);
    File zipFile;
    try {
      zipFile = new File(destination.getCanonicalFile(), ad);
    } catch (IOException ioe) {
      logger.error("Unable to create the zip file: {}", destination.getAbsolutePath());
      throw new IOException("Unable to create the zip file: {}" + destination.getAbsolutePath(), ioe);
    }

    try {
      if (!zipFile.isArchive()) {
        logger.error("The destination file does not represent a valid zip archive (.zip extension is required)");
        zipFile.deleteAll();
        throw new IllegalArgumentException(
                "The destination file does not represent a valid zip archive (.zip extension is required)");
      }

      if (!zipFile.mkdirs())
        throw new IOException("Couldn't create the destination file");

      for (java.io.File f : sourceFiles) {

        if (f == null) {
          logger.error("Null inputfile in array");
          zipFile.deleteAll();
          throw new IllegalArgumentException("Null inputfile in array");
        }

        logger.debug("Attempting to zip file {}...", f.getAbsolutePath());

        // TrueZip manual says that (archiveC|copy)All(From|To) methods work with either directories or regular files
        // Therefore, one could do zipFile.archiveCopyAllFrom(f), where f is a regular file, and it would work. Well, it
        // DOESN'T
        // This is why we have to tell if a file is a regular file or a directory BEFORE copying it with the appropriate
        // method
        boolean success = false;
        if (f.exists()) {
          if (!f.isDirectory() || recursive) {
            success = new File(zipFile, f.getName()).copyAllFrom(f);
            if (success)
              logger.debug("File {} zipped successfuly", f.getAbsolutePath());
            else {
              logger.error("File {} not zipped", f.getAbsolutePath());
              zipFile.deleteAll();
              throw new IOException("Failed to zip one of the input files: " + f.getAbsolutePath());
            }
          }
        } else {
          logger.error("Input file {} doesn't exist", f.getAbsolutePath());
          zipFile.deleteAll();
          throw new FileNotFoundException("One of the input files does not exist: " + f.getAbsolutePath());
        }
      }
    } catch (IOException e) {
      throw e;
    } finally {
      umount(zipFile);
    }

    return destination;
  }

  /**
   * Extracts a zip file to a directory.
   *
   * @param zipFile
   *          A {@link String} with the path to the source zip archive
   * @param destination
   *          A {@link String} with the location where the zip archive will be extracted. If this destination directory
   *          does not exist, it will be created.
   * @throws IOException
   *           if the zip file cannot be read, the destination directory cannot be created or the extraction is not
   *           successful
   */
  public static void unzip(java.io.File zipFile, java.io.File destination) throws IOException {

    boolean success;

    if (zipFile == null) {
      logger.error("The zip file cannot be null");
      throw new IllegalArgumentException("The zip file must be set");
    }

    if (!zipFile.exists()) {
      logger.error("The zip file does not exist: {}", zipFile.getCanonicalPath());
      throw new FileNotFoundException("The zip file does not exist: " + zipFile.getCanonicalPath());
    }

    if (destination == null) {
      logger.error("The destination file cannot be null");
      throw new IllegalArgumentException("Destination file cannot be null");
    }

    // FIXME Commented out for 3rd party compatibility. See comment in the zip method above -ruben.perez
    // File f = new File(zipFile.getCanonicalFile(), new DefaultArchiveDetector(ArchiveDetector.NULL, "zip", new
    // ZipDriver("utf-8")));
    File f;
    try {
      f = new File(zipFile.getCanonicalFile());
    } catch (IOException ioe) {
      logger.error("Unable to create the zip file: {}", destination.getAbsolutePath());
      throw new IOException("Unable to create the zip file: {}" + destination.getAbsolutePath(), ioe);
    }

    try {
      if (f.isArchive() && f.isDirectory()) {
        if (destination.exists()) {
          if (!destination.isDirectory()) {
            logger.error("Destination file must be a directory");
            throw new IllegalArgumentException("Destination file must be a directory");
          }
        }

        try {
          destination.mkdirs();
        } catch (SecurityException e) {
          logger.error("Cannot create destination directory: {}", e.getMessage());
          throw new IOException("Cannot create destination directory", e);
        }

        success = f.copyAllTo(destination);

        if (success)
          logger.debug("File {} unzipped successfully", zipFile.getCanonicalPath());
        else {
          logger.warn("File {} was not correctly unzipped", zipFile.getCanonicalPath());
          throw new IOException("File " + zipFile.getCanonicalPath() + " was not correctly unzipped");
        }
      } else {
        logger.error("The input file is not a valid zip file");
        throw new IllegalArgumentException("The input file is not a valid zip file");
      }
    } catch (IOException e) {
      throw (e);
    } finally {
      umount(f);
    }

  }

  /************************************************************************************* */
  /* "ALIASES" - For different types of input, but actually calling the previous methods */
  /***************************************************************************************/

  /**
   * Compresses source files into a zip archive
   *
   * @param sourceFiles
   *          A {@link String} array with the file names to be included in the root of the archive
   * @param destination
   *          A {@link String} with the path name of the resulting zip file
   * @param recursive
   *          Indicate whether or not recursively zipping nested directories
   * @param level
   *          The zip algorithm compression level. Ranges between 0 (no compression) and 9 (max. compression)
   * @return A {@link java.io.File} descriptor of the zip archive file
   * @throws IOException
   *           If the zip file can not be created, or the input files names can not be correctly parsed
   */
  public static java.io.File zip(String[] sourceFiles, String destination, boolean recursive, int level)
          throws IOException {

    if (sourceFiles == null) {
      logger.error("The input String array cannot be null");
      throw new IllegalArgumentException("The input String array cannot be null");
    }

    if (destination == null) {
      logger.error("Destination file cannot be null");
      throw new IllegalArgumentException("Destination file cannot be null");
    }

    if ("".equals(destination)) {
      logger.error("Destination file name must be set");
      throw new IllegalArgumentException("Destination file name must be set");
    }

    Vector files = new Vector();
    for (String name : sourceFiles) {
      if (name == null) {
        logger.error("One of the input file names is null");
        throw new IllegalArgumentException("One of the input file names is null");
      } else if ("".equals(name)) {
        logger.error("One of the input file names is blank");
        throw new IllegalArgumentException("One of the input file names is blank");
      }
      files.add(new java.io.File(name));
    }

    return zip(files.toArray(new java.io.File[files.size()]), new java.io.File(destination), recursive, level);

  }

  /**
   * Compresses source files into a zip archive
   *
   * @param sourceFiles
   *          A {@link String} array with the file names to be included in the root of the archive
   * @param destination
   *          A {@link java.io.File} with the path name of the resulting zip file
   * @param recursive
   *          Indicate whether or not recursively zipping nested directories
   * @param level
   *          The zip algorithm compression level. Ranges between 0 (no compression) and 9 (max. compression)
   * @return A {@link java.io.File} descriptor of the zip archive file
   * @throws IOException
   *           If the zip file can not be created, or the input files names can not be correctly parsed
   */
  public static java.io.File zip(String[] sourceFiles, java.io.File destination, boolean recursive, int level)
          throws IOException {

    if (sourceFiles == null) {
      logger.error("The input String array cannot be null");
      throw new IllegalArgumentException("The input String array cannot be null");
    }

    Vector files = new Vector();
    for (String name : sourceFiles) {
      if (name == null) {
        logger.error("One of the input file names is null");
        throw new IllegalArgumentException("One of the input file names is null");
      } else if ("".equals(name)) {
        logger.error("One of the input file names is blank");
        throw new IllegalArgumentException("One of the input file names is blank");
      }
      files.add(new java.io.File(name));
    }

    return zip(files.toArray(new java.io.File[files.size()]), destination, recursive, level);

  }

  /**
   * Compresses source files into a zip archive
   *
   * @param sourceFiles
   *          A {@link java.io.File} array with the file names to be included in the root of the archive
   * @param destination
   *          A {@link String} with the path name of the resulting zip file
   * @param recursive
   *          Indicate whether or not recursively zipping nested directories
   * @param level
   *          The zip algorithm compression level. Ranges between 0 (no compression) and 9 (max. compression)
   * @return A {@link java.io.File} descriptor of the zip archive file
   * @throws IOException
   *           If the zip file can not be created, or the input files names can not be correctly parsed
   */
  public static java.io.File zip(java.io.File[] sourceFiles, String destination, boolean recursive, int level)
          throws IOException {

    if (destination == null) {
      logger.error("Destination file cannot be null");
      throw new IllegalArgumentException("Destination file cannot be null");
    }

    if ("".equals(destination)) {
      logger.error("Destination file name must be set");
      throw new IllegalArgumentException("Destination file name must be set");
    }

    return zip(sourceFiles, new java.io.File(destination), recursive, level);

  }

  /**
   * Compresses source files into a zip archive (no recursive)
   *
   * @param sourceFiles
   *          A {@link java.io.File} array with the file names to be included in the root of the archive
   * @param destination
   *          A {@link java.io.File} with the path name of the resulting zip file
   * @param level
   *          The zip algorithm compression level. Ranges between 0 (no compression) and 9 (max. compression)
   * @return A {@link java.io.File} descriptor of the zip archive file
   * @throws IOException
   *           If the zip file can not be created, or the input files names can not be correctly parsed
   */
  public static java.io.File zip(java.io.File[] sourceFiles, java.io.File destination, int level) throws IOException {
    return zip(sourceFiles, destination, false, level);
  }

  /**
   * Extracts a zip file to a directory.
   *
   * @param zipFile
   *          A {@link String} with the path to the source zip archive
   * @param destination
   *          A {@link String} with the location where the zip archive will be extracted. If this destination directory
   *          does not exist, it will be created.
   * @throws IOException
   *           if the zip file cannot be read, the destination directory cannot be created or the extraction is not
   *           successful
   */
  public static void unzip(String zipFile, String destination) throws IOException {

    if (zipFile == null) {
      logger.error("Input filename cannot be null");
      throw new IllegalArgumentException("Input filename cannot be null");
    }

    if ("".equals(zipFile)) {
      logger.error("Input filename cannot be empty");
      throw new IllegalArgumentException("Input filename cannot be empty");
    }

    if (destination == null) {
      logger.error("Output filename cannot be null");
      throw new IllegalArgumentException("Output filename cannot be null");
    }

    if ("".equals(destination)) {
      logger.error("Output filename cannot be empty");
      throw new IllegalArgumentException("Output filename cannot be empty");
    }

    unzip(new java.io.File(zipFile), new java.io.File(destination));

  }

  /**
   * Extracts a zip file to a directory.
   *
   * @param zipFile
   *          A {@link java.io.File} with the path to the source zip archive
   * @param destination
   *          A {@link String} with the location where the zip archive will be extracted.
   * @throws IOException
   *           if the zip file cannot be read, the destination directory cannot be created or the extraction is not
   *           successful
   */
  public static void unzip(java.io.File zipFile, String destination) throws IOException {

    if (destination == null) {
      logger.error("Output filename cannot be null");
      throw new IllegalArgumentException("Output filename cannot be null");
    }

    if ("".equals(destination)) {
      logger.error("Output filename cannot be empty");
      throw new IllegalArgumentException("Output filename cannot be empty");
    }

    unzip(zipFile, new java.io.File(destination));

  }

  /**
   * Extracts a zip file to a directory.
   *
   * @param zipFile
   *          A {@link String} with the path to the source zip archive
   * @param destination
   *          A {@link java.io.File} with the location where the zip archive will be extracted.
   * @throws IOException
   *           if the zip file cannot be read, the destination directory cannot be created or the extraction is not
   *           successful
   */
  public static void unzip(String zipFile, java.io.File destination) throws IOException {

    if (zipFile == null) {
      logger.error("Input filename cannot be null");
      throw new IllegalArgumentException("Input filename cannot be null");
    }

    if ("".equals(zipFile)) {
      logger.error("Input filename cannot be empty");
      throw new IllegalArgumentException("Input filename cannot be empty");
    }

    unzip(new java.io.File(zipFile), destination);

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy