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

com.cedarsolutions.util.FilesystemUtils Maven / Gradle / Ivy

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *
 *              C E D A R
 *          S O L U T I O N S       "Software done right."
 *           S O F T W A R E
 *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *
 * Copyright (c) 2013-2016 Kenneth J. Pronovici.
 * All rights reserved.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the Apache License, Version 2.0.
 * See LICENSE for more information about the licensing terms.
 *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 *
 * Author   : Kenneth J. Pronovici 
 * Language : Java 6
 * Project  : Common Java Functionality
 *
 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
package com.cedarsolutions.util;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Reader;
import java.nio.channels.FileChannel;
import java.util.ArrayList;
import java.util.Date;
import java.util.Enumeration;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.FilenameUtils;
import org.joda.time.DateTime;
import org.joda.time.DateTimeZone;

import com.cedarsolutions.exception.CedarRuntimeException;
import com.esotericsoftware.wildcard.Paths;

/**
 * Filesystem-related utilities.
 * @author Kenneth J. Pronovici 
 */
public class FilesystemUtils {

    /** Get the current working directory, normalized to use all '/' characters. */
    public static String getCurrentWorkingDir() {
        return normalize(new File(".").getAbsolutePath());
    }

    /** Normalize a path, so it uses all '/' characters. */
    public static String normalize(String path) {
        if (path == null) {
            return null;
        } else if (path.length() == 0) {
            return "";
        } else {
            File file = new File(path);
            return file.getPath().replace("\\", "/");
        }
    }

    /**
     * Join a list of filesystem elements together, kind of like Python's os.path.join().
     * @param elements  List of elements to be joined.
     * @return Complete path, with elements separated by '/' characters.
     */
    public static String join(String ... elements) {
        StringBuffer buffer = new StringBuffer();

        if (elements.length > 0) {
            int index = 0;
            for (String element : elements) {
                if (element != null) {
                    if (index++ > 0) {
                        buffer.append("/");
                    }
                    buffer.append(element);
                }
            }
        }

        return buffer.toString();
    }

    /**
     * Get the base name for a path, like Python's os.path.basename().
     * @param path    Path to operate on
     * @return Base name for the path, possibly null.
     */
    public static String getBasename(String path) {
        if (StringUtils.isEmpty(path)) {
            return null;
        } else {
            File file = new File(path);
            return StringUtils.trimToNull(file.getName());
        }
    }

    /**
     * Get the directory name for a path, like Python's os.path.dirname().
     * @param path    Path to operate on
     * @return Directory name for the path, possibly null.
     */
    public static String getDirname(String path) {
        if (StringUtils.isEmpty(path)) {
            return null;
        } else {
            File file = new File(path);
            String parent = file.getParent();
            return parent == null ? null : parent.replace("\\", "/");
        }
    }

    /**
     * Get the length of a file on disk.
     * @param filePath  Path of the file to check
     * @return Length of the file in bytes, or zero if the file does not exist.
     */
    public static long getFileSize(String filePath) {
        File file = new File(filePath);
        return file.exists() && file.isFile() ? file.length() : 0;
    }

    /**
     * Check whether a path is absolute.
     * @param  path   Path to operate on
     * @return True if the path is absolute, false otherwise.
     * @see StackOverflow
     */
    public static boolean isAbsolutePath(String path) {
        return FilenameUtils.getPrefixLength(path) != 0;
    }

    /**
     * Copy a file into a directory, overwriting the file if it exists.
     * @param sourceFilePath  Source file path
     * @param targetDirPath   Target directory path
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void copyFileToDir(String sourceFilePath, String targetDirPath) {
        try {
            File sourceFile = new File(sourceFilePath);
            String targetFilePath = join(targetDirPath, sourceFile.getName());
            File targetFile = new File(targetFilePath);
            copyFile(sourceFile, targetFile);
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to copy directory: " + e.getMessage(), e);
        }
    }

    /**
     * Copy a file, overwriting the target if it exists.
     * @param sourceFilePath  Source file path
     * @param targetFilePath  Target file path
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void copyFile(String sourceFilePath, String targetFilePath) {
        File sourceFile = new File(sourceFilePath);
        File targetFile = new File(targetFilePath);
        copyFile(sourceFile, targetFile);
    }

    /**
     * Copy a file, overwriting the target if it exists.
     * @param sourceFile  Source file
     * @param targetFile  Target file
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void copyFile(File sourceFile, File targetFile) {
        FileInputStream sourceStream = null;
        FileOutputStream targetStream = null;
        FileChannel sourceChannel = null;
        FileChannel targetChannel = null;

        try {
            if (!targetFile.exists()) {
                targetFile.createNewFile();
            }

            sourceStream = new FileInputStream(sourceFile);
            targetStream = new FileOutputStream(targetFile);
            sourceChannel = sourceStream.getChannel();
            targetChannel = targetStream.getChannel();

            targetChannel.transferFrom(sourceChannel, 0, sourceChannel.size());
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to copy file: " + e.getMessage(), e);
        } finally {
            close(targetChannel);
            close(sourceChannel);
            close(targetStream);
            close(sourceStream);
        }
    }

    /**
     * Move a file, overwriting the target if it exists.
     * @param sourceFilePath  Source file path
     * @param targetFilePath  Target file path
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void moveFile(String sourceFilePath, String targetFilePath) {
        File sourceFile = new File(sourceFilePath);
        File targetFile = new File(targetFilePath);
        moveFile(sourceFile, targetFile);
    }

    /**
     * Move a file, overwriting the target if it exists.
     * @param sourceFile  Source file
     * @param targetFile  Target file
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void moveFile(File sourceFile, File targetFile) {
        try {
            FileUtils.moveFile(sourceFile, targetFile);
        } catch (IOException e) {
            throw new CedarRuntimeException("Failed to move file: " + e.getMessage(), e);
        }
    }

    /**
     * Create a file in an existing directory.
     * @param filePath  Path of the file to create
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void createFile(String filePath) {
        try {
            new File(filePath).createNewFile();
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to create file: " + e.getMessage(), e);
        }
    }

    /**
     * Remove a file, if it exists.
     * @param filePath  Path of the file to remove
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void removeFile(String filePath) {
        File file = new File(filePath);
        if (file.exists()) {
            if (!file.delete()) {
                throw new CedarRuntimeException("Failed to remove file.");
            } else if (fileExists(filePath)) {
                throw new CedarRuntimeException("Failed to remove file.");
            }
        }
    }

    /**
     * Indicate whether a file exists.
     * @param filePath  Path of the file to check
     * @return True if the file exists, false otherwise.
     */
    public static boolean fileExists(String filePath) {
        File file = new File(filePath);
        return file.exists() && file.isFile();
    }

    /**
     * Create a directory including all parent directories, like 'mkdir -p'.
     * @param dirPath  Path of the directory to create
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void createDir(String dirPath) {
        File dirFile = new File(dirPath);
        if (!dirFile.mkdirs() && !dirFile.exists()) {
            throw new CedarRuntimeException("Failed to create directory " + dirPath + ".");
        }
    }

    /**
     * Remove an empty directory, if it exists.
     * @param dirPath   Path of the directory to remove.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void removeEmptyDir(String dirPath) {
        removeDir(dirPath, false);
    }

    /**
     * Remove a directory, if it exists.
     * If recursive=false, the directory must be empty.
     * @param dirPath    Path of the directory to remove.
     * @param recursive  Whether the directory should be removed recursively.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void removeDir(String dirPath, boolean recursive) {
        File dirFile = new File(dirPath);
        if (!recursive) {
            if (dirFile.exists()) {
                if (!dirFile.delete()) {
                    throw new CedarRuntimeException("Failed to remove directory.");
                } else if (dirExists(dirPath)) {
                    throw new CedarRuntimeException("Failed to remove directory.");
                }
            }
        } else {
            removeDirRecursive(dirFile);
            if (dirExists(dirPath)) {
                throw new CedarRuntimeException("Failed to remove directory.");
            }
        }
    }

    /**
     * Recursively remove a directory and all of its contents.
     * @param dirFile   Directory to remove
     */
    private static void removeDirRecursive(File dirFile) {
        if (dirFile.exists()) {
            File[] targetFiles = dirFile.listFiles();
            for (int i = 0; i < targetFiles.length; i++) {
                if (targetFiles[i].isDirectory()) {
                    removeDirRecursive(targetFiles[i]);
                } else {
                    targetFiles[i].delete();
                }
            }

            dirFile.delete();
        }
    }

    /**
     * Indicates whether a directory exists.
     * @param dirPath   Path of directory to check.
     * @return True if the directory exists, false otherwise.
     */
    public static boolean dirExists(String dirPath) {
        File dirFile = new File(dirPath);
        return dirFile.exists() && dirFile.isDirectory();
    }

    /**
     * Writes string contents to a file, replacing the file if it already exists.
     * @param filePath  Path of the file to write
     * @param contents  String contents to be written to the file
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void writeFileContents(String filePath, String contents) {
        FileOutputStream stream = null;

        try {
            stream = new FileOutputStream(filePath);
            stream.write(StringUtils.getBytes(contents));
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to write file: " + e.getMessage(), e);
        } finally {
            close(stream);
        }
    }

    /**
     * Writes string contents to a file, replacing the file if it already exists.
     * @param filePath  Path of the file to write
     * @param contents  Set of lines to be written to the file
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static void writeFileContents(String filePath, List contents) {
        FileOutputStream stream = null;

        try {
            stream = new FileOutputStream(filePath);
            for (String line : contents) {
                stream.write(StringUtils.getBytes(line));
                stream.write(StringUtils.LINE_ENDING_BYTES);
            }
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to write file: " + e.getMessage(), e);
        } finally {
            close(stream);
        }
    }

    /**
     * Get the contents of a file.
     * @param filePath  Path of the file to read
     * @return Contents of the file as a list of strings.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static List getFileContents(String filePath) {
        String line = null;
        FileReader fileReader = null;
        BufferedReader bufferedReader = null;
        List lines = new ArrayList();

        try {
            fileReader = new FileReader(filePath);
            bufferedReader = new BufferedReader(fileReader);

            do {
                line = bufferedReader.readLine();
                if (line != null) {
                    lines.add(line);
                }
            } while (line != null);
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to read file: " + e.getMessage(), e);
        } finally {
            close(bufferedReader);
            close(fileReader);
        }

        return lines;
    }

    /**
     * Get the contents of a file.
     * @param filePath  Path of the file to read
     * @return Contents of the file as a single multi-line string.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static String getFileContentsAsString(String filePath) {
        FileReader fileReader = null;
        BufferedReader bufferedReader = null;
        StringBuffer buffer = new StringBuffer();

        try {
            fileReader = new FileReader(filePath);
            bufferedReader = new BufferedReader(fileReader);

            int character = -1;
            while ((character = bufferedReader.read()) != -1) {
                buffer.append((char) character);
            }
        } catch (Exception e) {
            throw new CedarRuntimeException("Failed to read file: " + e.getMessage(), e);
        } finally {
            close(bufferedReader);
            close(fileReader);
        }

        return buffer.toString();
    }

    /**
     * Get a list of files and directories that match a glob pattern.
     * @param dirPath  Directory to operate on
     * @param glob     Ant-style glob to use
     * @return List of files and directories that match the glob, possibly recursive (depending on glob).
     * @see Stack Overflow
     */
    public static List getGlobContents(String dirPath, String glob) {
        List contents = new ArrayList();

        File dirFile = new File(dirPath);
        if (!dirFile.exists() || !dirFile.isDirectory()) {
            throw new CedarRuntimeException("Directory does not exist.");
        }

        Paths paths = new Paths(dirPath, glob);
        for (String path : paths.getRelativePaths()) {
            contents.add(normalize(path));
        }

        return contents;
    }

    /**
     * Get a list of the files and directories immediately within a directory.
     * @param dirPath  Directory to operate on
     * @return List of files and directories within the directory.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static List getDirContents(String dirPath) {
        return getDirContents(dirPath, false);
    }

    /**
     * Get a list of the files and directories within a directory.
     * @param dirPath    Directory to operate on
     * @param recursive  Recursively dig through all subdirectories
     * @return List of files and directories within the directory.
     * @throws CedarRuntimeException If there is a problem with the filesystem operation.
     */
    public static List getDirContents(String dirPath, boolean recursive) {
        try {
            List contents = new ArrayList();

            File dirFile = new File(dirPath);
            if (!dirFile.exists() || !dirFile.isDirectory()) {
                throw new CedarRuntimeException("Directory does not exist.");
            }

            File[] files = dirFile.listFiles();
            for (int i = 0; i < files.length; i++) {
                File file = files[i];
                contents.add(file.getName());
                if (file.isDirectory()) {
                    if (recursive) {
                        List subdir = getDirContents(file.getCanonicalPath(), recursive);
                        for (String entry : subdir) {
                            String path = join(file.getName(), entry);
                            contents.add(path);
                        }
                    }
                }
            }

            return contents;
        } catch (Exception e) {
            throw new CedarRuntimeException("Error getting dir contents: " + e.getMessage(), e);
        }
    }

    /**
     * Unzips the passed-in zip file.
     * @param zipFilePath   Path to the zip file on disk
     * @param targetDirPath Target directory that the zip contents should be written into
     * @see Stack Overflow
     */
    @SuppressWarnings("unchecked")
    public static void unzip(String zipFilePath, String targetDirPath) {
        ZipFile zip = null;
        try {
            zip = new ZipFile(zipFilePath);
            Enumeration entries = (Enumeration) zip.entries();
            while (entries.hasMoreElements()) {
                ZipEntry entry = entries.nextElement();
                java.io.File f = new java.io.File(targetDirPath, entry.getName());
                if (entry.isDirectory()) { // if its a directory, create it
                    continue;
                }

                if (!f.exists()) {
                    f.getParentFile().mkdirs();
                    f.createNewFile();
                }

                InputStream is = zip.getInputStream(entry); // get the input stream
                OutputStream os = new java.io.FileOutputStream(f);
                try {
                    byte[] buf = new byte[4096];
                    int r;
                    while ((r = is.read(buf)) != -1) {
                        os.write(buf, 0, r);
                    }
                } finally {
                    close(os);
                    close(is);
                }
            }
        } catch (Exception e) {
            throw new CedarRuntimeException("Error unzipping file: " + e.getMessage(), e);
        } finally {
            close(zip);
        }
    }

    /**
     * Get the last modified date for a file, in UTC.
     * @param filePath  Path of the file to check
     * @return UTC date representing the last modified time for the file.
     */
    public static Date getLastModifiedDate(String filePath) {
        File file = new File(filePath);
        if (!file.exists() || !file.isFile()) {
            throw new CedarRuntimeException("File does not exist: " + filePath);
        } else {
            long instant = file.lastModified();
            return new DateTime(instant, DateTimeZone.UTC).toDate();
        }
    }

    /** Close a reader, ignoring errors. */
    private static void close(Reader reader) {
        if (reader != null) {
            try {
                reader.close();
            } catch (IOException e) { }
        }
    }

    /** Close an input stream, ignoring errors. */
    private static void close(InputStream stream) {
        if (stream != null) {
            try {
                stream.close();
            } catch (IOException e) { }
        }
    }

    /** Close an output stream, ignoring errors. */
    private static void close(OutputStream stream) {
        if (stream != null) {
            try {
                stream.close();
            } catch (IOException e) { }
        }
    }

    /** Close a file channel, ignoring errors. */
    private static void close(FileChannel channel) {
        if (channel != null) {
            try {
                channel.close();
            } catch (IOException e) { }
        }
    }

    /** Close a zip file channel, ignoring errors. */
    private static void close(ZipFile zip) {
        if (zip != null) {
            try {
                zip.close();
            } catch (IOException e) { }
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy