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

jodd.io.FileUtil Maven / Gradle / Ivy

// Copyright (c) 2003-present, Jodd Team (http://jodd.org)
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.

package jodd.io;

import jodd.core.JoddCore;
import jodd.util.StringPool;
import jodd.util.StringUtil;
import jodd.util.SystemUtil;
import jodd.util.URLDecoder;

import java.io.BufferedInputStream;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.RandomAccessFile;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.FileFilter;
import java.io.Writer;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URL;
import java.security.DigestInputStream;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.List;
import java.util.ArrayList;

import static jodd.core.JoddCore.fileUtilParams;

/**
 * File utilities.
 */
public class FileUtil {

	private static final String MSG_NOT_A_DIRECTORY = "Not a directory: ";
	private static final String MSG_CANT_CREATE = "Can't create: ";
	private static final String MSG_NOT_FOUND = "Not found: ";
	private static final String MSG_NOT_A_FILE = "Not a file: ";
	private static final String MSG_ALREADY_EXISTS = "Already exists: ";
	private static final String MSG_UNABLE_TO_DELETE = "Unable to delete: ";

	/**
	 * Simple factory for File objects.
	 */
	private static File file(String fileName) {
		return new File(fileName);
	}

	/**
	 * Simple factory for File objects.
	 */
	private static File file(File parent, String fileName) {
		return new File(parent, fileName);
	}

	// ---------------------------------------------------------------- misc shortcuts

	/**
	 * Checks if two files points to the same file.
	 */
	public static boolean equals(String file1, String file2) {
		return equals(file(file1), file(file2));
	}

	/**
	 * Checks if two files points to the same file.
	 */
	public static boolean equals(File file1, File file2) {
		try {
			file1 = file1.getCanonicalFile();
			file2 = file2.getCanonicalFile();
		} catch (IOException ignore) {
			return false;
		}
		return file1.equals(file2);
	}

	/**
	 * Converts file URLs to file. Ignores other schemes and returns null.
	 */
	public static File toFile(URL url) {
		String fileName = toFileName(url);
		if (fileName == null) {
			return null;
		}
		return file(fileName);
	}

	/**
	 * Converts file to URL in a correct way.
	 * Returns null in case of error.
	 */
	public static URL toURL(File file) throws MalformedURLException {
		return file.toURI().toURL();
	}

	/**
	 * Converts file URLs to file name. Accepts only URLs with 'file' protocol.
	 * Otherwise, for other schemes returns null.
	 */
	public static String toFileName(URL url) {
		if ((url == null) || !(url.getProtocol().equals("file"))) {
			return null;
		}
		String filename = url.getFile().replace('/', File.separatorChar);

		return URLDecoder.decode(filename, JoddCore.encoding);
	}

	/**
	 * Returns a file of either a folder or a containing archive.
	 */
	public static File toContainerFile(URL url) {
		String protocol = url.getProtocol();
		if (protocol.equals("file")) {
			return toFile(url);
		}

		String path = url.getPath();

		return new File(URI.create(
				path.substring(0, path.lastIndexOf("!/"))));
	}

	/**
	 * Returns true if file exists.
	 */
	public static boolean isExistingFile(File file) {
		if (file == null) {
			return false;
		}
		return file.exists() && file.isFile();
	}

	/**
	 * Returns true if folder exists.
	 */
	public static boolean isExistingFolder(File folder) {
		if (folder == null) {
			return false;
		}
		return folder.exists() && folder.isDirectory();
	}

	// ---------------------------------------------------------------- mkdirs

	/**
	 * Creates all folders at once.
	 * @see #mkdirs(java.io.File)
	 */
	public static void mkdirs(String dirs) throws IOException {
		mkdirs(file(dirs));
	}
	/**
	 * Creates all folders at once.
	 */
	public static void mkdirs(File dirs) throws IOException {
		if (dirs.exists()) {
			if (!dirs.isDirectory()) {
				throw new IOException(MSG_NOT_A_DIRECTORY + dirs);
			}
			return;
		}
		if (!dirs.mkdirs()) {
			throw new IOException(MSG_CANT_CREATE + dirs);
		}
	}

	/**
	 * Creates single folder.
	 * @see #mkdir(java.io.File)
	 */
	public static void mkdir(String dir) throws IOException {
		mkdir(file(dir));
	}
	/**
	 * Creates single folders.
	 */
	public static void mkdir(File dir) throws IOException {
		if (dir.exists()) {
			if (!dir.isDirectory()) {
				throw new IOException(MSG_NOT_A_DIRECTORY + dir);
			}
			return;
		}
		if (!dir.mkdir()) {
			throw new IOException(MSG_CANT_CREATE + dir);
		}
	}

	// ---------------------------------------------------------------- touch

	/**
	 * @see #touch(java.io.File)
	 */
	public static void touch(String file) throws IOException {
		touch(file(file));
	}
	/**
	 * Implements the Unix "touch" utility. It creates a new file
	 * with size 0 or, if the file exists already, it is opened and
	 * closed without modifying it, but updating the file date and time.
	 */
	public static void touch(File file) throws IOException {
		if (!file.exists()) {
			StreamUtil.close(new FileOutputStream(file));
		}
		file.setLastModified(System.currentTimeMillis());
	}


	// ---------------------------------------------------------------- params

	/**
	 * Creates new {@link FileUtilParams} instance by cloning current default params.
	 */
	public static FileUtilParams cloneParams() {
		try {
			return fileUtilParams.clone();
		} catch (CloneNotSupportedException ignore) {
			return null;
		}
	}

	/**
	 * Creates new {@link FileUtilParams} instance with default values.
	 */
	public static FileUtilParams params() {
		return new FileUtilParams();
	}

	// ---------------------------------------------------------------- copy file to file

	/**
	 * @see #copyFile(java.io.File, java.io.File, FileUtilParams)
	 */
	public static void copyFile(String src, String dest) throws IOException {
		copyFile(file(src), file(dest), fileUtilParams);
	}
	/**
	 * @see #copyFile(java.io.File, java.io.File, FileUtilParams)
	 */
	public static void copyFile(String src, String dest, FileUtilParams params) throws IOException {
		copyFile(file(src), file(dest), params);
	}
	/**
	 * @see #copyFile(java.io.File, java.io.File, FileUtilParams)
	 */
	public static void copyFile(File src, File dest) throws IOException {
		copyFile(src, dest, fileUtilParams);
	}

	/**
	 * Copies a file to another file with specified {@link FileUtilParams copy params}.
	 */
	public static void copyFile(File src, File dest, FileUtilParams params) throws IOException {
		checkFileCopy(src, dest, params);
		doCopyFile(src, dest, params);
	}

	private static void checkFileCopy(File src, File dest, FileUtilParams params) throws IOException {
		if (!src.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + src);
		}
		if (!src.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + src);
		}
		if (equals(src, dest)) {
			throw new IOException("Files '" + src + "' and '" + dest + "' are equal");
		}

		File destParent = dest.getParentFile();
		if (destParent != null && !destParent.exists()) {
			if (!params.createDirs) {
				throw new IOException(MSG_NOT_FOUND + destParent);
			}
			if (!destParent.mkdirs()) {
				throw new IOException(MSG_CANT_CREATE + destParent);
			}
		}
	}

	/**
	 * Internal file copy when most of the pre-checking has passed.
	 */
	private static void doCopyFile(File src, File dest, FileUtilParams params) throws IOException {
		if (dest.exists()) {
			if (dest.isDirectory()) {
				throw new IOException("Destination '" + dest + "' is a directory");
			}
			if (!params.overwrite) {
				throw new IOException(MSG_ALREADY_EXISTS + dest);
			}
		}

		// do copy file
		FileInputStream input = new FileInputStream(src);
		try {
			FileOutputStream output = new FileOutputStream(dest);
			try {
				StreamUtil.copy(input, output);
			} finally {
				StreamUtil.close(output);
			}
		} finally {
			StreamUtil.close(input);
		}

		// done

		if (src.length() != dest.length()) {
			throw new IOException("Copy file failed of '" + src + "' to '" + dest + "' due to different sizes");
		}
		if (params.preserveDate) {
			dest.setLastModified(src.lastModified());
		}
	}

	// ---------------------------------------------------------------- copy file to directory

	/**
	 * @see #copyFileToDir(java.io.File, java.io.File, FileUtilParams)
	 */
	public static File copyFileToDir(String src, String destDir) throws IOException {
		return copyFileToDir(file(src), file(destDir), fileUtilParams);
	}
	/**
	 * @see #copyFileToDir(java.io.File, java.io.File, FileUtilParams)
	 */
	public static File copyFileToDir(String src, String destDir, FileUtilParams params) throws IOException {
		return copyFileToDir(file(src), file(destDir), params);
	}
	/**
	 * @see #copyFileToDir(java.io.File, java.io.File, FileUtilParams)
	 */
	public static File copyFileToDir(File src, File destDir) throws IOException {
		return copyFileToDir(src, destDir, fileUtilParams);
	}
	/**
	 * Copies a file to folder with specified copy params and returns copied destination.
	 */
	public static File copyFileToDir(File src, File destDir, FileUtilParams params) throws IOException {
		if (destDir.exists() && !destDir.isDirectory()) {
			throw new IOException(MSG_NOT_A_DIRECTORY + destDir);
		}
		File dest = file(destDir, src.getName());
		copyFile(src, dest, params);
		return dest;
	}


	// ---------------------------------------------------------------- copy dir


	public static void copyDir(String srcDir, String destDir) throws IOException {
		copyDir(file(srcDir), file(destDir), fileUtilParams);
	}

	public static void copyDir(String srcDir, String destDir, FileUtilParams params) throws IOException {
		copyDir(file(srcDir), file(destDir), params);
	}

	public static void copyDir(File srcDir, File destDir) throws IOException {
		copyDir(srcDir, destDir, fileUtilParams);
	}

	/**
	 * Copies directory with specified copy params.
	 */
	public static void copyDir(File srcDir, File destDir, FileUtilParams params) throws IOException {
		checkDirCopy(srcDir, destDir);
		doCopyDirectory(srcDir, destDir, params);
	}

	private static void checkDirCopy(File srcDir, File destDir) throws IOException {
		if (!srcDir.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + srcDir);
		}
		if (!srcDir.isDirectory()) {
			throw new IOException(MSG_NOT_A_DIRECTORY + srcDir);
		}
		if (equals(srcDir, destDir)) {
			throw new IOException("Source '" + srcDir + "' and destination '" + destDir + "' are equal");
		}
	}

	private static void doCopyDirectory(File srcDir, File destDir, FileUtilParams params) throws IOException {
		if (destDir.exists()) {
			if (!destDir.isDirectory()) {
				throw new IOException(MSG_NOT_A_DIRECTORY + destDir);
			}
		} else {
			if (!params.createDirs) {
				throw new IOException(MSG_NOT_FOUND + destDir);
			}
			if (!destDir.mkdirs()) {
				throw new IOException(MSG_CANT_CREATE + destDir);
			}
			if (params.preserveDate) {
				destDir.setLastModified(srcDir.lastModified());
			}
		}

		File[] files = srcDir.listFiles();
		if (files == null) {
			throw new IOException("Failed to list contents of: " + srcDir);
		}

		IOException exception = null;
		for (File file : files) {
			File destFile = file(destDir, file.getName());
			try {
				if (file.isDirectory()) {
					if (params.recursive) {
						doCopyDirectory(file, destFile, params);
					}
				} else {
					doCopyFile(file, destFile, params);
				}
			} catch (IOException ioex) {
				if (params.continueOnError) {
					exception = ioex;
					continue;
				}
				throw ioex;
			}
		}

		if (exception != null) {
			throw exception;
		}
	}



	// ---------------------------------------------------------------- move file

	public static File moveFile(String src, String dest) throws IOException {
		return moveFile(file(src), file(dest), fileUtilParams);
	}

	public static File moveFile(String src, String dest, FileUtilParams params) throws IOException {
		return moveFile(file(src), file(dest), params);
	}

	public static File moveFile(File src, File dest) throws IOException {
		return moveFile(src, dest, fileUtilParams);
	}

	public static File moveFile(File src, File dest, FileUtilParams params) throws IOException {
		checkFileCopy(src, dest, params);
		doMoveFile(src, dest, params);
		return dest;
	}

	private static void doMoveFile(File src, File dest, FileUtilParams params) throws IOException {
		if (dest.exists()) {
			if (!dest.isFile()) {
				throw new IOException(MSG_NOT_A_FILE + dest);
			}
			if (!params.overwrite) {
				throw new IOException(MSG_ALREADY_EXISTS + dest);
			}
			dest.delete();
		}

		final boolean rename = src.renameTo(dest);
		if (!rename) {
			doCopyFile(src, dest, params);
			src.delete();
		}
	}

	// ---------------------------------------------------------------- move file to dir


	public static File moveFileToDir(String src, String destDir) throws IOException {
		return moveFileToDir(file(src), file(destDir), fileUtilParams);
	}
	public static File moveFileToDir(String src, String destDir, FileUtilParams params) throws IOException {
		return moveFileToDir(file(src), file(destDir), params);
	}

	public static File moveFileToDir(File src, File destDir) throws IOException {
		return moveFileToDir(src, destDir, fileUtilParams);
	}
	public static File moveFileToDir(File src, File destDir, FileUtilParams params) throws IOException {
		if (destDir.exists() && !destDir.isDirectory()) {
			throw new IOException(MSG_NOT_A_DIRECTORY + destDir);
		}
		return moveFile(src, file(destDir, src.getName()), params);
	}


	// ---------------------------------------------------------------- move dir

	public static File moveDir(String srcDir, String destDir) throws IOException {
		return moveDir(file(srcDir), file(destDir));
	}
	public static File moveDir(File srcDir, File destDir) throws IOException {
		checkDirCopy(srcDir, destDir);
		doMoveDirectory(srcDir, destDir);
		return destDir;
	}

	private static void doMoveDirectory(File src, File dest) throws IOException {
		if (dest.exists()) {
			if (!dest.isDirectory()) {
				throw new IOException(MSG_NOT_A_DIRECTORY + dest);
			}
			dest = file(dest, dest.getName());
			dest.mkdir();
		}

		final boolean rename = src.renameTo(dest);
		if (!rename) {
			doCopyDirectory(src, dest, params());
			deleteDir(src);
		}
	}

	// ---------------------------------------------------------------- delete file

	public static void deleteFile(String dest) throws IOException {
		deleteFile(file(dest));
	}

	public static void deleteFile(File dest) throws IOException {
		if (!dest.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + dest);
		}
		if (!dest.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + dest);
		}
		if (!dest.delete()) {
			throw new IOException(MSG_UNABLE_TO_DELETE + dest);
		}
	}


	// ---------------------------------------------------------------- delete dir

	public static void deleteDir(String dest) throws IOException {
		deleteDir(file(dest), fileUtilParams);
	}
	public static void deleteDir(String dest, FileUtilParams params) throws IOException {
		deleteDir(file(dest), params);
	}
	public static void deleteDir(File dest) throws IOException {
		deleteDir(dest, fileUtilParams);
	}
	/**
	 * Deletes a directory.
	 */
	public static void deleteDir(File dest, FileUtilParams params) throws IOException {
		cleanDir(dest, params);
		if (!dest.delete()) {
			throw new IOException(MSG_UNABLE_TO_DELETE + dest);
		}
	}



	public static void cleanDir(String dest) throws IOException {
		cleanDir(file(dest), fileUtilParams);
	}
	public static void cleanDir(String dest, FileUtilParams params) throws IOException {
		cleanDir(file(dest), params);
	}
	public static void cleanDir(File dest) throws IOException {
		cleanDir(dest, fileUtilParams);
	}

	/**
	 * Cleans a directory without deleting it.
	 */
	public static void cleanDir(File dest, FileUtilParams params) throws IOException {
		if (!dest.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + dest);
		}

		if (!dest.isDirectory()) {
			throw new IOException(MSG_NOT_A_DIRECTORY + dest);
		}

		File[] files = dest.listFiles();
		if (files == null) {
			throw new IOException("Failed to list contents of: " + dest);
		}

		IOException exception = null;
		for (File file : files) {
			try {
				if (file.isDirectory()) {
					if (params.recursive) {
						deleteDir(file, params);
					}
				} else {
					file.delete();
				}
			} catch (IOException ioex) {
				if (params.continueOnError) {
					exception = ioex;
					continue;
				}
				throw ioex;
			}
		}

		if (exception != null) {
			throw exception;
		}
	}

	// ---------------------------------------------------------------- read/write chars

	public static char[] readUTFChars(String fileName) throws IOException {
		return readUTFChars(file(fileName));
	}
	/**
	 * Reads UTF file content as char array.
	 * @see UnicodeInputStream
	 */
	public static char[] readUTFChars(File file) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		long len = file.length();
		if (len >= Integer.MAX_VALUE) {
			len = Integer.MAX_VALUE;
		}
		UnicodeInputStream in = null;
		try {
			in = new UnicodeInputStream(new FileInputStream(file), null);
			FastCharArrayWriter fastCharArrayWriter = new FastCharArrayWriter((int) len);
			String encoding = in.getDetectedEncoding();
			if (encoding == null) {
				encoding = StringPool.UTF_8;
			}
			StreamUtil.copy(in, fastCharArrayWriter, encoding);
			return fastCharArrayWriter.toCharArray();
		} finally {
			StreamUtil.close(in);
		}
	}

	public static char[] readChars(String fileName) throws IOException {
		return readChars(file(fileName), fileUtilParams.encoding);
	}

	public static char[] readChars(File file) throws IOException {
		return readChars(file, fileUtilParams.encoding);
	}

	public static char[] readChars(String fileName, String encoding) throws IOException {
		return readChars(file(fileName), encoding);
	}

	/**
	 * Reads file content as char array.
	 */
	public static char[] readChars(File file, String encoding) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		long len = file.length();
		if (len >= Integer.MAX_VALUE) {
			len = Integer.MAX_VALUE;
		}

		InputStream in = null;
		try {
			in = new FileInputStream(file);
			if (encoding.startsWith("UTF")) {
				in = new UnicodeInputStream(in, encoding);
			}
			FastCharArrayWriter fastCharArrayWriter = new FastCharArrayWriter((int) len);
			StreamUtil.copy(in, fastCharArrayWriter, encoding);
			return fastCharArrayWriter.toCharArray();
		} finally {
			StreamUtil.close(in);
		}
	}


	public static void writeChars(File dest, char[] data) throws IOException {
		outChars(dest, data, JoddCore.encoding, false);
	}
	public static void writeChars(String dest, char[] data) throws IOException {
		outChars(file(dest), data, JoddCore.encoding, false);
	}

	public static void writeChars(File dest, char[] data, String encoding) throws IOException {
		outChars(dest, data, encoding, false);
	}
	public static void writeChars(String dest, char[] data, String encoding) throws IOException {
		outChars(file(dest), data, encoding, false);
	}
	
	protected static void outChars(File dest, char[] data, String encoding, boolean append) throws IOException {
		if (dest.exists()) {
			if (!dest.isFile()) {
				throw new IOException(MSG_NOT_A_FILE + dest);
			}
		}
		Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(dest, append), encoding));
		try {
			out.write(data);
		} finally {
			StreamUtil.close(out);
		}
	}


	// ---------------------------------------------------------------- read/write string

	public static String readUTFString(String fileName) throws IOException {
		return readUTFString(file(fileName));
	}

	/**
	 * Detects optional BOM and reads UTF string from a file.
	 * If BOM is missing, UTF-8 is assumed.
	 * @see UnicodeInputStream
	 */
	public static String readUTFString(File file) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		long len = file.length();
		if (len >= Integer.MAX_VALUE) {
			len = Integer.MAX_VALUE;
		}
		UnicodeInputStream in = null;
		try {
			in = new UnicodeInputStream(new FileInputStream(file), null);
			FastCharArrayWriter out = new FastCharArrayWriter((int) len);
			String encoding = in.getDetectedEncoding();
			if (encoding == null) {
				encoding = StringPool.UTF_8;
			}
			StreamUtil.copy(in, out, encoding);
			return out.toString();
		} finally {
			StreamUtil.close(in);
		}
	}

	/**
	 * Detects optional BOM and reads UTF string from an input stream.
	 * If BOM is missing, UTF-8 is assumed.
	 */
	public static String readUTFString(InputStream inputStream) throws IOException {
		UnicodeInputStream in = null;
		try {
			in = new UnicodeInputStream(inputStream, null);
			FastCharArrayWriter out = new FastCharArrayWriter();
			String encoding = in.getDetectedEncoding();
			if (encoding == null) {
				encoding = StringPool.UTF_8;
			}
			StreamUtil.copy(in, out, encoding);
			return out.toString();
		} finally {
			StreamUtil.close(in);
		}
	}


	public static String readString(String source) throws IOException {
		return readString(file(source), fileUtilParams.encoding);
	}

	public static String readString(String source, String encoding) throws IOException {
		return readString(file(source), encoding);
	}

	public static String readString(File source) throws IOException {
		return readString(source, fileUtilParams.encoding);
	}

	/**
	 * Reads file content as string encoded in provided encoding.
	 * For UTF encoded files, detects optional BOM characters.
	 */
	public static String readString(File file, String encoding) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		long len = file.length();
		if (len >= Integer.MAX_VALUE) {
			len = Integer.MAX_VALUE;
		}
		InputStream in = null;
		try {
			in = new FileInputStream(file);
			if (encoding.startsWith("UTF")) {
				in = new UnicodeInputStream(in, encoding);
			}
			FastCharArrayWriter out = new FastCharArrayWriter((int) len);
			StreamUtil.copy(in, out, encoding);
			return out.toString();
		} finally {
			StreamUtil.close(in);
		}
	}


	public static void writeString(String dest, String data) throws IOException {
		outString(file(dest), data, fileUtilParams.encoding, false);
	}

	public static void writeString(String dest, String data, String encoding) throws IOException {
		outString(file(dest), data, encoding, false);
	}

	public static void writeString(File dest, String data) throws IOException {
		outString(dest, data, fileUtilParams.encoding, false);
	}

	public static void writeString(File dest, String data, String encoding) throws IOException {
		outString(dest, data, encoding, false);
	}


	public static void appendString(String dest, String data) throws IOException {
		outString(file(dest), data, fileUtilParams.encoding, true);
	}

	public static void appendString(String dest, String data, String encoding) throws IOException {
		outString(file(dest), data, encoding, true);
	}

	public static void appendString(File dest, String data) throws IOException {
		outString(dest, data, fileUtilParams.encoding, true);
	}

	public static void appendString(File dest, String data, String encoding) throws IOException {
		outString(dest, data, encoding, true);
	}

	protected static void outString(File dest, String data, String encoding, boolean append) throws IOException {
		if (dest.exists()) {
			if (!dest.isFile()) {
				throw new IOException(MSG_NOT_A_FILE + dest);
			}
		}
		FileOutputStream out = null;
		try {
			out = new FileOutputStream(dest, append);
			out.write(data.getBytes(encoding));
		} finally {
			StreamUtil.close(out);
		}
	}

	// ---------------------------------------------------------------- stream

	public static void writeStream(File dest, InputStream in) throws IOException {
		FileOutputStream out = null;
		try {
			out = new FileOutputStream(dest);
			StreamUtil.copy(in, out);
		} finally {
			StreamUtil.close(out);
		}
	}

	public static void writeStream(String dest, InputStream in) throws IOException {
		FileOutputStream out = null;
		try {
			out = new FileOutputStream(dest);
			StreamUtil.copy(in, out);
		} finally {
			StreamUtil.close(out);
		}
	}


	// ---------------------------------------------------------------- read/write string lines


	public static String[] readLines(String source) throws IOException {
		return readLines(file(source), fileUtilParams.encoding);
	}
	public static String[] readLines(String source, String encoding) throws IOException {
		return readLines(file(source), encoding);
	}
	public static String[] readLines(File source) throws IOException {
		return readLines(source, fileUtilParams.encoding);
	}

	/**
	 * Reads lines from source files.
	 */
	public static String[] readLines(File file, String encoding) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		List list = new ArrayList<>();

		InputStream in = null;
		try {
			in = new FileInputStream(file);
			if (encoding.startsWith("UTF")) {
				in = new UnicodeInputStream(in, encoding);
			}
			BufferedReader br = new BufferedReader(new InputStreamReader(in, encoding));
			String strLine;
			while ((strLine = br.readLine()) != null)   {
				list.add(strLine);
			}
		} finally {
			StreamUtil.close(in);
		}
		return list.toArray(new String[list.size()]);
	}



	// ---------------------------------------------------------------- read/write bytearray


	public static byte[] readBytes(String file) throws IOException {
		return readBytes(file(file));
	}

	public static byte[] readBytes(File file) throws IOException {
		return readBytes(file, -1);
	}
	public static byte[] readBytes(File file, int fixedLength) throws IOException {
		if (!file.exists()) {
			throw new FileNotFoundException(MSG_NOT_FOUND + file);
		}
		if (!file.isFile()) {
			throw new IOException(MSG_NOT_A_FILE + file);
		}
		long len = file.length();
		if (len >= Integer.MAX_VALUE) {
			throw new IOException("File is larger then max array size");
		}

		if (fixedLength > -1 && fixedLength < len) {
			len = fixedLength;
		}

		byte[] bytes = new byte[(int) len];
		RandomAccessFile randomAccessFile = new RandomAccessFile(file, "r");
		randomAccessFile.readFully(bytes);
		randomAccessFile.close();

		return bytes;
	}



	public static void writeBytes(String dest, byte[] data) throws IOException {
		outBytes(file(dest), data, 0, data.length, false);
	}

	public static void writeBytes(String dest, byte[] data, int off, int len) throws IOException {
		outBytes(file(dest), data, off, len, false);
	}

	public static void writeBytes(File dest, byte[] data) throws IOException {
		outBytes(dest, data, 0, data.length, false);
	}

	public static void writeBytes(File dest, byte[] data, int off, int len) throws IOException {
		outBytes(dest, data, off, len, false);
	}


	public static void appendBytes(String dest, byte[] data) throws IOException {
		outBytes(file(dest), data, 0, data.length, true);
	}

	public static void appendBytes(String dest, byte[] data, int off, int len) throws IOException {
		outBytes(file(dest), data, off, len, true);
	}

	public static void appendBytes(File dest, byte[] data) throws IOException {
		outBytes(dest, data, 0, data.length, true);
	}

	public static void appendBytes(File dest, byte[] data, int off, int len) throws IOException {
		outBytes(dest, data, off, len, true);
	}

	protected static void outBytes(File dest, byte[] data, int off, int len, boolean append) throws IOException {
		if (dest.exists()) {
			if (!dest.isFile()) {
				throw new IOException(MSG_NOT_A_FILE + dest);
			}
		}
		FileOutputStream out = null;
		try {
			out = new FileOutputStream(dest, append);
			out.write(data, off, len);
		} finally {
			StreamUtil.close(out);
		}
	}

	// ---------------------------------------------------------------- equals content

	public static boolean compare(String file1, String file2) throws IOException {
		return compare(file(file1), file(file2));
	}

	/**
	 * Compare the contents of two files to determine if they are equal or
	 * not.
	 * 

* This method checks to see if the two files are different lengths * or if they point to the same file, before resorting to byte-by-byte * comparison of the contents. *

* Code origin: Avalon */ public static boolean compare(File file1, File file2) throws IOException { boolean file1Exists = file1.exists(); if (file1Exists != file2.exists()) { return false; } if (!file1Exists) { return true; } if ((!file1.isFile()) || (!file2.isFile())) { throw new IOException("Only files can be compared"); } if (file1.length() != file2.length()) { return false; } if (equals(file1, file2)) { return true; } InputStream input1 = null; InputStream input2 = null; try { input1 = new FileInputStream(file1); input2 = new FileInputStream(file2); return StreamUtil.compare(input1, input2); } finally { StreamUtil.close(input1); StreamUtil.close(input2); } } // ---------------------------------------------------------------- time public static boolean isNewer(String file, String reference) { return isNewer(file(file), file(reference)); } /** * Test if specified File is newer than the reference File. * * @param file the File of which the modification date must be compared * @param reference the File of which the modification date is used * @return true if the File exists and has been modified more * recently than the reference File. */ public static boolean isNewer(File file, File reference) { if (!reference.exists()) { throw new IllegalArgumentException("Reference file not found: " + reference); } return isNewer(file, reference.lastModified()); } public static boolean isOlder(String file, String reference) { return isOlder(file(file), file(reference)); } public static boolean isOlder(File file, File reference) { if (!reference.exists()) { throw new IllegalArgumentException("Reference file not found: " + reference); } return isOlder(file, reference.lastModified()); } /** * Tests if the specified File is newer than the specified time reference. * * @param file the File of which the modification date must be compared. * @param timeMillis the time reference measured in milliseconds since the * epoch (00:00:00 GMT, January 1, 1970) * @return true if the File exists and has been modified after * the given time reference. */ public static boolean isNewer(File file, long timeMillis) { if (!file.exists()) { return false; } return file.lastModified() > timeMillis; } public static boolean isNewer(String file, long timeMillis) { return isNewer(file(file), timeMillis); } public static boolean isOlder(File file, long timeMillis) { if (!file.exists()) { return false; } return file.lastModified() < timeMillis; } public static boolean isOlder(String file, long timeMillis) { return isOlder(file(file), timeMillis); } // ---------------------------------------------------------------- smart copy public static void copy(String src, String dest) throws IOException { copy(file(src), file(dest), fileUtilParams); } public static void copy(String src, String dest, FileUtilParams params) throws IOException { copy(file(src), file(dest), params); } public static void copy(File src, File dest) throws IOException { copy(src, dest, fileUtilParams); } /** * Smart copy. If source is a directory, copy it to destination. * Otherwise, if destination is directory, copy source file to it. * Otherwise, try to copy source file to destination file. */ public static void copy(File src, File dest, FileUtilParams params) throws IOException { if (src.isDirectory()) { copyDir(src, dest, params); return; } if (dest.isDirectory()) { copyFileToDir(src, dest, params); return; } copyFile(src, dest, params); } // ---------------------------------------------------------------- smart move public static void move(String src, String dest) throws IOException { move(file(src), file(dest), fileUtilParams); } public static void move(String src, String dest, FileUtilParams params) throws IOException { move(file(src), file(dest), params); } public static void move(File src, File dest) throws IOException { move(src, dest, fileUtilParams); } /** * Smart move. If source is a directory, move it to destination. * Otherwise, if destination is directory, move source file to it. * Otherwise, try to move source file to destination file. */ public static void move(File src, File dest, FileUtilParams params) throws IOException { if (src.isDirectory()) { moveDir(src, dest); return; } if (dest.isDirectory()) { moveFileToDir(src, dest, params); return; } moveFile(src, dest, params); } // ---------------------------------------------------------------- smart delete public static void delete(String dest) throws IOException { delete(file(dest), fileUtilParams); } public static void delete(String dest, FileUtilParams params) throws IOException { delete(file(dest), params); } public static void delete(File dest) throws IOException { delete(dest, fileUtilParams); } /** * Smart delete of destination file or directory. */ public static void delete(File dest, FileUtilParams params) throws IOException { if (dest.isDirectory()) { deleteDir(dest, params); return; } deleteFile(dest); } // ---------------------------------------------------------------- misc /** * Check if one file is an ancestor of second one. * * @param strict if false then this method returns true if ancestor * and file are equal * @return true if ancestor is parent of file; false otherwise */ public static boolean isAncestor(File ancestor, File file, boolean strict) { File parent = strict ? getParentFile(file) : file; while (true) { if (parent == null) { return false; } if (parent.equals(ancestor)) { return true; } parent = getParentFile(parent); } } /** * Returns parent for the file. The method correctly * processes "." and ".." in file names. The name * remains relative if was relative before. * Returns null if the file has no parent. */ public static File getParentFile(final File file) { int skipCount = 0; File parentFile = file; while (true) { parentFile = parentFile.getParentFile(); if (parentFile == null) { return null; } if (StringPool.DOT.equals(parentFile.getName())) { continue; } if (StringPool.DOTDOT.equals(parentFile.getName())) { skipCount++; continue; } if (skipCount > 0) { skipCount--; continue; } return parentFile; } } public static boolean isFilePathAcceptable(File file, FileFilter fileFilter) { do { if (fileFilter != null && !fileFilter.accept(file)) { return false; } file = file.getParentFile(); } while (file != null); return true; } // ---------------------------------------------------------------- temp public static File createTempDirectory() throws IOException { return createTempDirectory(JoddCore.tempFilePrefix, null, null); } /** * Creates temporary directory. */ public static File createTempDirectory(String prefix, String suffix) throws IOException { return createTempDirectory(prefix, suffix, null); } /** * Creates temporary directory. */ public static File createTempDirectory(String prefix, String suffix, File tempDir) throws IOException { File file = createTempFile(prefix, suffix, tempDir); file.delete(); file.mkdir(); return file; } /** * Simple method that creates temp file. */ public static File createTempFile() throws IOException { return createTempFile(JoddCore.tempFilePrefix, null, null, true); } /** * Creates temporary file. * If create is set to true file will be * physically created on the file system. Otherwise, it will be created and then * deleted - trick that will make temp file exist only if they are used. */ public static File createTempFile(String prefix, String suffix, File tempDir, boolean create) throws IOException { File file = createTempFile(prefix, suffix, tempDir); file.delete(); if (create) { file.createNewFile(); } return file; } /** * Creates temporary file. Wraps java method and repeat creation several time * if something fail. */ public static File createTempFile(String prefix, String suffix, File dir) throws IOException { int exceptionsCount = 0; while (true) { try { return File.createTempFile(prefix, suffix, dir).getCanonicalFile(); } catch (IOException ioex) { // fixes java.io.WinNTFileSystem.createFileExclusively access denied if (++exceptionsCount >= 50) { throw ioex; } } } } // ---------------------------------------------------------------- symlink /** * Determines whether the specified file is a symbolic link rather than an actual file. * Always returns false on Windows. */ public static boolean isSymlink(final File file) throws IOException { if (SystemUtil.isHostWindows()) { return false; } File fileInCanonicalDir; if (file.getParent() == null) { fileInCanonicalDir = file; } else { File canonicalDir = file.getParentFile().getCanonicalFile(); fileInCanonicalDir = new File(canonicalDir, file.getName()); } return !fileInCanonicalDir.getCanonicalFile().equals(fileInCanonicalDir.getAbsoluteFile()); } // ---------------------------------------------------------------- digests /** * Calculates digest for a file using provided algorithm. */ public static byte[] digest(final File file, MessageDigest algorithm) throws IOException { algorithm.reset(); FileInputStream fis = new FileInputStream(file); BufferedInputStream bis = new BufferedInputStream(fis); DigestInputStream dis = new DigestInputStream(bis, algorithm); try { while (dis.read() != -1) { } } finally { StreamUtil.close(fis); } return algorithm.digest(); } /** * Creates MD5 digest of a file. */ public static String md5(final File file) throws IOException { MessageDigest md5Digest = null; try { md5Digest = MessageDigest.getInstance("MD5"); } catch (NoSuchAlgorithmException ignore) { } byte[] digest = digest(file, md5Digest); return StringUtil.toHexString(digest); } /** * Creates SHA-1 digest of a file. */ public static String sha(final File file) throws IOException { MessageDigest md5Digest = null; try { md5Digest = MessageDigest.getInstance("SHA-1"); } catch (NoSuchAlgorithmException ignore) { } byte[] digest = digest(file, md5Digest); return StringUtil.toHexString(digest); } /** * Creates SHA-256 digest of a file. */ public static String sha256(final File file) throws IOException { MessageDigest md5Digest = null; try { md5Digest = MessageDigest.getInstance("SHA-256"); } catch (NoSuchAlgorithmException ignore) { } byte[] digest = digest(file, md5Digest); return StringUtil.toHexString(digest); } /** * Checks the start of the file for ASCII control characters */ public static boolean isBinary(final File file) throws IOException { byte[] bytes = readBytes(file, 128); for (byte b : bytes) { if (b < 32) { return true; } } return false; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy