From 5b0c9ccc495b6bf8061c161d48f73238c7167bc0 Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Tue, 28 Sep 2021 23:53:49 +0000 Subject: [PATCH] 8274172: Convert JavadocTester to use NIO Reviewed-by: prappo --- .../doclet/testDocFileDir/TestDocFileDir.java | 13 +- .../doclet/testMetadata/TestMetadata.java | 12 +- .../testRelativeLinks/TestRelativeLinks.java | 7 +- .../testSearchScript/TestSearchScript.java | 11 +- .../TestSingletonLists.java | 2 +- .../doclet/testStylesheet/TestStylesheet.java | 2 +- .../lib/javadoc/tester/JavadocTester.java | 196 +++++++----------- test/langtools/tools/lib/toolbox/ToolBox.java | 49 ++++- 8 files changed, 152 insertions(+), 140 deletions(-) diff --git a/test/langtools/jdk/javadoc/doclet/testDocFileDir/TestDocFileDir.java b/test/langtools/jdk/javadoc/doclet/testDocFileDir/TestDocFileDir.java index 70660a4340eea..d2f9e98fa9c51 100644 --- a/test/langtools/jdk/javadoc/doclet/testDocFileDir/TestDocFileDir.java +++ b/test/langtools/jdk/javadoc/doclet/testDocFileDir/TestDocFileDir.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 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 @@ -28,13 +28,14 @@ * get overwritten when the sourcepath is equal to the destination * directory. * Also test that -docfilessubdirs and -excludedocfilessubdir both work. - * @library ../../lib + * @library /tools/lib ../../lib * @modules jdk.javadoc/jdk.javadoc.internal.tool - * @build javadoc.tester.* + * @build toolbox.ToolBox javadoc.tester.* * @run main TestDocFileDir */ import javadoc.tester.JavadocTester; +import toolbox.ToolBox; public class TestDocFileDir extends JavadocTester { @@ -43,10 +44,12 @@ public static void main(String... args) throws Exception { tester.runTests(); } + ToolBox tb = new ToolBox(); + // Output dir = "", Input dir = "" @Test public void test1() { - copyDir(testSrc("pkg"), "."); + tb.copyDir(testSrc("pkg"), "pkg"); setOutputDirectoryCheck(DirectoryCheck.NO_HTML_FILES); javadoc("pkg/C.java"); checkExit(Exit.OK); @@ -58,7 +61,7 @@ public void test1() { @Test public void test2() { String outdir = "out2"; - copyDir(testSrc("pkg"), outdir); + tb.copyDir(testSrc("pkg"), outdir + "/pkg"); setOutputDirectoryCheck(DirectoryCheck.NO_HTML_FILES); javadoc("-d", outdir, "-sourcepath", "blah" + PS + outdir + PS + "blah", diff --git a/test/langtools/jdk/javadoc/doclet/testMetadata/TestMetadata.java b/test/langtools/jdk/javadoc/doclet/testMetadata/TestMetadata.java index 46e48f6490794..040d2d1c78c56 100644 --- a/test/langtools/jdk/javadoc/doclet/testMetadata/TestMetadata.java +++ b/test/langtools/jdk/javadoc/doclet/testMetadata/TestMetadata.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 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 @@ -165,9 +165,8 @@ public void runTests() throws Exception { ); void checkBodyClasses() throws IOException { - Path outputDirPath = outputDir.toPath(); - for (Path p : tb.findFiles(".html", outputDirPath)) { - checkBodyClass(outputDirPath.relativize(p)); + for (Path p : tb.findFiles(".html", outputDir)) { + checkBodyClass(outputDir.relativize(p)); } } @@ -231,9 +230,8 @@ void checkBodyClass(Path p) { ); void checkMetadata() throws IOException { - Path outputDirPath = outputDir.toPath(); - for (Path p : tb.findFiles(".html", outputDirPath)) { - checkMetadata(outputDirPath.relativize(p)); + for (Path p : tb.findFiles(".html", outputDir)) { + checkMetadata(outputDir.relativize(p)); } } diff --git a/test/langtools/jdk/javadoc/doclet/testRelativeLinks/TestRelativeLinks.java b/test/langtools/jdk/javadoc/doclet/testRelativeLinks/TestRelativeLinks.java index 30ef9ec4d6fca..c0619aeaf0d17 100644 --- a/test/langtools/jdk/javadoc/doclet/testRelativeLinks/TestRelativeLinks.java +++ b/test/langtools/jdk/javadoc/doclet/testRelativeLinks/TestRelativeLinks.java @@ -35,6 +35,9 @@ import java.io.File; import java.io.FileOutputStream; import java.io.IOException; +import java.io.OutputStream; +import java.nio.file.Files; +import java.nio.file.Path; import javadoc.tester.JavadocTester; @@ -200,9 +203,9 @@ public void checkLinks() { } private void touch(String file) { - File f = new File(outputDir, file); + Path f = outputDir.resolve(file); out.println("touch " + f); - try (FileOutputStream fos = new FileOutputStream(f)) { + try (OutputStream fos = Files.newOutputStream(f)) { } catch (IOException e) { checking("Touch file"); failed("Error creating file: " + e); diff --git a/test/langtools/jdk/javadoc/doclet/testSearchScript/TestSearchScript.java b/test/langtools/jdk/javadoc/doclet/testSearchScript/TestSearchScript.java index 8c5e77c3c15da..ad92d3f0e65eb 100644 --- a/test/langtools/jdk/javadoc/doclet/testSearchScript/TestSearchScript.java +++ b/test/langtools/jdk/javadoc/doclet/testSearchScript/TestSearchScript.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, 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 @@ -41,10 +41,9 @@ import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; import javax.script.ScriptException; -import java.io.BufferedReader; -import java.io.File; -import java.io.FileReader; import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Path; import java.util.List; import jtreg.SkippedException; @@ -71,9 +70,9 @@ private Invocable getEngine() throws ScriptException, IOException, NoSuchMethodE // see https://github.com/graalvm/graaljs/blob/master/docs/user/ScriptEngine.md Bindings bindings = engine.getBindings(ScriptContext.ENGINE_SCOPE); bindings.put("polyglot.js.nashorn-compat", true); - engine.eval(new BufferedReader(new FileReader(new File(testSrc, "javadoc-search.js")))); + engine.eval(Files.newBufferedReader(Path.of(testSrc).resolve("javadoc-search.js"))); Invocable inv = (Invocable) engine; - inv.invokeFunction("loadIndexFiles", outputDir.getAbsolutePath()); + inv.invokeFunction("loadIndexFiles", outputDir.toAbsolutePath().toString()); return inv; } diff --git a/test/langtools/jdk/javadoc/doclet/testSingletonLists/TestSingletonLists.java b/test/langtools/jdk/javadoc/doclet/testSingletonLists/TestSingletonLists.java index 409d77573c3b2..0df4f1e25c0d6 100644 --- a/test/langtools/jdk/javadoc/doclet/testSingletonLists/TestSingletonLists.java +++ b/test/langtools/jdk/javadoc/doclet/testSingletonLists/TestSingletonLists.java @@ -206,7 +206,7 @@ void checkLists() { checking("Check lists"); ListChecker c = new ListChecker(out, this::readFile); try { - c.checkDirectory(outputDir.toPath()); + c.checkDirectory(outputDir); c.report(); int errors = c.getErrorCount(); if (errors == 0) { diff --git a/test/langtools/jdk/javadoc/doclet/testStylesheet/TestStylesheet.java b/test/langtools/jdk/javadoc/doclet/testStylesheet/TestStylesheet.java index 0e2643a22fb37..95d5fd0267c0d 100644 --- a/test/langtools/jdk/javadoc/doclet/testStylesheet/TestStylesheet.java +++ b/test/langtools/jdk/javadoc/doclet/testStylesheet/TestStylesheet.java @@ -381,7 +381,7 @@ void checkStyles(Set styles) { checking("Check CSS class names"); CSSClassChecker c = new CSSClassChecker(out, this::readFile, styles); try { - c.checkDirectory(outputDir.toPath()); + c.checkDirectory(outputDir); c.report(); int errors = c.getErrorCount(); if (errors == 0) { diff --git a/test/langtools/jdk/javadoc/lib/javadoc/tester/JavadocTester.java b/test/langtools/jdk/javadoc/lib/javadoc/tester/JavadocTester.java index 1cf6098e862bd..79b20ff7e9ec2 100644 --- a/test/langtools/jdk/javadoc/lib/javadoc/tester/JavadocTester.java +++ b/test/langtools/jdk/javadoc/lib/javadoc/tester/JavadocTester.java @@ -25,10 +25,7 @@ import java.io.BufferedWriter; import java.io.ByteArrayOutputStream; -import java.io.File; import java.io.FileNotFoundException; -import java.io.FileWriter; -import java.io.FilenameFilter; import java.io.IOException; import java.io.PrintStream; import java.io.PrintWriter; @@ -41,13 +38,18 @@ import java.lang.reflect.Method; import java.nio.charset.Charset; import java.nio.charset.UnsupportedCharsetException; +import java.nio.file.DirectoryStream; +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.attribute.BasicFileAttributes; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.Collections; +import java.util.Comparator; import java.util.EnumMap; import java.util.HashMap; import java.util.List; @@ -157,7 +159,7 @@ public enum Output { } /** The output directory used in the most recent call of javadoc. */ - protected File outputDir; + protected Path outputDir; /** The output charset used in the most recent call of javadoc. */ protected Charset charset = Charset.defaultCharset(); @@ -169,7 +171,7 @@ public enum Output { private final Map outputMap = new EnumMap<>(Output.class); /** A cache of file content, to avoid reading files unnecessarily. */ - private final Map> fileContentCache = new HashMap<>(); + private final Map> fileContentCache = new HashMap<>(); /** The charset used for files in the fileContentCache. */ private Charset fileContentCacheCharset = null; @@ -185,7 +187,7 @@ public enum Output { * @return the full path of the specified file */ public static String testSrc(String path) { - return new File(testSrc, path).getPath(); + return Path.of(testSrc).resolve(path).toString(); } /** @@ -195,35 +197,40 @@ public enum DirectoryCheck { /** * Check that the directory is empty. */ - EMPTY((file, name) -> true), + EMPTY(p -> true), /** * Check that the directory does not contain any HTML files, * such as may have been generated by a prior run of javadoc * using this directory. * For now, the check is only performed on the top level directory. */ - NO_HTML_FILES((file, name) -> name.endsWith(".html")), + NO_HTML_FILES(p -> p.getFileName().toString().endsWith(".html")), /** * No check is performed on the directory contents. */ - NONE(null) { @Override void check(File dir) { } }; + NONE(null) { @Override void check(Path dir) { } }; /** The filter used to detect that files should not be present. */ - FilenameFilter filter; + DirectoryStream.Filter filter; - DirectoryCheck(FilenameFilter f) { + DirectoryCheck(DirectoryStream.Filter f) { filter = f; } - void check(File dir) { - if (dir.isDirectory()) { - String[] contents = dir.list(filter); - if (contents == null) - throw new Error("cannot list directory: " + dir); - if (contents.length > 0) { - System.err.println("Found extraneous files in dir:" + dir.getAbsolutePath()); - for (String x : contents) { - System.err.println(x); + void check(Path dir) { + if (Files.isDirectory(dir)) { + List contents = new ArrayList<>(); + try (var ds = Files.newDirectoryStream(dir, filter)) { + for (Path p : ds) { + contents.add(p); + } + } catch (IOException e) { + throw new Error("cannot list directory: " + dir + "; " + e, e); + } + if (!contents.isEmpty()) { + System.err.println("Found extraneous files in dir:" + dir.toAbsolutePath()); + for (Path p : contents) { + System.err.println(p); } throw new Error("directory has unexpected content: " + dir); } @@ -316,13 +323,13 @@ public void javadoc(String... args) { out.println("Running javadoc (run "+ javadocRunNum + ")..."); } - outputDir = new File("."); + outputDir = Path.of("."); String charsetArg = null; String docencodingArg = null; String encodingArg = null; for (int i = 0; i < args.length - 2; i++) { switch (args[i]) { - case "-d" -> outputDir = new File(args[++i]); + case "-d" -> outputDir = Path.of(args[++i]); case "-charset" -> charsetArg = args[++i]; case "-docencoding" -> docencodingArg = args[++i]; case "-encoding" -> encodingArg = args[++i]; @@ -377,7 +384,7 @@ public void javadoc(String... args) { } }); - if (exitCode == Exit.OK.code && outputDir.exists()) { + if (exitCode == Exit.OK.code && Files.exists(outputDir)) { if (automaticCheckLinks) { checkLinks(); } @@ -502,8 +509,8 @@ public void checkFileAndOutput(String path, boolean expectedFound, String... str public void checkOutput(String path, boolean expectedFound, String... strings) { // Read contents of file try { - String fileString = readFile(outputDir, path); - checkOutput(new File(outputDir, path).getPath(), fileString, expectedFound, strings); + String fileString = readFile(outputDir, Path.of(path)); + checkOutput(outputDir.resolve(path).toString(), fileString, expectedFound, strings); } catch (Error e) { checking("Read file"); failed("Error reading file: " + e); @@ -594,7 +601,7 @@ public void checkAccessibility() { checking("Check accessibility"); A11yChecker c = new A11yChecker(out, this::readFile); try { - c.checkDirectory(outputDir.toPath()); + c.checkDirectory(outputDir); c.report(); int errors = c.getErrorCount(); if (errors == 0) { @@ -616,7 +623,7 @@ public void checkLinks() { checking("Check links"); LinkChecker c = new LinkChecker(out, this::readFile); try { - c.checkDirectory(outputDir.toPath()); + c.checkDirectory(outputDir); c.report(); int errors = c.getErrorCount(); if (errors == 0) { @@ -641,8 +648,8 @@ public void showHeadings(String... paths) { ShowHeadings s = new ShowHeadings(out, this::readFile); for (String p : paths) { try { - File f = new File(outputDir, p); - s.checkFiles(List.of(f.toPath()), false, Collections.emptySet()); + Path f = outputDir.resolve(p); + s.checkFiles(List.of(f), false, Collections.emptySet()); } catch (IOException e) { checking("Read file"); failed("Error reading file: " + e); @@ -691,8 +698,8 @@ public void checkFiles(boolean expectedFound, Collection paths) { for (String path: paths) { // log.logCheckFile(path, expectedFound); checking("checkFile"); - File file = new File(outputDir, path); - boolean isFound = file.exists(); + Path file = outputDir.resolve(path); + boolean isFound = Files.exists(file); if (isFound == expectedFound) { passed(file, "file " + (isFound ? "found:" : "not found:") + "\n"); } else { @@ -708,7 +715,7 @@ public void checkFiles(boolean expectedFound, Collection paths) { * @param strings the strings whose order to check */ public void checkOrder(String path, String... strings) { - File file = new File(outputDir, path); + Path file = outputDir.resolve(path); String fileString = readOutputFile(path); int prevIndex = -1; for (String s : strings) { @@ -736,7 +743,7 @@ public void checkOrder(String path, String... strings) { * @param strings ensure each are unique */ public void checkUnique(String path, String... strings) { - File file = new File(outputDir, path); + Path file = outputDir.resolve(path); String fileString = readOutputFile(path); for (String s : strings) { int currentIndex = fileString.indexOf(s); @@ -762,58 +769,13 @@ public void checkUnique(String path, String... strings) { * @param files the set of files to be compared */ public void diff(String baseDir1, String baseDir2, String... files) { - File bd1 = new File(baseDir1); - File bd2 = new File(baseDir2); + Path bd1 = Path.of(baseDir1); + Path bd2 = Path.of(baseDir2); for (String file : files) { - diff(bd1, bd2, file); + diff(bd1, bd2, Path.of(file)); } } - /** - * Copies a directory from one place to another. - * - * @param targetDir the directory to copy. - * @param destDir the destination to copy the directory to. - */ - // TODO: convert to using java.nio.Files.walkFileTree - public void copyDir(String targetDir, String destDir) { - try { - File targetDirObj = new File(targetDir); - File destDirParentObj = new File(destDir); - File destDirObj = new File(destDirParentObj, targetDirObj.getName()); - if (! destDirParentObj.exists()) { - destDirParentObj.mkdir(); - } - if (! destDirObj.exists()) { - destDirObj.mkdir(); - } - String[] files = targetDirObj.list(); - for (String file : files) { - File srcFile = new File(targetDirObj, file); - File destFile = new File(destDirObj, file); - if (srcFile.isFile()) { - out.println("Copying " + srcFile + " to " + destFile); - copyFile(destFile, srcFile); - } else if(srcFile.isDirectory()) { - copyDir(srcFile.getAbsolutePath(), destDirObj.getAbsolutePath()); - } - } - } catch (IOException exc) { - throw new Error("Could not copy " + targetDir + " to " + destDir); - } - } - - /** - * Copies a file. - * - * @param destfile the destination file - * @param srcfile the source file - * @throws IOException - */ - public void copyFile(File destfile, File srcfile) throws IOException { - Files.copy(srcfile.toPath(), destfile.toPath()); - } - /** * Read a file from the output directory. * @@ -821,27 +783,27 @@ public void copyFile(File destfile, File srcfile) throws IOException { * @return the file in string format */ public String readOutputFile(String fileName) throws Error { - return readFile(outputDir, fileName); + return readFile(outputDir, Path.of(fileName)); } protected String readFile(String fileName) throws Error { - return readFile(outputDir, fileName); + return readFile(outputDir, Path.of(fileName)); } protected String readFile(String baseDir, String fileName) throws Error { - return readFile(new File(baseDir), fileName); + return readFile(Path.of(baseDir), Path.of(fileName)); } protected String readFile(Path file) { - File baseDir; - if (file.startsWith(outputDir.toPath())) { + Path baseDir; + if (file.startsWith(outputDir)) { baseDir = outputDir; } else if (file.startsWith(currDir)) { - baseDir = currDir.toFile(); + baseDir = currDir; } else { - baseDir = file.getParent().toFile(); + baseDir = file.getParent(); } - String fileName = baseDir.toPath().relativize(file).toString(); + Path fileName = baseDir.relativize(file); return readFile(baseDir, fileName); } @@ -852,20 +814,20 @@ protected String readFile(Path file) { * @param fileName the name of the file to read * @return the file in string format */ - private String readFile(File baseDir, String fileName) throws Error { + private String readFile(Path baseDir, Path fileName) throws Error { if (!Objects.equals(fileContentCacheCharset, charset)) { fileContentCache.clear(); fileContentCacheCharset = charset; } try { - File file = new File(baseDir, fileName); + Path file = baseDir.resolve(fileName); SoftReference ref = fileContentCache.get(file); String content = (ref == null) ? null : ref.get(); if (content != null) return content; // charset defaults to a value inferred from latest javadoc run - content = new String(Files.readAllBytes(file.toPath()), charset); + content = new String(Files.readAllBytes(file), charset); fileContentCache.put(file, new SoftReference<>(content)); return content; } catch (FileNotFoundException e) { @@ -897,7 +859,7 @@ protected void checking(String message) { * @param file the file that was the focus of the check * @param message a short description of the outcome */ - protected void passed(File file, String message) { + protected void passed(Path file, String message) { passed(file + ": " + message); } @@ -922,7 +884,7 @@ protected void passed(String message) { * @param file the file that was the focus of the check * @param message a short description of the outcome */ - protected void failed(File file, String message) { + protected void failed(Path file, String message) { failed(file + ": " + message); } @@ -1002,10 +964,10 @@ private boolean findString(String fileString, String stringToFind) { * @param baseDir2 the directory in which to locate the second file * @param file the file to compare in the two base directories */ - private void diff(File baseDir1, File baseDir2, String file) { + private void diff(Path baseDir1, Path baseDir2, Path file) { String file1Contents = readFile(baseDir1, file); String file2Contents = readFile(baseDir2, file); - checking("diff " + new File(baseDir1, file) + ", " + new File(baseDir2, file)); + checking("diff " + baseDir1.resolve(file) + ", " + baseDir2.resolve(file)); if (file1Contents.trim().compareTo(file2Contents.trim()) == 0) { passed("files are equal"); } else { @@ -1068,44 +1030,46 @@ static class Logger { private static final int SUFFIX = 20; private static final int MAX = PREFIX + SUFFIX; List tests = new ArrayList<>(); - String outDir; - String rootDir = rootDir(); - - static String rootDir() { - File f = new File(".").getAbsoluteFile(); - while (!new File(f, ".hg").exists()) - f = f.getParentFile(); - return f.getPath(); + Path outDir; + Path rootDir = rootDir(); + + static Path rootDir() { + Path f = Path.of(".").toAbsolutePath(); + while (f != null && !Files.exists(f.resolve(".git"))) + f = f.getParent(); + return f; } - void setOutDir(File outDir) { - this.outDir = outDir.getPath(); + void setOutDir(Path outDir) { + this.outDir = outDir; } void logCheckFile(String file, boolean positive) { // Strip the outdir because that will typically not be the same - if (file.startsWith(outDir + "/")) - file = file.substring(outDir.length() + 1); - tests.add(file + " " + positive); + Path p = Path.of(file); + if (p.startsWith(outDir)) + p = p.relativize(outDir); + tests.add(p + " " + positive); } void logCheckOutput(String file, boolean positive, String text) { // Compress the string to be displayed in the log file - String simpleText = text.replaceAll("\\s+", " ").replace(rootDir, "[ROOT]"); + String simpleText = text.replaceAll("\\s+", " ").replace(rootDir.toString(), "[ROOT]"); if (simpleText.length() > MAX) simpleText = simpleText.substring(0, PREFIX) + "..." + simpleText.substring(simpleText.length() - SUFFIX); // Strip the outdir because that will typically not be the same - if (file.startsWith(outDir + "/")) - file = file.substring(outDir.length() + 1); + Path p = Path.of(file); + if (p.startsWith(outDir)) + p = p.relativize(outDir); // The use of text.hashCode ensure that all of "text" is taken into account - tests.add(file + " " + positive + " " + text.hashCode() + " " + simpleText); + tests.add(p + " " + positive + " " + text.hashCode() + " " + simpleText); } void write() { // sort the log entries because the subtests may not be executed in the same order - tests.sort((a, b) -> a.compareTo(b)); - try (BufferedWriter bw = new BufferedWriter(new FileWriter("tester.log"))) { + tests.sort(Comparator.naturalOrder()); + try (var bw = Files.newBufferedWriter(Path.of("tester.log"))) { for (String t: tests) { bw.write(t); bw.newLine(); @@ -1116,6 +1080,4 @@ void write() { } } - // Support classes for checkLinks - } \ No newline at end of file diff --git a/test/langtools/tools/lib/toolbox/ToolBox.java b/test/langtools/tools/lib/toolbox/ToolBox.java index 4332c3418a0db..9dbb26e144814 100644 --- a/test/langtools/tools/lib/toolbox/ToolBox.java +++ b/test/langtools/tools/lib/toolbox/ToolBox.java @@ -34,6 +34,8 @@ 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; @@ -46,6 +48,7 @@ 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; @@ -249,7 +252,51 @@ public void copyFile(Path from, Path to) throws IOException { } /** - * Creates one of more directories. + * 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() { + @Override + public FileVisitResult preVisitDirectory(Path fromSubdir, BasicFileAttributes attrs) + throws IOException { + Files.copy(fromSubdir, toDir.resolve(fromDir.relativize(fromSubdir))); + return FileVisitResult.CONTINUE; + } + + @Override + public FileVisitResult visitFile(Path fromFile, BasicFileAttributes attrs) + throws IOException { + Files.copy(fromFile, toDir.resolve(fromDir.relativize(fromFile))); + return FileVisitResult.CONTINUE; + } + }); + } catch (IOException e) { + throw new Error("Could not copy " + fromDir + " to " + toDir + ": " + e, e); + } + } + + /** + * 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}.