/* * Copyright (c) 2013, 2021, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package toolbox; import java.io.BufferedWriter; import java.io.ByteArrayOutputStream; import java.io.FilterOutputStream; import java.io.FilterWriter; import java.io.IOException; import java.io.OutputStream; import java.io.PrintStream; import java.io.StringWriter; import java.io.Writer; import java.net.URI; import java.nio.charset.Charset; import java.nio.file.FileAlreadyExistsException; import java.nio.file.FileVisitOption; import java.nio.file.FileVisitResult; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.SimpleFileVisitor; import java.nio.file.StandardCopyOption; import java.nio.file.attribute.BasicFileAttributes; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.Deque; import java.util.EnumSet; import java.util.HashMap; import java.util.LinkedList; import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Objects; import java.util.Set; import java.util.TreeSet; import java.util.regex.Matcher; import java.util.regex.Pattern; import java.util.stream.Collectors; import java.util.stream.StreamSupport; import javax.tools.FileObject; import javax.tools.ForwardingJavaFileManager; import javax.tools.JavaFileManager; import javax.tools.JavaFileObject; import javax.tools.SimpleJavaFileObject; import javax.tools.ToolProvider; /** * Utility methods and classes for writing jtreg tests for * javac, javah, javap, and sjavac. (For javadoc support, * see JavadocTester.) * *
There is support for common file operations similar to * shell commands like cat, cp, diff, mv, rm, grep. * *
There is also support for invoking various tools, like * javac, javah, javap, jar, java and other JDK tools. * *
File separators: for convenience, many operations accept strings
* to represent filenames. On all platforms on which JDK is supported,
* "/" is a legal filename component separator. In particular, even
* on Windows, where the official file separator is "\", "/" is a legal
* alternative. It is therefore recommended that any client code using
* strings to specify filenames should use "/".
*
* @author Vicente Romero (original)
* @author Jonathan Gibbons (revised)
*/
public class ToolBox {
/** The platform line separator. */
public static final String lineSeparator = System.getProperty("line.separator");
/** The platform OS name. */
public static final String osName = System.getProperty("os.name");
/** The location of the class files for this test, or null if not set. */
public static final String testClasses = System.getProperty("test.classes");
/** The location of the source files for this test, or null if not set. */
public static final String testSrc = System.getProperty("test.src");
/** The location of the test JDK for this test, or null if not set. */
public static final String testJDK = System.getProperty("test.jdk");
/** The timeout factor for slow systems. */
public static final float timeoutFactor;
static {
String ttf = System.getProperty("test.timeout.factor");
timeoutFactor = (ttf == null) ? 1.0f : Float.parseFloat(ttf);
}
/** The current directory. */
public static final Path currDir = Path.of(".");
/** The stream used for logging output. */
public PrintStream out = System.err;
/**
* Checks if the host OS is some version of Windows.
* @return true if the host OS is some version of Windows
*/
public static boolean isWindows() {
return osName.toLowerCase(Locale.ENGLISH).startsWith("windows");
}
/**
* Splits a string around matches of the given regular expression.
* If the string is empty, an empty list will be returned.
*
* @param text the string to be split
* @param sep the delimiting regular expression
* @return the strings between the separators
*/
public List Similar to the shell "cp" command: {@code cp from to}.
*
* @param from the file to be copied
* @param to where to copy the file
* @throws IOException if any error occurred while copying the file
*/
public void copyFile(String from, String to) throws IOException {
copyFile(Path.of(from), Path.of(to));
}
/**
* Copies a file.
* If the given destination exists and is a directory, the copy is created
* in that directory. Otherwise, the copy will be placed at the destination,
* possibly overwriting any existing file.
* Similar to the shell "cp" command: {@code cp from to}.
*
* @param from the file to be copied
* @param to where to copy the file
* @throws IOException if an error occurred while copying the file
*/
public void copyFile(Path from, Path to) throws IOException {
if (Files.isDirectory(to)) {
to = to.resolve(from.getFileName());
} else {
Files.createDirectories(to.getParent());
}
Files.copy(from, to, StandardCopyOption.REPLACE_EXISTING);
}
/**
* Copies the contents of a directory to another directory.
* Similar to the shell command: {@code rsync fromDir/ toDir/}.
*
* @param fromDir the directory containing the files to be copied
* @param toDir the destination to which to copy the files
*/
public void copyDir(String fromDir, String toDir) {
copyDir(Path.of(fromDir), Path.of(toDir));
}
/**
* Copies the contents of a directory to another directory.
* The destination direction should not already exist.
* Similar to the shell command: {@code rsync fromDir/ toDir/}.
*
* @param fromDir the directory containing the files to be copied
* @param toDir the destination to which to copy the files
*/
public void copyDir(Path fromDir, Path toDir) {
try {
if (toDir.getParent() != null) {
Files.createDirectories(toDir.getParent());
}
Files.walkFileTree(fromDir, new SimpleFileVisitor Similar to the shell command: {@code mkdir -p paths}.
*
* @param paths the directories to be created
* @throws IOException if an error occurred while creating the directories
*/
public void createDirectories(String... paths) throws IOException {
if (paths.length == 0)
throw new IllegalArgumentException("no directories specified");
for (String p : paths)
Files.createDirectories(Path.of(p));
}
/**
* Creates one or more directories.
* For each of the series of paths, a directory will be created,
* including any necessary parent directories.
* Similar to the shell command: {@code mkdir -p paths}.
*
* @param paths the directories to be created
* @throws IOException if an error occurred while creating the directories
*/
public void createDirectories(Path... paths) throws IOException {
if (paths.length == 0)
throw new IllegalArgumentException("no directories specified");
for (Path p : paths)
Files.createDirectories(p);
}
/**
* Deletes one or more files, awaiting confirmation that the files
* no longer exist. Any directories to be deleted must be empty.
* Similar to the shell command: {@code rm files}.
*
* @param files the names of the files to be deleted
* @throws IOException if an error occurred while deleting the files
*/
public void deleteFiles(String... files) throws IOException {
deleteFiles(List.of(files).stream().map(Paths::get).collect(Collectors.toList()));
}
/**
* Deletes one or more files, awaiting confirmation that the files
* no longer exist. Any directories to be deleted must be empty.
* Similar to the shell command: {@code rm files}.
*
* @param paths the paths for the files to be deleted
* @throws IOException if an error occurred while deleting the files
*/
public void deleteFiles(Path... paths) throws IOException {
deleteFiles(List.of(paths));
}
/**
* Deletes one or more files, awaiting confirmation that the files
* no longer exist. Any directories to be deleted must be empty.
* Similar to the shell command: {@code rm files}.
*
* @param paths the paths for the files to be deleted
* @throws IOException if an error occurred while deleting the files
*/
public void deleteFiles(List Similar to the shell "mv" command: {@code mv from to}.
*
* @param from the file to be moved
* @param to where to move the file
* @throws IOException if an error occurred while moving the file
*/
public void moveFile(String from, String to) throws IOException {
moveFile(Path.of(from), Path.of(to));
}
/**
* Moves a file.
* If the given destination exists and is a directory, the file will be moved
* to that directory. Otherwise, the file will be moved to the destination,
* possibly overwriting any existing file.
* Similar to the shell "mv" command: {@code mv from to}.
*
* @param from the file to be moved
* @param to where to move the file
* @throws IOException if an error occurred while moving the file
*/
public void moveFile(Path from, Path to) throws IOException {
if (Files.isDirectory(to)) {
to = to.resolve(from.getFileName());
} else {
Files.createDirectories(to.getParent());
}
Files.move(from, to, StandardCopyOption.REPLACE_EXISTING);
}
/**
* Reads the lines of a file.
* The file is read using the default character encoding.
*
* @param path the file to be read
* @return the lines of the file
* @throws IOException if an error occurred while reading the file
*/
public List Similar to the shell "find" command: {@code find paths -name \*.java}.
*
* @param paths the directories in which to search for .java files
* @return the .java files found
* @throws IOException if an error occurred while searching for files
*/
public Path[] findJavaFiles(Path... paths) throws IOException {
return findFiles(".java", paths);
}
/**
* Find files matching the file extension, in one or more directories.
* Similar to the shell "find" command: {@code find paths -name \*.ext}.
*
* @param fileExtension the extension to search for
* @param paths the directories in which to search for files
* @return the files matching the file extension
* @throws IOException if an error occurred while searching for files
*/
public Path[] findFiles(String fileExtension, Path... paths) throws IOException {
Set For example, if the base directory is /my/dir/ and the content
* contains "package p; class C { }", the file will be written to
* /my/dir/p/C.java.
* Note: the content is analyzed using regular expressions;
* errors can occur if any contents have initial comments that might trip
* up the analysis.
*
* @param dir the base directory
* @param contents the contents of the files to be written
* @throws IOException if an error occurred while writing any of the files.
*/
public void writeJavaFiles(Path dir, String... contents) throws IOException {
if (contents.length == 0)
throw new IllegalArgumentException("no content specified for any files");
for (String c : contents) {
new JavaSource(c).write(dir);
}
}
/**
* Returns the path for the binary of a JDK tool within {@link #testJDK}.
*
* @param tool the name of the tool
* @return the path of the tool
*/
public Path getJDKTool(String tool) {
return Path.of(testJDK, "bin", tool);
}
/**
* Returns a string representing the contents of an {@code Iterable} as a list.
*
* @param > dirFiles = new LinkedList<>();
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes a) {
ioe = deleteFile(file, ioe);
dirFiles.peekFirst().add(file);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes a) {
if (!dir.equals(root)) {
dirFiles.peekFirst().add(dir);
}
dirFiles.addFirst(new ArrayList<>());
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException e) throws IOException {
if (e != null) {
throw e;
}
if (ioe != null) {
throw ioe;
}
ensureDeleted(dirFiles.removeFirst());
if (!dir.equals(root)) {
ioe = deleteFile(dir, ioe);
}
return FileVisitResult.CONTINUE;
}
});
}
/**
* Internal method to delete a file, using {@code Files.delete}.
* It does not wait to confirm deletion, nor does it retry.
* If an exception occurs it is either returned or added to the set of
* suppressed exceptions for an earlier exception.
*
* @param path the path for the file to be deleted
* @param ioe the earlier exception, or null
* @return the earlier exception or an exception that occurred while
* trying to delete the file
*/
private IOException deleteFile(Path path, IOException ioe) {
try {
Files.delete(path);
} catch (IOException e) {
if (ioe == null) {
ioe = e;
} else {
ioe.addSuppressed(e);
}
}
return ioe;
}
/**
* Wait until it is confirmed that a set of files have been deleted.
*
* @param paths the paths for the files to be deleted
* @throws IOException if a file has not been deleted
*/
private void ensureDeleted(Collection