From 93b0516d5d6e88a7e93f3ac06e02c9771def753f Mon Sep 17 00:00:00 2001 From: Pavel Rappo Date: Thu, 7 May 2020 13:59:18 +0100 Subject: [PATCH] 8224613: javadoc should better handle bad options Reviewed-by: jjg --- .../javadoc/internal/tool/JavadocTool.java | 6 +- .../jdk/javadoc/internal/tool/Main.java | 8 +- .../jdk/javadoc/internal/tool/Start.java | 53 ++- .../javadoc/internal/tool/ToolOptions.java | 4 +- .../8224613/OptionProcessingFailureTest.java | 335 ++++++++++++++++++ 5 files changed, 383 insertions(+), 23 deletions(-) create mode 100644 test/langtools/jdk/javadoc/tool/8224613/OptionProcessingFailureTest.java diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/JavadocTool.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/JavadocTool.java index 319364356c7..c01a9f23ee8 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/JavadocTool.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/JavadocTool.java @@ -127,8 +127,10 @@ public class JavadocTool extends com.sun.tools.javac.main.JavaCompiler { } public DocletEnvironment getEnvironment(ToolOptions toolOptions, - List javaNames, - Iterable fileObjects) throws ToolException { + List javaNames, + Iterable fileObjects) + throws ToolException + { toolEnv = ToolEnvironment.instance(context); toolEnv.initialize(toolOptions); ElementsTable etable = new ElementsTable(context, toolOptions); diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Main.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Main.java index 5925c226c54..467c28adba1 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Main.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Main.java @@ -46,7 +46,7 @@ public class Main { * The main entry point called by the launcher. This will call * System.exit with an appropriate return value. * - * @param args The command line parameters. + * @param args the command-line parameters */ public static void main(String... args) { System.exit(execute(args)); @@ -55,7 +55,7 @@ public class Main { /** * Programmatic interface. * - * @param args The command line parameters. + * @param args the command-line parameters * @return The return code. */ public static int execute(String... args) { @@ -67,7 +67,7 @@ public class Main { * Programmatic interface. * * @param writer a stream for all output - * @param args The command line parameters. + * @param args the command-line parameters * @return The return code. */ public static int execute(String[] args, PrintWriter writer) { @@ -80,7 +80,7 @@ public class Main { * * @param outWriter a stream for expected output * @param errWriter a stream for diagnostic output - * @param args The command line parameters. + * @param args the command-line parameters * @return The return code. */ public static int execute(String[] args, PrintWriter outWriter, PrintWriter errWriter) { diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Start.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Start.java index bc237e8773e..bb213615f1b 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Start.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/Start.java @@ -495,7 +495,12 @@ public class Start { arguments.allowEmpty(); doclet.init(locale, messager); - parseArgs(argList, javaNames); + int beforeCount = messager.nerrors; + boolean success = parseArgs(argList, javaNames); + int afterCount = messager.nerrors; + if (!success && beforeCount == afterCount) { // if there were failures but they have not been reported + return CMDERR; + } if (!arguments.handleReleaseOptions(extra -> true)) { // Arguments does not always increase the error count in the @@ -579,8 +584,13 @@ public class Start { } private Set docletOptions = null; - int handleDocletOption(int idx, List args, boolean isToolOption) - throws OptionException { + + /* + * Consumes an option along with its arguments. Returns an advanced index + * modulo the sign. If the value is negative, it means there was a failure + * processing one or more options. + */ + int consumeDocletOption(int idx, List args, boolean isToolOption) throws OptionException { if (docletOptions == null) { docletOptions = getSupportedOptionsOf(doclet); } @@ -594,6 +604,7 @@ public class Start { argBase = arg; argVal = null; } + int m = 1; String text = null; for (Doclet.Option opt : docletOptions) { if (matches(opt, argBase)) { @@ -603,21 +614,25 @@ public class Start { text = messager.getText("main.unnecessary_arg_provided", argBase); throw new OptionException(ERROR, this::showUsage, text); case 1: - opt.process(arg, Arrays.asList(argVal)); + if (!opt.process(arg, Collections.singletonList(argVal))) { + m = -1; + } break; default: text = messager.getText("main.only_one_argument_with_equals", argBase); throw new OptionException(ERROR, this::showUsage, text); } } else { - if (args.size() - idx -1 < opt.getArgumentCount()) { + if (args.size() - idx - 1 < opt.getArgumentCount()) { text = messager.getText("main.requires_argument", arg); throw new OptionException(ERROR, this::showUsage, text); } - opt.process(arg, args.subList(idx + 1, args.size())); + if (!opt.process(arg, args.subList(idx + 1, idx + 1 + opt.getArgumentCount()))) { + m = -1; + } idx += opt.getArgumentCount(); } - return idx; + return m * idx; } } // check if arg is accepted by the tool before emitting error @@ -625,7 +640,7 @@ public class Start { text = messager.getText("main.invalid_flag", arg); throw new OptionException(ERROR, this::showUsage, text); } - return idx; + return m * idx; } private static Set getSupportedOptionsOf(Doclet doclet) { @@ -649,8 +664,7 @@ public class Start { * @throws ToolException if an error occurs initializing the doclet * @throws OptionException if an error occurs while processing an option */ - private Doclet preprocess(List argv) - throws ToolException, OptionException { + private Doclet preprocess(List argv) throws ToolException, OptionException { // doclet specifying arguments String userDocletPath = null; String userDocletName = null; @@ -774,15 +788,19 @@ public class Start { } } - private void parseArgs(List args, List javaNames) throws ToolException, - OptionException, com.sun.tools.javac.main.Option.InvalidValueException { + private boolean parseArgs(List args, List javaNames) + throws OptionException, com.sun.tools.javac.main.Option.InvalidValueException + { + boolean success = true; for (int i = 0; i < args.size(); i++) { String arg = args.get(i); ToolOption o = options.getOption(arg); if (o != null) { // handle a doclet argument that may be needed however // don't increment the index, and allow the tool to consume args - handleDocletOption(i, args, true); + if (consumeDocletOption(i, args, true) < 0) { + success = false; + } if (o.hasArg) { if (arg.startsWith("--") && arg.contains("=")) { o.process(arg.substring(arg.indexOf('=') + 1)); @@ -800,14 +818,19 @@ public class Start { String s = arg.substring("-XD".length()); int eq = s.indexOf('='); String key = (eq < 0) ? s : s.substring(0, eq); - String value = (eq < 0) ? s : s.substring(eq+1); + String value = (eq < 0) ? s : s.substring(eq + 1); options.compilerOptions().put(key, value); } else if (arg.startsWith("-")) { - i = handleDocletOption(i, args, false); + i = consumeDocletOption(i, args, false); + if (i < 0) { + i = -i; + success = false; + } } else { javaNames.add(arg); } } + return success; } private boolean isEmpty(Iterable iter) { diff --git a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/ToolOptions.java b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/ToolOptions.java index 1fe138b3804..160d4c01660 100644 --- a/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/ToolOptions.java +++ b/src/jdk.javadoc/share/classes/jdk/javadoc/internal/tool/ToolOptions.java @@ -696,7 +696,7 @@ public class ToolOptions { } void setDumpOnError(boolean v) { - dumpOnError = true; + dumpOnError = v; } /** @@ -816,7 +816,7 @@ public class ToolOptions { /** * Returns an {@code IllegalOptionValue} exception. * - * @param arg the arghument to include in the detail message + * @param arg the argument to include in the detail message * @return the exception */ private IllegalOptionValue illegalOptionValue(String arg) { diff --git a/test/langtools/jdk/javadoc/tool/8224613/OptionProcessingFailureTest.java b/test/langtools/jdk/javadoc/tool/8224613/OptionProcessingFailureTest.java new file mode 100644 index 00000000000..dab4144d335 --- /dev/null +++ b/test/langtools/jdk/javadoc/tool/8224613/OptionProcessingFailureTest.java @@ -0,0 +1,335 @@ +/* + * Copyright (c) 2020, 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 8224613 + * @library /tools/lib ../../lib + * @modules jdk.javadoc/jdk.javadoc.internal.tool + * @build javadoc.tester.* toolbox.ToolBox builder.ClassBuilder + * @run main/othervm OptionProcessingFailureTest + */ + +import builder.ClassBuilder; +import javadoc.tester.JavadocTester; +import jdk.javadoc.doclet.Doclet; +import jdk.javadoc.doclet.DocletEnvironment; +import jdk.javadoc.doclet.Reporter; +import toolbox.ToolBox; + +import javax.lang.model.SourceVersion; +import javax.tools.Diagnostic; +import java.io.IOException; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashSet; +import java.util.List; +import java.util.Locale; +import java.util.Objects; +import java.util.Set; +import java.util.stream.Collectors; + +/* + * Tests that a particular error is raised only if a doclet has not reported any + * errors (to Reporter), and yet at least one of that Doclet's options returned + * `false` from the `Doclet.Option.process` method. + * + * This test explores scenarios generated by combining a few independent factors + * involved in failing a doclet run. These factors are: + * + * 1. Reporting errors in Doclet.init(...) + * 2. Reporting errors in Doclet.Option.process(...) + * 3. Returning `false` from Doclet.Option.process(...) + * + * A doclet that behaves according to a scenario is run by the javadoc tool. An + * output of that run is then examined for presence of a particular error. That + * error must be in the output if and only if one or more options returned + * `false` from Doclet.Option.process(...) and no other errors were reported. + * + * Configuration of the doclet is performed out-of-band, using System Properties + * (when running the javadoc tool from the command line this could be achieved + * using -J-Dsystem.property.name=value syntax). The "puppet" doclet is + * instructed on which options is has, how many errors it should report, + * and how each individual option should be processed. + */ +public class OptionProcessingFailureTest extends JavadocTester { + + private final ToolBox tb = new ToolBox(); + + public static void main(String... args) throws Exception { + new OptionProcessingFailureTest().runTests(m -> new Object[]{Paths.get(m.getName())}); + } + + @Test + public void test(Path base) throws IOException { + Path srcDir = base.resolve("src"); + + // Since the minimal successful run of the javadoc tool with a custom + // doclet involves source files, a package with a class in it is generated: + new ClassBuilder(tb, "pkg.A") + .setModifiers("public", "class") + .write(srcDir); + + generateScenarios(base, this::testScenario); + } + + private static void generateScenarios(Path base, ScenarioConsumer consumer) { + for (int nInitErrors : List.of(0, 1)) { // the number of errors the Doclet.init method should report + for (int nOptions : List.of(0, 1, 2)) { // the number of options a doclet should have + generateOptionsCombinations(base, nInitErrors, nOptions, consumer); + } + } + } + + private static void generateOptionsCombinations(Path base, + int nInitErrors, + int nOptions, + ScenarioConsumer consumer) { + var emptyDescriptions = new PuppetOption.Description[nOptions]; + generateOptionsCombinations(base, nInitErrors, 0, emptyDescriptions, consumer); + } + + + private static void generateOptionsCombinations(Path base, + int nInitErrors, + int descriptionsIndex, + PuppetOption.Description[] descriptions, + ScenarioConsumer consumer) { + if (descriptionsIndex >= descriptions.length) { + consumer.consume(base, nInitErrors, List.of(descriptions)); + return; + } + for (boolean success : List.of(true, false)) { // return values of Doclet.Options.process + for (int nErrors : List.of(0, 1)) { // the number of errors Doclet.Options.process should report + String optionName = "--doclet-option-" + descriptionsIndex; + descriptions[descriptionsIndex] = new PuppetOption.Description(optionName, success, nErrors); + generateOptionsCombinations(base, nInitErrors, descriptionsIndex + 1, descriptions, consumer); + } + } + } + + private void testScenario(Path base, + int nInitErrors, + List optionDescriptions) { + System.out.printf("nInitErrors=%s, optionDescriptions=%s%n", nInitErrors, optionDescriptions); + + List args = new ArrayList<>( + List.of("-doclet", PuppetDoclet.class.getName(), + "-docletpath", System.getProperty("test.classes", "."), + "-sourcepath", base.resolve("src").toString(), + "pkg")); + + optionDescriptions.forEach(d -> args.add(d.name)); // options passed to the doclet + + /* The puppet doclet does not create any output directories, so there is + no need for any related checks; other checks are disabled to speed up + the processing and reduce the size of the output. */ + + setOutputDirectoryCheck(DirectoryCheck.NONE); + setAutomaticCheckAccessibility(false); + setAutomaticCheckLinks(false); + + /* Ideally, the system properties should've been passed to the `javadoc` + method below. However, since there's no such option, the system + properties are manipulated in an external fashion. */ + + String descriptions = System.getProperty("puppet.descriptions"); + if (descriptions != null) { + throw new IllegalStateException(descriptions); + } + String errors = System.getProperty("puppet.errors"); + if (errors != null) { + throw new IllegalStateException(errors); + } + try { + System.setProperty("puppet.descriptions", PuppetDoclet.descriptionsToString(optionDescriptions)); + System.setProperty("puppet.errors", String.valueOf(nInitErrors)); + javadoc(args.toArray(new String[0])); + } finally { + System.clearProperty("puppet.descriptions"); + System.clearProperty("puppet.errors"); + } + + long sumErrors = optionDescriptions.stream().mapToLong(d -> d.nProcessErrors).sum() + nInitErrors; + boolean success = optionDescriptions.stream().allMatch(d -> d.success); + + checkOutput(Output.OUT, sumErrors != 0 || !success, "error - "); + } + + /* Creating a specialized consumer is even more lightweight than creating a POJO */ + @FunctionalInterface + public interface ScenarioConsumer { + void consume(Path src, int nInitErrors, List optionDescriptions); + } + + public static final class PuppetDoclet implements Doclet { + + private final int nErrorsInInit; + private final Set descriptions; + private Set