|  | // Copyright 2014 The Bazel Authors. All rights reserved. | 
|  | // | 
|  | // 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 com.google.devtools.build.lib.vfs; | 
|  |  | 
|  | import static java.nio.charset.StandardCharsets.ISO_8859_1; | 
|  |  | 
|  | import com.google.common.base.Preconditions; | 
|  | import com.google.common.io.ByteSink; | 
|  | import com.google.common.io.ByteSource; | 
|  | import com.google.common.io.ByteStreams; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.ConditionallyThreadSafe; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; | 
|  | import java.io.IOException; | 
|  | import java.io.InputStream; | 
|  | import java.io.OutputStream; | 
|  | import java.nio.charset.Charset; | 
|  | import java.util.ArrayList; | 
|  | import java.util.Arrays; | 
|  | import java.util.Collection; | 
|  | import java.util.List; | 
|  | import java.util.function.Predicate; | 
|  |  | 
|  | /** Helper functions that implement often-used complex operations on file systems. */ | 
|  | @ConditionallyThreadSafe | 
|  | public class FileSystemUtils { | 
|  |  | 
|  | private FileSystemUtils() {} | 
|  |  | 
|  | /** | 
|  | * Throws exceptions if {@code baseName} is not a valid base name. A valid | 
|  | * base name: | 
|  | * <ul> | 
|  | * <li>Is not null | 
|  | * <li>Is not an empty string | 
|  | * <li>Is not "." or ".." | 
|  | * <li>Does not contain a slash | 
|  | * </ul> | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static void checkBaseName(String baseName) { | 
|  | if (baseName.length() == 0) { | 
|  | throw new IllegalArgumentException("Child must not be empty string ('')"); | 
|  | } | 
|  | if (baseName.equals(".") || baseName.equals("..")) { | 
|  | throw new IllegalArgumentException("baseName must not be '" + baseName + "'"); | 
|  | } | 
|  | if (baseName.indexOf('/') != -1) { | 
|  | throw new IllegalArgumentException("baseName must not contain a slash: '" + baseName + "'"); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the common ancestor between two paths, or null if none (including | 
|  | * if they are on different filesystems). | 
|  | */ | 
|  | public static Path commonAncestor(Path a, Path b) { | 
|  | while (a != null && !b.startsWith(a)) { | 
|  | a = a.getParentDirectory();  // returns null at root | 
|  | } | 
|  | return a; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the longest common ancestor of the two path fragments, or either "/" or "" (depending | 
|  | * on whether {@code a} is absolute or relative) if there is none. | 
|  | */ | 
|  | public static PathFragment commonAncestor(PathFragment a, PathFragment b) { | 
|  | while (a != null && !b.startsWith(a)) { | 
|  | a = a.getParentDirectory(); | 
|  | } | 
|  |  | 
|  | return a; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a path fragment from a given from-dir to a given to-path. | 
|  | */ | 
|  | public static PathFragment relativePath(PathFragment fromDir, PathFragment to) { | 
|  | if (to.equals(fromDir)) { | 
|  | return PathFragment.EMPTY_FRAGMENT; | 
|  | } | 
|  | if (to.startsWith(fromDir)) { | 
|  | return to.relativeTo(fromDir);  // easy case--it's a descendant | 
|  | } | 
|  | PathFragment ancestor = commonAncestor(fromDir, to); | 
|  | if (ancestor == null) { | 
|  | return to;  // no common ancestor, use 'to' | 
|  | } | 
|  | int levels = fromDir.relativeTo(ancestor).segmentCount(); | 
|  | StringBuilder dotdots = new StringBuilder(); | 
|  | for (int i = 0; i < levels; i++) { | 
|  | dotdots.append("../"); | 
|  | } | 
|  | return PathFragment.create(dotdots.toString()).getRelative(to.relativeTo(ancestor)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Removes the shortest suffix beginning with '.' from the basename of the | 
|  | * filename string. If the basename contains no '.', the filename is returned | 
|  | * unchanged. | 
|  | * | 
|  | * <p>e.g. "foo/bar.x" -> "foo/bar" | 
|  | * | 
|  | * <p>Note that if the filename is composed entirely of ".", this method will return the string | 
|  | * with one fewer ".", which may have surprising effects. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static String removeExtension(String filename) { | 
|  | int lastDotIndex = filename.lastIndexOf('.'); | 
|  | if (lastDotIndex == -1) { return filename; } | 
|  | int lastSlashIndex = filename.lastIndexOf('/'); | 
|  | if (lastSlashIndex > lastDotIndex) { | 
|  | return filename; | 
|  | } | 
|  | return filename.substring(0, lastDotIndex); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Removes the shortest suffix beginning with '.' from the basename of the | 
|  | * PathFragment. If the basename contains no '.', the filename is returned | 
|  | * unchanged. | 
|  | * | 
|  | * <p>e.g. "foo/bar.x" -> "foo/bar" | 
|  | * | 
|  | * <p>Note that if the base filename is composed entirely of ".", this method will return the | 
|  | * filename with one fewer "." in the base filename, which may have surprising effects. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static PathFragment removeExtension(PathFragment path) { | 
|  | return path.replaceName(removeExtension(path.getBaseName())); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Removes the shortest suffix beginning with '.' from the basename of the | 
|  | * Path. If the basename contains no '.', the filename is returned | 
|  | * unchanged. | 
|  | * | 
|  | * <p>e.g. "foo/bar.x" -> "foo/bar" | 
|  | * | 
|  | * <p>Note that if the base filename is composed entirely of ".", this method will return the | 
|  | * filename with one fewer "." in the base filename, which may have surprising effects. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static Path removeExtension(Path path) { | 
|  | return path.getFileSystem().getPath(removeExtension(path.asFragment())); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a new {@code PathFragment} formed by replacing the extension of the | 
|  | * last path segment of {@code path} with {@code newExtension}. Null is | 
|  | * returned iff {@code path} has zero segments. | 
|  | */ | 
|  | public static PathFragment replaceExtension(PathFragment path, String newExtension) { | 
|  | return path.replaceName(removeExtension(path.getBaseName()) + newExtension); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a new {@code PathFragment} formed by replacing the extension of the | 
|  | * last path segment of {@code path} with {@code newExtension}. Null is | 
|  | * returned iff {@code path} has zero segments or it doesn't end with {@code oldExtension}. | 
|  | */ | 
|  | public static PathFragment replaceExtension(PathFragment path, String newExtension, | 
|  | String oldExtension) { | 
|  | String base = path.getBaseName(); | 
|  | if (!base.endsWith(oldExtension)) { | 
|  | return null; | 
|  | } | 
|  | String newBase = base.substring(0, base.length() - oldExtension.length()) + newExtension; | 
|  | return path.replaceName(newBase); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a new {@code Path} formed by replacing the extension of the | 
|  | * last path segment of {@code path} with {@code newExtension}. Null is | 
|  | * returned iff {@code path} has zero segments. | 
|  | */ | 
|  | public static Path replaceExtension(Path path, String newExtension) { | 
|  | PathFragment fragment = replaceExtension(path.asFragment(), newExtension); | 
|  | return fragment == null ? null : path.getFileSystem().getPath(fragment); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a new {@code PathFragment} formed by adding the extension to the last path segment of | 
|  | * {@code path}. Null is returned if {@code path} has zero segments. | 
|  | */ | 
|  | public static PathFragment appendExtension(PathFragment path, String newExtension) { | 
|  | return path.replaceName(path.getBaseName() + newExtension); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a new {@code PathFragment} formed by appending the given string to the last path | 
|  | * segment of {@code path} without removing the extension.  Returns null if {@code path} | 
|  | * has zero segments. | 
|  | */ | 
|  | public static PathFragment appendWithoutExtension(PathFragment path, String toAppend) { | 
|  | return path.replaceName(appendWithoutExtension(path.getBaseName(), toAppend)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Given a string that represents a file with an extension separated by a '.' and a string | 
|  | * to append, return a string in which {@code toAppend} has been appended to {@code name} | 
|  | * before the last '.' character.  If {@code name} does not include a '.', appends {@code | 
|  | * toAppend} at the end. | 
|  | * | 
|  | * <p>For example, | 
|  | * ("libfoo.jar", "-src") ==> "libfoo-src.jar" | 
|  | * ("libfoo", "-src") ==> "libfoo-src" | 
|  | */ | 
|  | private static String appendWithoutExtension(String name, String toAppend) { | 
|  | int dotIndex = name.lastIndexOf('.'); | 
|  | if (dotIndex > 0) { | 
|  | String baseName = name.substring(0, dotIndex); | 
|  | String extension = name.substring(dotIndex); | 
|  | return baseName + toAppend + extension; | 
|  | } else { | 
|  | return name + toAppend; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Return the current working directory as expressed by the System property | 
|  | * 'user.dir'. | 
|  | */ | 
|  | public static Path getWorkingDirectory(FileSystem fs) { | 
|  | return fs.getPath(getWorkingDirectory()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the current working directory as expressed by the System property | 
|  | * 'user.dir'. This version does not require a {@link FileSystem}. | 
|  | */ | 
|  | public static PathFragment getWorkingDirectory() { | 
|  | return PathFragment.create(System.getProperty("user.dir", "/")); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * "Touches" the file or directory specified by the path, following symbolic | 
|  | * links. If it does not exist, it is created as an empty file; otherwise, the | 
|  | * time of last access is updated to the current time. | 
|  | * | 
|  | * @throws IOException if there was an error while touching the file | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static void touchFile(Path path) throws IOException { | 
|  | if (path.exists()) { | 
|  | // -1L means "use the current time", and is ultimately implemented by | 
|  | // utime(path, null), thereby using the kernel's clock, not the JVM's. | 
|  | // (A previous implementation based on the JVM clock was found to be | 
|  | // skewy.) | 
|  | path.setLastModifiedTime(-1L); | 
|  | } else { | 
|  | createEmptyFile(path); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates an empty regular file with the name of the current path, following | 
|  | * symbolic links. | 
|  | * | 
|  | * @throws IOException if the file could not be created for any reason | 
|  | *         (including that there was already a file at that location) | 
|  | */ | 
|  | public static void createEmptyFile(Path path) throws IOException { | 
|  | path.getOutputStream().close(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates or updates a symbolic link from 'link' to 'target'. Replaces | 
|  | * existing symbolic links with target, and skips the link creation if it is | 
|  | * already present. Will also create any missing ancestor directories of the | 
|  | * link. This method is non-atomic | 
|  | * | 
|  | * <p>Note: this method will throw an IOException if there is an unequal | 
|  | * non-symlink at link. | 
|  | * | 
|  | * @throws IOException if the creation of the symbolic link was unsuccessful | 
|  | *         for any reason. | 
|  | */ | 
|  | @ThreadSafe  // but not atomic | 
|  | public static void ensureSymbolicLink(Path link, Path target) throws IOException { | 
|  | ensureSymbolicLink(link, target.asFragment()); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates or updates a symbolic link from 'link' to 'target'. Replaces | 
|  | * existing symbolic links with target, and skips the link creation if it is | 
|  | * already present. Will also create any missing ancestor directories of the | 
|  | * link. This method is non-atomic | 
|  | * | 
|  | * <p>Note: this method will throw an IOException if there is an unequal | 
|  | * non-symlink at link. | 
|  | * | 
|  | * @throws IOException if the creation of the symbolic link was unsuccessful | 
|  | *         for any reason. | 
|  | */ | 
|  | @ThreadSafe  // but not atomic | 
|  | public static void ensureSymbolicLink(Path link, String target) throws IOException { | 
|  | ensureSymbolicLink(link, PathFragment.create(target)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates or updates a symbolic link from 'link' to 'target'. Replaces | 
|  | * existing symbolic links with target, and skips the link creation if it is | 
|  | * already present. Will also create any missing ancestor directories of the | 
|  | * link. This method is non-atomic | 
|  | * | 
|  | * <p>Note: this method will throw an IOException if there is an unequal | 
|  | * non-symlink at link. | 
|  | * | 
|  | * @throws IOException if the creation of the symbolic link was unsuccessful | 
|  | *         for any reason. | 
|  | */ | 
|  | @ThreadSafe  // but not atomic | 
|  | public static void ensureSymbolicLink(Path link, PathFragment target) throws IOException { | 
|  | // TODO(bazel-team): (2009) consider adding the logic for recovering from the case when | 
|  | // we have already created a parent directory symlink earlier. | 
|  | try { | 
|  | if (link.readSymbolicLink().equals(target)) { | 
|  | return;  // Do nothing if the link is already there. | 
|  | } | 
|  | } catch (IOException e) { // link missing or broken | 
|  | /* fallthru and do the work below */ | 
|  | } | 
|  | if (link.isSymbolicLink()) { | 
|  | link.delete(); // Remove the symlink since it is pointing somewhere else. | 
|  | } else { | 
|  | createDirectoryAndParents(link.getParentDirectory()); | 
|  | } | 
|  | try { | 
|  | link.createSymbolicLink(target); | 
|  | } catch (IOException e) { | 
|  | // Only pass on exceptions caused by a true link creation failure. | 
|  | if (!link.isSymbolicLink() || | 
|  | !link.resolveSymbolicLinks().equals(link.getRelative(target))) { | 
|  | throw e; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | public static ByteSource asByteSource(final Path path) { | 
|  | return new ByteSource() { | 
|  | @Override public InputStream openStream() throws IOException { | 
|  | return path.getInputStream(); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | public static ByteSink asByteSink(final Path path, final boolean append) { | 
|  | return new ByteSink() { | 
|  | @Override public OutputStream openStream() throws IOException { | 
|  | return path.getOutputStream(append); | 
|  | } | 
|  | }; | 
|  | } | 
|  |  | 
|  | public static ByteSink asByteSink(final Path path) { | 
|  | return asByteSink(path, false); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Copies the file from location "from" to location "to", while overwriting a | 
|  | * potentially existing "to". File's last modified time, executable and | 
|  | * writable bits are also preserved. | 
|  | * | 
|  | * <p>If no error occurs, the method returns normally. If a parent directory does | 
|  | * not exist, a FileNotFoundException is thrown. An IOException is thrown when | 
|  | * other erroneous situations occur. (e.g. read errors) | 
|  | */ | 
|  | @ThreadSafe  // but not atomic | 
|  | public static void copyFile(Path from, Path to) throws IOException { | 
|  | try { | 
|  | to.delete(); | 
|  | } catch (IOException e) { | 
|  | throw new IOException("error copying file: " | 
|  | + "couldn't delete destination: " + e.getMessage()); | 
|  | } | 
|  | try (InputStream in = from.getInputStream(); | 
|  | OutputStream out = to.getOutputStream()) { | 
|  | ByteStreams.copy(in, out); | 
|  | } | 
|  | to.setLastModifiedTime(from.getLastModifiedTime()); // Preserve mtime. | 
|  | if (!from.isWritable()) { | 
|  | to.setWritable(false); // Make file read-only if original was read-only. | 
|  | } | 
|  | to.setExecutable(from.isExecutable()); // Copy executable bit. | 
|  | } | 
|  |  | 
|  | /** Describes the behavior of a {@link #moveFile(Path, Path)} operation. */ | 
|  | public enum MoveResult { | 
|  | /** The file was moved at the file system level. */ | 
|  | FILE_MOVED, | 
|  |  | 
|  | /** The file had to be copied and then deleted because the move failed. */ | 
|  | FILE_COPIED, | 
|  | } | 
|  |  | 
|  | /** | 
|  | * copyLargeBuffer is a replacement for ByteStreams.copy which uses a larger buffer. Increasing | 
|  | * the buffer size is a performance improvement when copying from/to FUSE file systems, where | 
|  | * individual requests are more costly, but can also be larger. | 
|  | */ | 
|  | private static long copyLargeBuffer(InputStream from, OutputStream to) throws IOException { | 
|  | byte[] buf = new byte[131072]; | 
|  | long total = 0; | 
|  | while (true) { | 
|  | int r = from.read(buf); | 
|  | if (r == -1) { | 
|  | break; | 
|  | } | 
|  | to.write(buf, 0, r); | 
|  | total += r; | 
|  | } | 
|  | return total; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Moves the file from location "from" to location "to", while overwriting a potentially existing | 
|  | * "to". If "from" is a regular file, its last modified time, executable and writable bits are | 
|  | * also preserved. Symlinks are also supported but not directories or special files. | 
|  | * | 
|  | * <p>If the move fails (usually because the "from" and "to" live in different file systems), this | 
|  | * falls back to copying the file. Note that these two operations have very different performance | 
|  | * characteristics and is why this operation reports back to the caller what actually happened. | 
|  | * | 
|  | * <p>If no error occurs, the method returns normally. If a parent directory does not exist, a | 
|  | * FileNotFoundException is thrown. {@link IOException} is thrown when other erroneous situations | 
|  | * occur. (e.g. read errors) | 
|  | * | 
|  | * @param from location of the file to move | 
|  | * @param to destination to where to move the file | 
|  | * @return a description of how the move was performed | 
|  | * @throws IOException if the move fails | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static MoveResult moveFile(Path from, Path to) throws IOException { | 
|  | // We don't try-catch here for better performance. | 
|  | to.delete(); | 
|  | try { | 
|  | from.renameTo(to); | 
|  | return MoveResult.FILE_MOVED; | 
|  | } catch (IOException e) { | 
|  | // Fallback to a copy. | 
|  | FileStatus stat = from.stat(Symlinks.NOFOLLOW); | 
|  | if (stat.isFile()) { | 
|  | try (InputStream in = from.getInputStream(); | 
|  | OutputStream out = to.getOutputStream()) { | 
|  | copyLargeBuffer(in, out); | 
|  | } | 
|  | to.setLastModifiedTime(stat.getLastModifiedTime()); // Preserve mtime. | 
|  | if (!from.isWritable()) { | 
|  | to.setWritable(false); // Make file read-only if original was read-only. | 
|  | } | 
|  | to.setExecutable(from.isExecutable()); // Copy executable bit. | 
|  | } else if (stat.isSymbolicLink()) { | 
|  | to.createSymbolicLink(from.readSymbolicLink()); | 
|  | } else { | 
|  | throw new IOException("Don't know how to copy " + from); | 
|  | } | 
|  | if (!from.delete()) { | 
|  | if (!to.delete()) { | 
|  | throw new IOException("Unable to delete " + to); | 
|  | } | 
|  | throw new IOException("Unable to delete " + from); | 
|  | } | 
|  | return MoveResult.FILE_COPIED; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Copies a tool binary from one path to another, returning the target path. | 
|  | * The directory of the target path must already exist. The target copy's time | 
|  | * is set to match, as well as its read-only and executable flags. The | 
|  | * operation is skipped if the target file has the same time and size as the | 
|  | * source. | 
|  | */ | 
|  | public static Path copyTool(Path source, Path target) throws IOException { | 
|  | FileStatus sourceStat = null; | 
|  | FileStatus targetStat = target.statNullable(); | 
|  | if (targetStat != null) { | 
|  | // stat the source file only if we'll need the stat. | 
|  | sourceStat = source.stat(Symlinks.FOLLOW); | 
|  | } | 
|  | if (targetStat == null || | 
|  | targetStat.getLastModifiedTime() != sourceStat.getLastModifiedTime() || | 
|  | targetStat.getSize() != sourceStat.getSize()) { | 
|  | copyFile(source, target); | 
|  | target.setWritable(source.isWritable()); | 
|  | target.setExecutable(source.isExecutable()); | 
|  | target.setLastModifiedTime(source.getLastModifiedTime()); | 
|  | } | 
|  | return target; | 
|  | } | 
|  |  | 
|  | /* Directory tree operations. */ | 
|  |  | 
|  | /** | 
|  | * Returns a new collection containing all of the paths below a given root path, for which the | 
|  | * given predicate is true. Symbolic links are not followed, and may appear in the result. | 
|  | * | 
|  | * @throws IOException If the root does not denote a directory | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static Collection<Path> traverseTree(Path root, Predicate<Path> predicate) | 
|  | throws IOException { | 
|  | List<Path> paths = new ArrayList<>(); | 
|  | traverseTree(paths, root, predicate); | 
|  | return paths; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Populates an existing Path List, adding all of the paths below a given root path for which the | 
|  | * given predicate is true. Symbolic links are not followed, and may appear in the result. | 
|  | * | 
|  | * @throws IOException If the root does not denote a directory | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static void traverseTree(Collection<Path> paths, Path root, Predicate<Path> predicate) | 
|  | throws IOException { | 
|  | for (Path p : root.getDirectoryEntries()) { | 
|  | if (predicate.test(p)) { | 
|  | paths.add(p); | 
|  | } | 
|  | if (p.isDirectory(Symlinks.NOFOLLOW)) { | 
|  | traverseTree(paths, p, predicate); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Copies all dir trees under a given 'from' dir to location 'to', while overwriting all files in | 
|  | * the potentially existing 'to'. Resolves symbolic links if {@code followSymlinks == | 
|  | * Symlinks#FOLLOW}. Otherwise copies symlinks as-is. | 
|  | * | 
|  | * <p>The source and the destination must be non-overlapping, otherwise an | 
|  | * IllegalArgumentException will be thrown. This method cannot be used to copy a dir tree to a sub | 
|  | * tree of itself. | 
|  | * | 
|  | * <p>If no error occurs, the method returns normally. If the given 'from' does not exist, a | 
|  | * FileNotFoundException is thrown. An IOException is thrown when other erroneous situations | 
|  | * occur. (e.g. read errors) | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static void copyTreesBelow(Path from, Path to, Symlinks followSymlinks) | 
|  | throws IOException { | 
|  | if (to.startsWith(from)) { | 
|  | throw new IllegalArgumentException(to + " is a subdirectory of " + from); | 
|  | } | 
|  |  | 
|  | Collection<Path> entries = from.getDirectoryEntries(); | 
|  | for (Path entry : entries) { | 
|  | Path toPath = to.getChild(entry.getBaseName()); | 
|  | if (!followSymlinks.toBoolean() && entry.isSymbolicLink()) { | 
|  | FileSystemUtils.ensureSymbolicLink(toPath, entry.readSymbolicLink()); | 
|  | } else if (entry.isFile()) { | 
|  | copyFile(entry, toPath); | 
|  | } else { | 
|  | toPath.createDirectory(); | 
|  | copyTreesBelow(entry, toPath, followSymlinks); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Moves all dir trees under a given 'from' dir to location 'to', while overwriting | 
|  | * all files in the potentially existing 'to'. Doesn't resolve symbolic links. | 
|  | * | 
|  | * <p>The source and the destination must be non-overlapping, otherwise an | 
|  | * IllegalArgumentException will be thrown. This method cannot be used to copy | 
|  | * a dir tree to a sub tree of itself. | 
|  | * | 
|  | * <p>If no error occurs, the method returns normally. If the given 'from' does | 
|  | * not exist, a FileNotFoundException is thrown. An IOException is thrown when | 
|  | * other erroneous situations occur. (e.g. read errors) | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static void moveTreesBelow(Path from , Path to) throws IOException { | 
|  | if (to.startsWith(from)) { | 
|  | throw new IllegalArgumentException(to + " is a subdirectory of " + from); | 
|  | } | 
|  |  | 
|  | Collection<Path> entries = from.getDirectoryEntries(); | 
|  | for (Path entry : entries) { | 
|  | if (entry.isDirectory(Symlinks.NOFOLLOW)) { | 
|  | Path subDir = to.getChild(entry.getBaseName()); | 
|  | subDir.createDirectory(); | 
|  | moveTreesBelow(entry, subDir); | 
|  | } else { | 
|  | Path newEntry = to.getChild(entry.getBaseName()); | 
|  | moveFile(entry, newEntry); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Attempts to create a directory with the name of the given path, creating ancestors as | 
|  | * necessary. | 
|  | * | 
|  | * <p>Deprecated. Prefer to call {@link Path#createDirectoryAndParents()} directly. | 
|  | */ | 
|  | @Deprecated | 
|  | @ThreadSafe | 
|  | public static void createDirectoryAndParents(Path dir) throws IOException { | 
|  | dir.createDirectoryAndParents(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Attempts to remove a relative chain of directories under a given base. | 
|  | * Returns {@code true} if the removal was successful, and returns {@code | 
|  | * false} if the removal fails because a directory was not empty. An | 
|  | * {@link IOException} is thrown for any other errors. | 
|  | */ | 
|  | @ThreadSafe | 
|  | public static boolean removeDirectoryAndParents(Path base, PathFragment toRemove) { | 
|  | if (toRemove.isAbsolute()) { | 
|  | return false; | 
|  | } | 
|  | try { | 
|  | for (; toRemove.segmentCount() > 0; toRemove = toRemove.getParentDirectory()) { | 
|  | Path p = base.getRelative(toRemove); | 
|  | if (p.exists()) { | 
|  | p.delete(); | 
|  | } | 
|  | } | 
|  | } catch (IOException e) { | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Decodes the given byte array assumed to be encoded with ISO-8859-1 encoding (isolatin1). | 
|  | */ | 
|  | public static char[] convertFromLatin1(byte[] content) { | 
|  | char[] latin1 = new char[content.length]; | 
|  | for (int i = 0; i < latin1.length; i++) { // yeah, latin1 is this easy! :-) | 
|  | latin1[i] = (char) (0xff & content[i]); | 
|  | } | 
|  | return latin1; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes lines to file using ISO-8859-1 encoding (isolatin1). | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void writeIsoLatin1(Path file, String... lines) throws IOException { | 
|  | writeLinesAs(file, ISO_8859_1, lines); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Append lines to file using ISO-8859-1 encoding (isolatin1). | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void appendIsoLatin1(Path file, String... lines) throws IOException { | 
|  | appendLinesAs(file, ISO_8859_1, lines); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes the specified String as ISO-8859-1 (latin1) encoded bytes to the | 
|  | * file. Follows symbolic links. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static void writeContentAsLatin1(Path outputFile, String content) throws IOException { | 
|  | writeContent(outputFile, ISO_8859_1, content); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes the specified String using the specified encoding to the file. | 
|  | * Follows symbolic links. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static void writeContent(Path outputFile, Charset charset, String content) | 
|  | throws IOException { | 
|  | asByteSink(outputFile).asCharSink(charset).write(content); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes lines to file using the given encoding, ending every line with a | 
|  | * line break '\n' character. | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void writeLinesAs(Path file, Charset charset, String... lines) | 
|  | throws IOException { | 
|  | writeLinesAs(file, charset, Arrays.asList(lines)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Appends lines to file using the given encoding, ending every line with a | 
|  | * line break '\n' character. | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void appendLinesAs(Path file, Charset charset, String... lines) | 
|  | throws IOException { | 
|  | appendLinesAs(file, charset, Arrays.asList(lines)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes lines to file using the given encoding, ending every line with a | 
|  | * line break '\n' character. | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void writeLinesAs(Path file, Charset charset, Iterable<String> lines) | 
|  | throws IOException { | 
|  | createDirectoryAndParents(file.getParentDirectory()); | 
|  | asByteSink(file).asCharSink(charset).writeLines(lines); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Appends lines to file using the given encoding, ending every line with a | 
|  | * line break '\n' character. | 
|  | */ | 
|  | @ThreadSafe // but not atomic | 
|  | public static void appendLinesAs(Path file, Charset charset, Iterable<String> lines) | 
|  | throws IOException { | 
|  | createDirectoryAndParents(file.getParentDirectory()); | 
|  | asByteSink(file, true).asCharSink(charset).writeLines(lines); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Writes the specified byte array to the output file. Follows symbolic links. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static void writeContent(Path outputFile, byte[] content) throws IOException { | 
|  | asByteSink(outputFile).write(content); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Updates the contents of the output file if they do not match the given array, thus maintaining | 
|  | * the mtime and ctime in case of no updates. Follows symbolic links. | 
|  | * | 
|  | * <p>If the output file already exists but is unreadable, this tries to overwrite it with the new | 
|  | * contents. In other words: unreadable or missing files are considered to be non-matching. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static void maybeUpdateContent(Path outputFile, byte[] newContent) throws IOException { | 
|  | byte[] currentContent; | 
|  | try { | 
|  | currentContent = readContent(outputFile); | 
|  | } catch (IOException e) { | 
|  | // Ignore error per the rationale given in the docstring. Keep in mind that what we are doing | 
|  | // here is for performance reasons only so we should only break if the real action (that is, | 
|  | // the write) fails -- not any of the optimization steps. | 
|  | currentContent = null; | 
|  | } | 
|  |  | 
|  | if (currentContent == null) { | 
|  | writeContent(outputFile, newContent); | 
|  | } else { | 
|  | if (!Arrays.equals(newContent, currentContent)) { | 
|  | if (!outputFile.isWritable()) { | 
|  | outputFile.delete(); | 
|  | } | 
|  | writeContent(outputFile, newContent); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the entirety of the specified input stream and returns it as a char | 
|  | * array, decoding characters using ISO-8859-1 (Latin1). | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static char[] readContentAsLatin1(InputStream in) throws IOException { | 
|  | return convertFromLatin1(ByteStreams.toByteArray(in)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the entirety of the specified file and returns it as a char array, | 
|  | * decoding characters using ISO-8859-1 (Latin1). | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static char[] readContentAsLatin1(Path inputFile) throws IOException { | 
|  | return convertFromLatin1(readContent(inputFile)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns an iterable that allows iterating over ISO-8859-1 (Latin1) text | 
|  | * file contents line by line. If the file ends in a line break, the iterator | 
|  | * will return an empty string as the last element. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static Iterable<String> iterateLinesAsLatin1(Path inputFile) throws IOException { | 
|  | return readLines(inputFile, ISO_8859_1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns an iterable that allows iterating over text file contents line by line in the given | 
|  | * {@link Charset}. If the file ends in a line break, the iterator will return an empty string | 
|  | * as the last element. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static Iterable<String> readLines(Path inputFile, Charset charset) throws IOException { | 
|  | return asByteSource(inputFile).asCharSource(charset).readLines(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the entirety of the specified file and returns it as a byte array. | 
|  | * | 
|  | * @throws IOException if there was an error | 
|  | */ | 
|  | public static byte[] readContent(Path inputFile) throws IOException { | 
|  | return asByteSource(inputFile).read(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Reads the entire file using the given charset and returns the contents as a string | 
|  | */ | 
|  | public static String readContent(Path inputFile, Charset charset) throws IOException { | 
|  | return asByteSource(inputFile).asCharSource(charset).read(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Reads at most {@code limit} bytes from {@code inputFile} and returns it as a byte array. | 
|  | * | 
|  | * @throws IOException if there was an error. | 
|  | */ | 
|  | public static byte[] readContentWithLimit(Path inputFile, int limit) throws IOException { | 
|  | Preconditions.checkArgument(limit >= 0, "limit needs to be >=0, but it is %s", limit); | 
|  | ByteSource byteSource = asByteSource(inputFile); | 
|  | byte[] buffer = new byte[limit]; | 
|  | try (InputStream inputStream = byteSource.openBufferedStream()) { | 
|  | int read = ByteStreams.read(inputStream, buffer, 0, limit); | 
|  | return read == limit ? buffer : Arrays.copyOf(buffer, read); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * The type of {@link IOException} thrown by {@link #readWithKnownFileSize} when fewer bytes than | 
|  | * expected are read. | 
|  | */ | 
|  | public static class ShortReadIOException extends IOException { | 
|  | public final Path path; | 
|  | public final int fileSize; | 
|  | public final int numBytesRead; | 
|  |  | 
|  | private ShortReadIOException(Path path, int fileSize, int numBytesRead) { | 
|  | super("Unexpected short read from file '" + path + "' (expected " + fileSize + ", got " | 
|  | + numBytesRead + " bytes)"); | 
|  | this.path = path; | 
|  | this.fileSize = fileSize; | 
|  | this.numBytesRead = numBytesRead; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Reads the given file {@code path}, assumed to have size {@code fileSize}, and does a sanity | 
|  | * check on the number of bytes read. | 
|  | * | 
|  | * <p>Use this method when you already know the size of the file. The sanity check is intended to | 
|  | * catch issues where filesystems incorrectly truncate files. | 
|  | * | 
|  | * @throws IOException if there was an error, or if fewer than {@code fileSize} bytes were read. | 
|  | */ | 
|  | public static byte[] readWithKnownFileSize(Path path, long fileSize) throws IOException { | 
|  | if (fileSize > Integer.MAX_VALUE) { | 
|  | throw new IOException("Cannot read file with size larger than 2GB"); | 
|  | } | 
|  | int fileSizeInt = (int) fileSize; | 
|  | byte[] bytes = readContentWithLimit(path, fileSizeInt); | 
|  | if (fileSizeInt > bytes.length) { | 
|  | throw new ShortReadIOException(path, fileSizeInt, bytes.length); | 
|  | } | 
|  | return bytes; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the type of the file system path belongs to. | 
|  | */ | 
|  | public static String getFileSystem(Path path) { | 
|  | return path.getFileSystem().getFileSystemType(path); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns whether the given path starts with any of the paths in the given | 
|  | * list of prefixes. | 
|  | */ | 
|  | public static boolean startsWithAny(Path path, Iterable<Path> prefixes) { | 
|  | for (Path prefix : prefixes) { | 
|  | if (path.startsWith(prefix)) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns whether the given path starts with any of the paths in the given | 
|  | * list of prefixes. | 
|  | */ | 
|  | public static boolean startsWithAny(PathFragment path, Iterable<PathFragment> prefixes) { | 
|  | for (PathFragment prefix : prefixes) { | 
|  | if (path.startsWith(prefix)) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  |  | 
|  | /** | 
|  | * Create a new hard link file at "linkPath" for file at "originalPath". If "originalPath" is a | 
|  | * directory, then for each entry, create link under "linkPath" recursively. | 
|  | * | 
|  | * @param linkPath The path of the new link file to be created | 
|  | * @param originalPath The path of the original file | 
|  | * @throws IOException if there was an error executing {@link Path#createHardLink} | 
|  | */ | 
|  | public static void createHardLink(Path linkPath, Path originalPath) throws IOException { | 
|  |  | 
|  | // Directory | 
|  | if (originalPath.isDirectory()) { | 
|  | for (Path originalSubpath : originalPath.getDirectoryEntries()) { | 
|  | Path linkSubpath = linkPath.getRelative(originalSubpath.relativeTo(originalPath)); | 
|  | createHardLink(linkSubpath, originalSubpath); | 
|  | } | 
|  | // Other types of file | 
|  | } else { | 
|  | Path parentDir = linkPath.getParentDirectory(); | 
|  | if (!parentDir.exists()) { | 
|  | FileSystemUtils.createDirectoryAndParents(parentDir); | 
|  | } | 
|  | originalPath.createHardLink(linkPath); | 
|  | } | 
|  | } | 
|  | } |