From e9f3e325c274f19b0f6eceea2367708e3be689e9 Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Wed, 2 Jun 2021 22:45:58 +0000 Subject: [PATCH] 8259530: Generated docs contain MIT/GPL-licenced works without reproducing the licence Reviewed-by: prappo --- .../doclets/formats/html/HtmlDoclet.java | 52 ++++++ .../doclets/formats/html/HtmlOptions.java | 20 +++ .../html/resources/standard.properties | 10 ++ .../doclets/toolkit/util/DocFile.java | 6 + .../doclets/toolkit/util/DocFileFactory.java | 4 + .../doclets/toolkit/util/DocPaths.java | 5 +- .../toolkit/util/StandardDocFileFactory.java | 9 +- .../testLegalNotices/TestLegalNotices.java | 159 ++++++++++++++++++ .../jdk/javadoc/tool/CheckResourceKeys.java | 4 +- .../jdk/javadoc/tool/api/basic/APITest.java | 4 +- 10 files changed, 267 insertions(+), 6 deletions(-) create mode 100644 test/langtools/jdk/javadoc/doclet/testLegalNotices/TestLegalNotices.java diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDoclet.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDoclet.java index 6bce838d8d5..1ff3ce38d53 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDoclet.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlDoclet.java @@ -25,6 +25,13 @@ package jdk.javadoc.internal.doclets.formats.html; +import java.io.IOException; +import java.io.OutputStream; +import java.io.Writer; +import java.nio.file.DirectoryStream; +import java.nio.file.Files; +import java.nio.file.InvalidPathException; +import java.nio.file.Path; import java.util.*; import java.util.function.Function; @@ -285,6 +292,8 @@ public class HtmlDoclet extends AbstractDoclet { f = DocFile.createFileForOutput(configuration, DocPaths.JQUERY_OVERRIDES_CSS); f.copyResource(DOCLET_RESOURCES.resolve(DocPaths.JQUERY_OVERRIDES_CSS), true, true); } + + copyLegalFiles(options.createIndex()); } private void copyJqueryFiles() throws DocletException { @@ -312,6 +321,49 @@ public class HtmlDoclet extends AbstractDoclet { } } + private void copyLegalFiles(boolean includeJQuery) throws DocletException { + Path legalNoticesDir; + String legalNotices = configuration.getOptions().legalNotices(); + switch (legalNotices) { + case "", "default" -> { + Path javaHome = Path.of(System.getProperty("java.home")); + legalNoticesDir = javaHome.resolve("legal").resolve(getClass().getModule().getName()); + } + + case "none" -> { + return; + } + + default -> { + try { + legalNoticesDir = Path.of(legalNotices); + } catch (InvalidPathException e) { + messages.error("doclet.Error_invalid_path_for_legal_notices", + legalNotices, e.getMessage()); + return; + } + } + } + + if (Files.exists(legalNoticesDir)) { + try (DirectoryStream ds = Files.newDirectoryStream(legalNoticesDir)) { + for (Path entry: ds) { + if (!Files.isRegularFile(entry)) { + continue; + } + if (entry.getFileName().toString().startsWith("jquery") && !includeJQuery) { + continue; + } + DocPath filePath = DocPaths.LEGAL.resolve(entry.getFileName().toString()); + DocFile df = DocFile.createFileForOutput(configuration, filePath); + df.copyFile(DocFile.createFileForInput(configuration, entry)); + } + } catch (IOException e) { + messages.error("doclet.Error_copying_legal_notices", e); + } + } + } + @Override // defined by AbstractDoclet protected void generateClassFiles(SortedSet typeElems, ClassTree classTree) throws DocletException { diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlOptions.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlOptions.java index ce5f7def06f..30760a6c8f7 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlOptions.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/HtmlOptions.java @@ -121,6 +121,11 @@ public class HtmlOptions extends BaseOptions { */ private String helpFile = ""; + /** + * Argument for command-line option {@code --legal-notices}. + */ + private String legalNotices = ""; + /** * Argument for command-line option {@code -nodeprecatedlist}. * True if command-line option "-nodeprecatedlist" is used. Default value is @@ -265,6 +270,14 @@ public class HtmlOptions extends BaseOptions { } }, + new XOption(resources, "--legal-notices", 1) { + @Override + public boolean process(String opt, List args) { + legalNotices = args.get(0); + return true; + } + }, + new Option(resources, "-nohelp") { @Override public boolean process(String opt, List args) { @@ -600,6 +613,13 @@ public class HtmlOptions extends BaseOptions { return helpFile; } + /** + * Argument for command-line option {@code --legal-notices}. + */ + public String legalNotices() { + return legalNotices; + } + /** * Argument for command-line option {@code -nodeprecated}. * True if command-line option "-nodeprecated" is used. Default value is diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/standard.properties b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/standard.properties index 0f9e18cdaf1..c53e8f2df76 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/standard.properties +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/resources/standard.properties @@ -353,6 +353,11 @@ doclet.Declared_Using_Preview.SEALED_PERMITS=Sealed Classes doclet.PreviewPlatformLeadingNote={0} is a preview API of the Java platform. doclet.ReflectivePreviewPlatformLeadingNote={0} is a reflective preview API of the Java platform. +# 0: an exception +doclet.Error_copying_legal_notices=Error while copying legal notices: {0} +# 0: the path; 1: the detail message for the exception +doclet.Error_invalid_path_for_legal_notices=Invalid path ''{0}'' for legal notices: {1} + # option specifiers doclet.usage.add-stylesheet.parameters=\ @@ -441,6 +446,11 @@ doclet.usage.group.parameters=\ doclet.usage.group.description=\ Group specified elements together in overview page +doclet.usage.legal-notices.parameters=\ + 'default' | 'none' | +doclet.usage.legal-notices.description=\ + Control legal notices in the generated output + doclet.usage.nocomment.description=\ Suppress description and tags, generate only declarations diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFile.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFile.java index aa2088e14c2..70868dc7381 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFile.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFile.java @@ -32,6 +32,7 @@ import java.io.InputStreamReader; import java.io.OutputStream; import java.io.UnsupportedEncodingException; import java.io.Writer; +import java.nio.file.Path; import java.util.MissingResourceException; import java.util.regex.Matcher; import java.util.regex.Pattern; @@ -66,6 +67,11 @@ public abstract class DocFile { return DocFileFactory.getFactory(configuration).createFileForInput(file); } + /** Create a DocFile for a file that will be opened for reading. */ + public static DocFile createFileForInput(BaseConfiguration configuration, Path file) { + return DocFileFactory.getFactory(configuration).createFileForInput(file); + } + /** Create a DocFile for a file that will be opened for writing. */ public static DocFile createFileForOutput(BaseConfiguration configuration, DocPath path) { return DocFileFactory.getFactory(configuration).createFileForOutput(path); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFileFactory.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFileFactory.java index e93ce5fc1a3..6b909961abf 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFileFactory.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocFileFactory.java @@ -25,6 +25,7 @@ package jdk.javadoc.internal.doclets.toolkit.util; +import java.nio.file.Path; import javax.tools.JavaFileManager; import javax.tools.JavaFileManager.Location; import javax.tools.StandardJavaFileManager; @@ -78,6 +79,9 @@ public abstract class DocFileFactory { /** Create a DocFile for a file that will be opened for reading. */ abstract DocFile createFileForInput(String file); + /** Create a DocFile for a file that will be opened for reading. */ + abstract DocFile createFileForInput(Path file); + /** Create a DocFile for a file that will be opened for writing. */ abstract DocFile createFileForOutput(DocPath path); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocPaths.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocPaths.java index 6d057f7f1ce..30f46c99df7 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocPaths.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/DocPaths.java @@ -100,7 +100,7 @@ public class DocPaths { /** The name of the stylesheet file overriding jQuery UI stylesheet. */ public static final DocPath JQUERY_OVERRIDES_CSS = DocPath.create("jquery-ui.overrides.css"); - /** The name of the directory for the jQuery. */ + /** The name of the directory for the jQuery files. */ public static final DocPath JQUERY_FILES = DocPath.create("script-dir"); /** The name of the default jQuery javascript file. */ @@ -112,6 +112,9 @@ public class DocPaths { /** The name of the default jQuery UI javascript file. */ public static final DocPath JQUERY_UI_JS = DocPath.create("jquery-ui.min.js"); + /** The name of the directory for legal files. */ + public static final DocPath LEGAL = DocPath.create("legal"); + /** The name of the member search index js file. */ public static final DocPath MEMBER_SEARCH_INDEX_JS = DocPath.create("member-search-index.js"); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/StandardDocFileFactory.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/StandardDocFileFactory.java index 288876779bb..b621d74f884 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/StandardDocFileFactory.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/StandardDocFileFactory.java @@ -109,6 +109,11 @@ class StandardDocFileFactory extends DocFileFactory { return new StandardDocFile(Paths.get(file)); } + @Override + public DocFile createFileForInput(Path file) { + return new StandardDocFile(file); + } + @Override public DocFile createFileForOutput(DocPath path) { return new StandardDocFile(DocumentationTool.Location.DOCUMENTATION_OUTPUT, path); @@ -140,12 +145,12 @@ class StandardDocFileFactory extends DocFileFactory { private final Path file; /** Create a StandardDocFile for a given file. */ - private StandardDocFile(Path file) { + StandardDocFile(Path file) { this.file = file; } /** Create a StandardDocFile for a given location and relative path. */ - private StandardDocFile(Location location, DocPath path) { + StandardDocFile(Location location, DocPath path) { super(location, path); Assert.check(location == DocumentationTool.Location.DOCUMENTATION_OUTPUT); this.file = newFile(getDestDir(), path.getPath()); diff --git a/test/langtools/jdk/javadoc/doclet/testLegalNotices/TestLegalNotices.java b/test/langtools/jdk/javadoc/doclet/testLegalNotices/TestLegalNotices.java new file mode 100644 index 00000000000..e844d14d72d --- /dev/null +++ b/test/langtools/jdk/javadoc/doclet/testLegalNotices/TestLegalNotices.java @@ -0,0 +1,159 @@ +/* + * Copyright (c) 2003, 2019, 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. + */ + +/* + * @test + * @bug 8259530 + * @summary Generated docs contain MIT/GPL-licenced works without reproducing the licence + * @library /tools/lib ../../lib + * @modules jdk.javadoc/jdk.javadoc.internal.tool + * @build toolbox.ToolBox javadoc.tester.* + * @run main TestLegalNotices + */ + +import java.io.IOException; +import java.nio.file.DirectoryStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.Set; +import java.util.TreeSet; +import java.util.function.Predicate; + +import javadoc.tester.JavadocTester; +import toolbox.ToolBox; + +public class TestLegalNotices extends JavadocTester { + public static void main(String... args) throws Exception { + TestLegalNotices tester = new TestLegalNotices(); + tester.runTests(m -> new Object[]{Path.of(m.getName())}); + } + + private final ToolBox tb = new ToolBox(); + + enum OptionKind { + UNSET, DEFAULT, NONE, DIR + } + + enum IndexKind { + INDEX, NO_INDEX + } + + + @Test + public void testCombo(Path base) throws IOException { + Path src = base.resolve("src"); + tb.writeJavaFiles(src, "package p; public class C { }"); + Path legal = base.resolve("toy-legal"); + tb.writeFile(legal.resolve("TOY-LICENSE"), "This is a demo license."); + + for (var optionKind : OptionKind.values()) { + for (var indexKind : IndexKind.values()) { + test(base, src, legal, optionKind, indexKind); + } + } + } + + void test(Path base, Path src, Path legal, OptionKind optionKind, IndexKind indexKind) throws IOException { + System.out.println("testing " + optionKind + " " + indexKind); + Path out = base.resolve(optionKind + "-" + indexKind).resolve("out"); + List args = new ArrayList<>(); + args.addAll(List.of( + "-d", out.toString())); + + if (indexKind == IndexKind.NO_INDEX) { + args.add("-noindex"); + } + + args.addAll(List.of( + "-Xdoclint:-missing", + "--source-path", src.toString(), + "p")); + + String value = switch (optionKind) { + case UNSET -> null; + case DEFAULT -> "default"; + case NONE -> "none"; + case DIR -> legal.toString(); + }; + if (value != null) { + args.addAll(List.of("--legal-notices", value)); + } + javadoc(args.toArray(new String[0])); + + Set expectFiles = getExpectFiles(optionKind, indexKind, legal); + Set foundFiles = listFiles(out.resolve("legal")); + + checking("Checking legal notice files"); + super.out.println("Expected: " + expectFiles); + super.out.println(" Found: " + foundFiles); + if (foundFiles.equals(expectFiles)) { + passed("Found all expected files"); + } + } + + Set getExpectFiles(OptionKind optionKind, IndexKind indexKind, Path legal) throws IOException { + switch (optionKind) { + case UNSET, DEFAULT -> { + Path javaHome = Path.of(System.getProperty("java.home")); + Path legal_javadoc = javaHome.resolve("legal").resolve("jdk.javadoc"); + return listFiles(legal_javadoc, p -> + switch (indexKind) { + case INDEX -> true; + case NO_INDEX -> !p.getFileName().toString().contains("jquery"); + }); + } + + case NONE -> { + return Collections.emptySet(); + } + + case DIR -> { + return listFiles(legal); + } + } + throw new IllegalStateException(); + } + + Set listFiles(Path dir) throws IOException { + return listFiles(dir, p -> true); + } + + Set listFiles(Path dir, Predicate filter) throws IOException { + if (!Files.exists(dir)) { + return Collections.emptySet(); + } + + try (DirectoryStream ds = Files.newDirectoryStream(dir)) { + Set files = new TreeSet<>(); + for (Path p : ds) { + if (!Files.isDirectory(p) && filter.test(p)) { + files.add(p.getFileName()); + } + } + return files; + } + } +} \ No newline at end of file diff --git a/test/langtools/jdk/javadoc/tool/CheckResourceKeys.java b/test/langtools/jdk/javadoc/tool/CheckResourceKeys.java index e7795df75f1..36b48559e59 100644 --- a/test/langtools/jdk/javadoc/tool/CheckResourceKeys.java +++ b/test/langtools/jdk/javadoc/tool/CheckResourceKeys.java @@ -173,8 +173,8 @@ public class CheckResourceKeys { // ignore this partial key, tested by usageTests if (ck.equals("main.opt.")) continue; - // ignore this system property name - if (ck.equals("javadoc.internal.show.taglets")) + // ignore these system property names + if (ck.equals("javadoc.internal.show.taglets") || ck.equals("javadoc.legal-notices")) continue; if (resourceKeys.contains(ck)) continue; diff --git a/test/langtools/jdk/javadoc/tool/api/basic/APITest.java b/test/langtools/jdk/javadoc/tool/api/basic/APITest.java index 8c7c9453622..c1ec5c25a7f 100644 --- a/test/langtools/jdk/javadoc/tool/api/basic/APITest.java +++ b/test/langtools/jdk/javadoc/tool/api/basic/APITest.java @@ -150,7 +150,9 @@ class APITest { missing.removeAll(foundFiles); if (!missing.isEmpty()) error("the following files were not found in " + where + ": " + missing); - Set unexpected = new TreeSet(foundFiles); + Set unexpected = foundFiles.stream() + .filter(p -> !p.startsWith("legal")) + .collect(Collectors.toCollection(TreeSet::new)); unexpected.removeAll(expectFiles); if (!unexpected.isEmpty()) error("the following unexpected files were found in " + where + ": " + unexpected);