From 2612bef51e2b7ac961e2c4a0d7c4eac66465583e Mon Sep 17 00:00:00 2001 From: Jonathan Gibbons Date: Tue, 15 May 2018 14:41:06 -0700 Subject: [PATCH] 8202614: Add ability to validate links in JavadocTester Reviewed-by: ksrini --- .../jdk/javadoc/doclet/5093723/T5093723.java | 1 + .../doclet/DocRootSlash/DocRootSlash.java | 4 +- .../jdk/javadoc/doclet/lib/JavadocTester.java | 887 +++++++++++++++++- .../testAnchorNames/TestAnchorNames.java | 2 + .../doclet/testHrefInDocComment/pkg/I1.java | 6 +- .../doclet/testHtmlTag/TestHtmlTag.java | 4 +- .../javadoc/doclet/testHtmlTag/pkg3/A.java | 4 +- .../testHtmlVersion/TestHtmlVersion.java | 6 + .../doclet/testModules/TestModules.java | 2 + .../TestPackageDeprecation.java | 3 +- .../javadoc/doclet/testSearch/TestSearch.java | 2 + .../doclet/testUseOption/TestUseOption.java | 1 + 12 files changed, 911 insertions(+), 11 deletions(-) diff --git a/test/langtools/jdk/javadoc/doclet/5093723/T5093723.java b/test/langtools/jdk/javadoc/doclet/5093723/T5093723.java index 3fa71fb0937..46bcf0a565b 100644 --- a/test/langtools/jdk/javadoc/doclet/5093723/T5093723.java +++ b/test/langtools/jdk/javadoc/doclet/5093723/T5093723.java @@ -40,6 +40,7 @@ public class T5093723 extends JavadocTester { @Test void test() { + setAutomaticCheckLinks(false); // @ignore JDK-8202617 javadoc("-d", "out", "-Xdoclint:none", testSrc("DocumentedClass.java"), diff --git a/test/langtools/jdk/javadoc/doclet/DocRootSlash/DocRootSlash.java b/test/langtools/jdk/javadoc/doclet/DocRootSlash/DocRootSlash.java index a4612f24153..15e3028d41f 100644 --- a/test/langtools/jdk/javadoc/doclet/DocRootSlash/DocRootSlash.java +++ b/test/langtools/jdk/javadoc/doclet/DocRootSlash/DocRootSlash.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2018, 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 @@ -51,6 +51,8 @@ public class DocRootSlash extends JavadocTester { // Directory that contains source files that javadoc runs on String srcdir = System.getProperty("test.src", "."); + setAutomaticCheckLinks(false); // @ignore JDK-8202621 + javadoc("-d", "out", "-Xdoclint:none", "-overview", (srcdir + "/overview.html"), diff --git a/test/langtools/jdk/javadoc/doclet/lib/JavadocTester.java b/test/langtools/jdk/javadoc/doclet/lib/JavadocTester.java index dd3d64b3544..71a09d003e9 100644 --- a/test/langtools/jdk/javadoc/doclet/lib/JavadocTester.java +++ b/test/langtools/jdk/javadoc/doclet/lib/JavadocTester.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2018, 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 @@ -21,15 +21,18 @@ * questions. */ +import java.io.BufferedReader; 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.InputStreamReader; import java.io.IOException; import java.io.PrintStream; import java.io.PrintWriter; +import java.io.StringReader; import java.io.StringWriter; import java.lang.annotation.Annotation; import java.lang.annotation.Retention; @@ -37,19 +40,36 @@ import java.lang.annotation.RetentionPolicy; import java.lang.ref.SoftReference; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; +import java.net.URI; +import java.net.URISyntaxException; import java.nio.charset.Charset; +import java.nio.charset.CharsetDecoder; +import java.nio.charset.CodingErrorAction; import java.nio.charset.UnsupportedCharsetException; +import java.nio.file.FileVisitResult; import java.nio.file.Files; -import java.util.Arrays; +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.LinkedHashMap; import java.util.List; +import java.util.Locale; import java.util.Map; import java.util.Objects; +import java.util.Set; +import java.util.TreeMap; +import java.util.TreeSet; import java.util.function.Function; +import java.util.regex.Pattern; +import java.util.stream.Collectors; /** @@ -140,6 +160,7 @@ public abstract class JavadocTester { public static final String FS = System.getProperty("file.separator"); public static final String PS = System.getProperty("path.separator"); public static final String NL = System.getProperty("line.separator"); + public static final Path currDir = Paths.get(".").toAbsolutePath().normalize(); public enum Output { /** The name of the output stream from javadoc. */ @@ -227,6 +248,8 @@ public abstract class JavadocTester { private DirectoryCheck outputDirectoryCheck = DirectoryCheck.EMPTY; + private boolean automaticCheckLinks = true; + /** The current subtest number. Incremented when checking(...) is called. */ private int numTestsRun = 0; @@ -371,6 +394,10 @@ public abstract class JavadocTester { out.println(text); } }); + + if (automaticCheckLinks && exitCode == Exit.OK.code && outputDir.exists()) { + checkLinks(); + } } /** @@ -383,6 +410,13 @@ public abstract class JavadocTester { outputDirectoryCheck = c; } + /** + * Set whether or not to perform an automatic call of checkLinks. + */ + public void setAutomaticCheckLinks(boolean b) { + automaticCheckLinks = b; + } + /** * The exit codes returned by the javadoc tool. * @see jdk.javadoc.internal.tool.Main.Result @@ -493,6 +527,23 @@ public abstract class JavadocTester { } } + public void checkLinks() { + checking("Check links"); + LinkChecker c = new LinkChecker(out, this::readFile); + try { + c.checkDirectory(outputDir.toPath()); + c.report(); + int errors = c.getErrorCount(); + if (errors == 0) { + passed("Links are OK"); + } else { + failed(errors + " errors found when checking links"); + } + } catch (IOException e) { + failed("exception thrown when reading files: " + e); + } + } + /** * Get the content of the one of the output streams written by javadoc. * @param output the name of the output stream @@ -675,6 +726,19 @@ public abstract class JavadocTester { return readFile(new File(baseDir), fileName); } + private String readFile(Path file) { + File baseDir; + if (file.startsWith(outputDir.toPath())) { + baseDir = outputDir; + } else if (file.startsWith(currDir)) { + baseDir = currDir.toFile(); + } else { + baseDir = file.getParent().toFile(); + } + String fileName = baseDir.toPath().relativize(file).toString(); + return readFile(baseDir, fileName); + } + /** * Read the file and return it as a string. * @@ -910,4 +974,823 @@ public abstract class JavadocTester { } } } + + // Support classes for checkLinks + + /** + * A basic HTML parser. Override the protected methods as needed to get notified + * of significant items in any file that is read. + */ + static abstract class HtmlParser { + + protected final PrintStream out; + protected final Function fileReader; + + private Path file; + private StringReader in; + private int ch; + private int lineNumber; + private boolean inScript; + private boolean xml; + + HtmlParser(PrintStream out, Function fileReader) { + this.out = out; + this.fileReader = fileReader; + } + + /** + * Read a file. + * @param file the file to be read + * @throws IOException if an error occurs while reading the file + */ + void read(Path file) throws IOException { + try (StringReader r = new StringReader(fileReader.apply(file))) { + this.file = file; + this.in = r; + + startFile(file); + try { + lineNumber = 1; + xml = false; + nextChar(); + + while (ch != -1) { + switch (ch) { + + case '<': + html(); + break; + + default: + nextChar(); + } + } + } finally { + endFile(); + } + } catch (IOException e) { + error(file, lineNumber, e); + } catch (Throwable t) { + error(file, lineNumber, t); + t.printStackTrace(out); + } + } + + + int getLineNumber() { + return lineNumber; + } + + /** + * Called when a file has been opened, before parsing begins. + * This is always the first notification when reading a file. + * This implementation does nothing. + * + * @param file the file + */ + protected void startFile(Path file) { } + + /** + * Called when the parser has finished reading a file. + * This is always the last notification when reading a file, + * unless any errors occur while closing the file. + * This implementation does nothing. + */ + protected void endFile() { } + + /** + * Called when a doctype declaration is found, at the beginning of the file. + * This implementation does nothing. + * @param s the doctype declaration + */ + protected void docType(String s) { } + + /** + * Called when the opening tag of an HTML element is encountered. + * This implementation does nothing. + * @param name the name of the tag + * @param attrs the attribute + * @param selfClosing whether or not this is a self-closing tag + */ + protected void startElement(String name, Map attrs, boolean selfClosing) { } + + /** + * Called when the closing tag of an HTML tag is encountered. + * This implementation does nothing. + * @param name the name of the tag + */ + protected void endElement(String name) { } + + /** + * Called when an error has been encountered. + * @param file the file being read + * @param lineNumber the line number of line containing the error + * @param message a description of the error + */ + protected void error(Path file, int lineNumber, String message) { + out.println(file + ":" + lineNumber + ": " + message); + } + + /** + * Called when an exception has been encountered. + * @param file the file being read + * @param lineNumber the line number of the line being read when the exception was found + * @param t the exception + */ + protected void error(Path file, int lineNumber, Throwable t) { + out.println(file + ":" + lineNumber + ": " + t); + } + + private void nextChar() throws IOException { + ch = in.read(); + if (ch == '\n') + lineNumber++; + } + + /** + * Read the start or end of an HTML tag, or an HTML comment + * {@literal } or {@literal } + * @throws java.io.IOException if there is a problem reading the file + */ + private void html() throws IOException { + nextChar(); + if (isIdentifierStart((char) ch)) { + String name = readIdentifier().toLowerCase(Locale.US); + Map attrs = htmlAttrs(); + if (attrs != null) { + boolean selfClosing = false; + if (ch == '/') { + nextChar(); + selfClosing = true; + } + if (ch == '>') { + nextChar(); + startElement(name, attrs, selfClosing); + if (name.equals("script")) { + inScript = true; + } + return; + } + } + } else if (ch == '/') { + nextChar(); + if (isIdentifierStart((char) ch)) { + String name = readIdentifier().toLowerCase(Locale.US); + skipWhitespace(); + if (ch == '>') { + nextChar(); + endElement(name); + if (name.equals("script")) { + inScript = false; + } + return; + } + } + } else if (ch == '!') { + nextChar(); + if (ch == '-') { + nextChar(); + if (ch == '-') { + nextChar(); + while (ch != -1) { + int dash = 0; + while (ch == '-') { + dash++; + nextChar(); + } + // Strictly speaking, a comment should not contain "--" + // so dash > 2 is an error, dash == 2 implies ch == '>' + // See http://www.w3.org/TR/html-markup/syntax.html#syntax-comments + // for more details. + if (dash >= 2 && ch == '>') { + nextChar(); + return; + } + + nextChar(); + } + } + } else if (ch == '[') { + nextChar(); + if (ch == 'C') { + nextChar(); + if (ch == 'D') { + nextChar(); + if (ch == 'A') { + nextChar(); + if (ch == 'T') { + nextChar(); + if (ch == 'A') { + nextChar(); + if (ch == '[') { + while (true) { + nextChar(); + if (ch == ']') { + nextChar(); + if (ch == ']') { + nextChar(); + if (ch == '>') { + nextChar(); + return; + } + } + } + } + + } + } + } + } + } + } + } else { + StringBuilder sb = new StringBuilder(); + while (ch != -1 && ch != '>') { + sb.append((char) ch); + nextChar(); + } + Pattern p = Pattern.compile("(?is)doctype\\s+html\\s?.*"); + String s = sb.toString(); + if (p.matcher(s).matches()) { + docType(s); + return; + } + } + } else if (ch == '?') { + nextChar(); + if (ch == 'x') { + nextChar(); + if (ch == 'm') { + nextChar(); + if (ch == 'l') { + Map attrs = htmlAttrs(); + if (ch == '?') { + nextChar(); + if (ch == '>') { + nextChar(); + xml = true; + return; + } + } + } + } + + } + } + + if (!inScript) { + error(file, lineNumber, "bad html"); + } + } + + /** + * Read a series of HTML attributes, terminated by {@literal > }. + * Each attribute is of the form {@literal identifier[=value] }. + * "value" may be unquoted, single-quoted, or double-quoted. + */ + private Map htmlAttrs() throws IOException { + Map map = new LinkedHashMap<>(); + skipWhitespace(); + + loop: + while (isIdentifierStart((char) ch)) { + String name = readAttributeName().toLowerCase(Locale.US); + skipWhitespace(); + String value = null; + if (ch == '=') { + nextChar(); + skipWhitespace(); + if (ch == '\'' || ch == '"') { + char quote = (char) ch; + nextChar(); + StringBuilder sb = new StringBuilder(); + while (ch != -1 && ch != quote) { + sb.append((char) ch); + nextChar(); + } + value = sb.toString() // hack to replace common entities + .replace("<", "<") + .replace(">", ">") + .replace("&", "&"); + nextChar(); + } else { + StringBuilder sb = new StringBuilder(); + while (ch != -1 && !isUnquotedAttrValueTerminator((char) ch)) { + sb.append((char) ch); + nextChar(); + } + value = sb.toString(); + } + skipWhitespace(); + } + map.put(name, value); + } + + return map; + } + + private boolean isIdentifierStart(char ch) { + return Character.isUnicodeIdentifierStart(ch); + } + + private String readIdentifier() throws IOException { + StringBuilder sb = new StringBuilder(); + sb.append((char) ch); + nextChar(); + while (ch != -1 && Character.isUnicodeIdentifierPart(ch)) { + sb.append((char) ch); + nextChar(); + } + return sb.toString(); + } + + private String readAttributeName() throws IOException { + StringBuilder sb = new StringBuilder(); + sb.append((char) ch); + nextChar(); + while (ch != -1 && Character.isUnicodeIdentifierPart(ch) + || ch == '-' + || xml && ch == ':') { + sb.append((char) ch); + nextChar(); + } + return sb.toString(); + } + + private boolean isWhitespace(char ch) { + return Character.isWhitespace(ch); + } + + private void skipWhitespace() throws IOException { + while (isWhitespace((char) ch)) { + nextChar(); + } + } + + private boolean isUnquotedAttrValueTerminator(char ch) { + switch (ch) { + case '\f': case '\n': case '\r': case '\t': + case ' ': + case '"': case '\'': case '`': + case '=': case '<': case '>': + return true; + default: + return false; + } + } + } + + /** + * A class to check the links in a set of HTML files. + */ + static class LinkChecker extends HtmlParser { + private final Map allFiles; + private final Map allURIs; + + private int files; + private int links; + private int badSchemes; + private int duplicateIds; + private int missingIds; + + private Path currFile; + private IDTable currTable; + private boolean html5; + private boolean xml; + + private int errors; + + LinkChecker(PrintStream out, Function fileReader) { + super(out, fileReader); + allFiles = new HashMap<>(); + allURIs = new HashMap<>(); + } + + void checkDirectory(Path dir) throws IOException { + checkFiles(List.of(dir), false, Collections.emptySet()); + } + + void checkFiles(List files, boolean skipSubdirs, Set excludeFiles) throws IOException { + for (Path file : files) { + Files.walkFileTree(file, new SimpleFileVisitor() { + int depth = 0; + + @Override + public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) { + if ((skipSubdirs && depth > 0) || excludeFiles.contains(dir)) { + return FileVisitResult.SKIP_SUBTREE; + } + depth++; + return FileVisitResult.CONTINUE; + } + + @Override + public FileVisitResult visitFile(Path p, BasicFileAttributes attrs) { + if (excludeFiles.contains(p)) { + return FileVisitResult.CONTINUE; + } + + if (Files.isRegularFile(p) && p.getFileName().toString().endsWith(".html")) { + checkFile(p); + } + return FileVisitResult.CONTINUE; + } + + @Override + public FileVisitResult postVisitDirectory(Path dir, IOException e) throws IOException { + depth--; + return super.postVisitDirectory(dir, e); + } + }); + } + } + + void checkFile(Path file) { + try { + read(file); + } catch (IOException e) { + error(file, 0, e); + } + } + + int getErrorCount() { + return errors; + } + + public void report() { + List missingFiles = getMissingFiles(); + if (!missingFiles.isEmpty()) { + report("Missing files: (" + missingFiles.size() + ")"); + missingFiles.stream() + .sorted() + .forEach(this::reportMissingFile); + + } + + if (!allURIs.isEmpty()) { + report(false, "External URLs:"); + allURIs.keySet().stream() + .sorted(new URIComparator()) + .forEach(uri -> report(false, " %s", uri.toString())); + } + + int anchors = 0; + for (IDTable t : allFiles.values()) { + anchors += t.map.values().stream() + .filter(e -> !e.getReferences().isEmpty()) + .count(); + } + for (IDTable t : allURIs.values()) { + anchors += t.map.values().stream() + .filter(e -> !e.references.isEmpty()) + .count(); + } + + report(false, "Checked " + files + " files."); + report(false, "Found " + links + " references to " + anchors + " anchors " + + "in " + allFiles.size() + " files and " + allURIs.size() + " other URIs."); + report(!missingFiles.isEmpty(), "%6d missing files", missingFiles.size()); + report(duplicateIds > 0, "%6d duplicate ids", duplicateIds); + report(missingIds > 0, "%6d missing ids", missingIds); + + Map schemeCounts = new TreeMap<>(); + Map hostCounts = new TreeMap<>(new HostComparator()); + for (URI uri : allURIs.keySet()) { + String scheme = uri.getScheme(); + if (scheme != null) { + schemeCounts.put(scheme, schemeCounts.computeIfAbsent(scheme, s -> 0) + 1); + } + String host = uri.getHost(); + if (host != null) { + hostCounts.put(host, hostCounts.computeIfAbsent(host, h -> 0) + 1); + } + } + + if (schemeCounts.size() > 0) { + report(false, "Schemes"); + schemeCounts.forEach((s, n) -> report(!isSchemeOK(s), "%6d %s", n, s)); + } + + if (hostCounts.size() > 0) { + report(false, "Hosts"); + hostCounts.forEach((h, n) -> report(false, "%6d %s", n, h)); + } + } + + private void report(String message, Object... args) { + out.println(String.format(message, args)); + } + + private void report(boolean highlight, String message, Object... args) { + out.print(highlight ? "* " : " "); + out.println(String.format(message, args)); + } + + private void reportMissingFile(Path file) { + report("%s", relativePath(file)); + IDTable table = allFiles.get(file); + Set refs = new TreeSet<>(); + for (ID id : table.map.values()) { + if (id.references != null) { + for (Position p : id.references) { + refs.add(p.path); + } + } + } + int n = 0; + int MAX_REFS = 10; + for (Path ref : refs) { + report(" in " + relativePath(ref)); + if (++n == MAX_REFS) { + report(" ... and %d more", refs.size() - n); + break; + } + } + } + + @Override + public void startFile(Path path) { + currFile = path.toAbsolutePath().normalize(); + currTable = allFiles.computeIfAbsent(currFile, p -> new IDTable(p)); + html5 = false; + files++; + } + + @Override + public void endFile() { + currTable.check(); + } + + @Override + public void docType(String doctype) { + html5 = doctype.matches("(?i)<\\?doctype\\s+html>"); + } + + @Override @SuppressWarnings("fallthrough") + public void startElement(String name, Map attrs, boolean selfClosing) { + int line = getLineNumber(); + switch (name) { + case "a": + String nameAttr = html5 ? null : attrs.get("name"); + if (nameAttr != null) { + foundAnchor(line, nameAttr); + } + // fallthrough + case "link": + String href = attrs.get("href"); + if (href != null) { + foundReference(line, href); + } + break; + } + + String idAttr = attrs.get("id"); + if (idAttr != null) { + foundAnchor(line, idAttr); + } + } + + @Override + public void endElement(String name) { } + + private void foundAnchor(int line, String name) { + currTable.addID(line, name); + } + + private void foundReference(int line, String ref) { + links++; + try { + URI uri = new URI(ref); + if (uri.isAbsolute()) { + foundReference(line, uri); + } else { + Path p; + String uriPath = uri.getPath(); + if (uriPath == null || uriPath.isEmpty()) { + p = currFile; + } else { + p = currFile.getParent().resolve(uriPath).normalize(); + } + foundReference(line, p, uri.getFragment()); + } + } catch (URISyntaxException e) { + error(currFile, line, "invalid URI: " + e); + } + } + + private void foundReference(int line, Path p, String fragment) { + IDTable t = allFiles.computeIfAbsent(p, key -> new IDTable(key)); + t.addReference(fragment, currFile, line); + } + + private void foundReference(int line, URI uri) { + if (!isSchemeOK(uri.getScheme())) { + error(currFile, line, "bad scheme in URI"); + badSchemes++; + } + + String fragment = uri.getFragment(); + try { + URI noFrag = new URI(uri.toString().replaceAll("#\\Q" + fragment + "\\E$", "")); + IDTable t = allURIs.computeIfAbsent(noFrag, key -> new IDTable(key.toString())); + t.addReference(fragment, currFile, line); + } catch (URISyntaxException e) { + throw new Error(e); + } + } + + private boolean isSchemeOK(String uriScheme) { + if (uriScheme == null) { + return true; + } + + switch (uriScheme) { + case "file": + case "ftp": + case "http": + case "https": + case "javascript": + case "mailto": + return true; + + default: + return false; + } + } + + private List getMissingFiles() { + return allFiles.entrySet().stream() + .filter(e -> !Files.exists(e.getKey())) + .map(e -> e.getKey()) + .collect(Collectors.toList()); + } + + @Override + protected void error(Path file, int lineNumber, String message) { + super.error(relativePath(file), lineNumber, message); + errors++; + } + + @Override + protected void error(Path file, int lineNumber, Throwable t) { + super.error(relativePath(file), lineNumber, t); + errors++; + } + + private Path relativePath(Path path) { + return path.startsWith(currDir) ? currDir.relativize(path) : path; + } + + /** + * A position in a file, as identified by a file name and line number. + */ + static class Position implements Comparable { + Path path; + int line; + + Position(Path path, int line) { + this.path = path; + this.line = line; + } + + @Override + public int compareTo(Position o) { + int v = path.compareTo(o.path); + return v != 0 ? v : Integer.compare(line, o.line); + } + + @Override + public boolean equals(Object obj) { + if (this == obj) { + return true; + } else if (obj == null || getClass() != obj.getClass()) { + return false; + } else { + final Position other = (Position) obj; + return Objects.equals(this.path, other.path) + && this.line == other.line; + } + } + + @Override + public int hashCode() { + return Objects.hashCode(path) * 37 + line; + } + } + + /** + * Infor for an ID within an HTML file, and a set of positions that reference it. + */ + static class ID { + boolean declared; + Set references; + + Set getReferences() { + return (references) == null ? Collections.emptySet() : references; + } + } + + /** + * A table for the set of IDs in an HTML file. + */ + class IDTable { + private String name; + private boolean checked; + private final Map map = new HashMap<>(); + + IDTable(Path p) { + this(relativePath(p).toString()); + } + + IDTable(String name) { + this.name = name; + } + + void addID(int line, String name) { + if (checked) { + throw new IllegalStateException("Adding ID after file has been read"); + } + Objects.requireNonNull(name); + ID id = map.computeIfAbsent(name, x -> new ID()); + if (id.declared) { + error(currFile, line, "name already declared: " + name); + duplicateIds++; + } else { + id.declared = true; + } + } + + void addReference(String name, Path from, int line) { + if (checked) { + if (name != null) { + ID id = map.get(name); + if (id == null || !id.declared) { + error(from, line, "id not found: " + this.name + "#" + name); + } + } + } else { + ID id = map.computeIfAbsent(name, x -> new ID()); + if (id.references == null) { + id.references = new TreeSet<>(); + } + id.references.add(new Position(from, line)); + } + } + + void check() { + map.forEach((name, id) -> { + if (name != null && !id.declared) { + //log.error(currFile, 0, "id not declared: " + name); + for (Position ref : id.references) { + error(ref.path, ref.line, "id not found: " + this.name + "#" + name); + } + missingIds++; + } + }); + checked = true; + } + } + + static class URIComparator implements Comparator { + final HostComparator hostComparator = new HostComparator(); + + @Override + public int compare(URI o1, URI o2) { + if (o1.isOpaque() || o2.isOpaque()) { + return o1.compareTo(o2); + } + String h1 = o1.getHost(); + String h2 = o2.getHost(); + String s1 = o1.getScheme(); + String s2 = o2.getScheme(); + if (h1 == null || h1.isEmpty() || s1 == null || s1.isEmpty() + || h2 == null || h2.isEmpty() || s2 == null || s2.isEmpty()) { + return o1.compareTo(o2); + } + int v = hostComparator.compare(h1, h2); + if (v != 0) { + return v; + } + v = s1.compareTo(s2); + if (v != 0) { + return v; + } + return o1.compareTo(o2); + } + } + + static class HostComparator implements Comparator { + @Override + public int compare(String h1, String h2) { + List l1 = new ArrayList<>(Arrays.asList(h1.split("\\."))); + Collections.reverse(l1); + String r1 = String.join(".", l1); + List l2 = new ArrayList<>(Arrays.asList(h2.split("\\."))); + Collections.reverse(l2); + String r2 = String.join(".", l2); + return r1.compareTo(r2); + } + } + + } } diff --git a/test/langtools/jdk/javadoc/doclet/testAnchorNames/TestAnchorNames.java b/test/langtools/jdk/javadoc/doclet/testAnchorNames/TestAnchorNames.java index 9eed8b8905d..c12f26fc79c 100644 --- a/test/langtools/jdk/javadoc/doclet/testAnchorNames/TestAnchorNames.java +++ b/test/langtools/jdk/javadoc/doclet/testAnchorNames/TestAnchorNames.java @@ -52,12 +52,14 @@ public class TestAnchorNames extends JavadocTester { @Test void testHtml4(Path ignore) { + setAutomaticCheckLinks(false); // @ignore JDK-8202622 javadoc("-d", "out-html4", "-html4", "-sourcepath", testSrc, "-source", "8", //so that '_' can be used as an identifier "-use", "pkg1"); + setAutomaticCheckLinks(true); // @ignore JDK-8202622 checkExit(Exit.OK); // Test some section markers and links to these markers diff --git a/test/langtools/jdk/javadoc/doclet/testHrefInDocComment/pkg/I1.java b/test/langtools/jdk/javadoc/doclet/testHrefInDocComment/pkg/I1.java index c734943e6bc..04b7c180423 100644 --- a/test/langtools/jdk/javadoc/doclet/testHrefInDocComment/pkg/I1.java +++ b/test/langtools/jdk/javadoc/doclet/testHrefInDocComment/pkg/I1.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2002, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2002, 2018, 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 @@ -24,7 +24,7 @@ package pkg; /** - * Here's the first - * and here's the second. + * Here's the first + * and here's the second. */ public interface I1 {} diff --git a/test/langtools/jdk/javadoc/doclet/testHtmlTag/TestHtmlTag.java b/test/langtools/jdk/javadoc/doclet/testHtmlTag/TestHtmlTag.java index 247f1da8f8f..ce573bbe718 100644 --- a/test/langtools/jdk/javadoc/doclet/testHtmlTag/TestHtmlTag.java +++ b/test/langtools/jdk/javadoc/doclet/testHtmlTag/TestHtmlTag.java @@ -118,7 +118,7 @@ public class TestHtmlTag extends JavadocTester { + " Factory that creates new javax.xml.datatype\n" + " Objects that map XML to/from Java Objects.

\n" + "\n" - + "

\n" + + "

\n" + " A new instance of the DatatypeFactory is created through the\n" + " newInstance() method that uses the following implementation\n" + " resolution mechanisms to determine an implementation:

\n" @@ -215,7 +215,7 @@ public class TestHtmlTag extends JavadocTester { + " Factory that creates new javax.xml.datatype\n" + " Objects that map XML to/from Java Objects.

\n" + "\n" - + "

\n" + + "

\n" + " A new instance of the DatatypeFactory is created through the\n" + " newInstance() method that uses the following implementation\n" + " resolution mechanisms to determine an implementation:

\n" diff --git a/test/langtools/jdk/javadoc/doclet/testHtmlTag/pkg3/A.java b/test/langtools/jdk/javadoc/doclet/testHtmlTag/pkg3/A.java index ee7813957de..f6b9e0ce4f6 100644 --- a/test/langtools/jdk/javadoc/doclet/testHtmlTag/pkg3/A.java +++ b/test/langtools/jdk/javadoc/doclet/testHtmlTag/pkg3/A.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2015, 2018, 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 @@ -33,7 +33,7 @@ public class A { * Factory that creates new javax.xml.datatype * Objects that map XML to/from Java Objects.

* - *

+ *

* A new instance of the DatatypeFactory is created through the * {@link #newInstance()} method that uses the following implementation * resolution mechanisms to determine an implementation:

diff --git a/test/langtools/jdk/javadoc/doclet/testHtmlVersion/TestHtmlVersion.java b/test/langtools/jdk/javadoc/doclet/testHtmlVersion/TestHtmlVersion.java index 0e24a1ea9c4..da0cb945a0f 100644 --- a/test/langtools/jdk/javadoc/doclet/testHtmlVersion/TestHtmlVersion.java +++ b/test/langtools/jdk/javadoc/doclet/testHtmlVersion/TestHtmlVersion.java @@ -41,12 +41,14 @@ public class TestHtmlVersion extends JavadocTester { @Test void test1() { + setAutomaticCheckLinks(false); // @ignore JDK-8202624 javadoc("-d", "out-1", "-private", "-linksource", "-sourcepath", testSrc, "-use", "pkg", "pkg1", "pkg2", "pkg3"); + setAutomaticCheckLinks(true); // @ignore JDK-8202624 checkExit(Exit.OK); html5Output(); @@ -55,6 +57,7 @@ public class TestHtmlVersion extends JavadocTester { @Test void test2() { + setAutomaticCheckLinks(false); // @ignore JDK-8202624 javadoc("-d", "out-2", "-html4", "-private", @@ -62,6 +65,7 @@ public class TestHtmlVersion extends JavadocTester { "-sourcepath", testSrc, "-use", "pkg", "pkg1", "pkg2", "pkg3"); + setAutomaticCheckLinks(true); // @ignore JDK-8202624 checkExit(Exit.OK); html4Output(); @@ -70,6 +74,7 @@ public class TestHtmlVersion extends JavadocTester { @Test void test3() { + setAutomaticCheckLinks(false); // @ignore JDK-8202624 javadoc("-d", "out-3", "-html4", "-private", @@ -77,6 +82,7 @@ public class TestHtmlVersion extends JavadocTester { "-sourcepath", testSrc, "-use", "pkg", "pkg1", "pkg2", "pkg3"); + setAutomaticCheckLinks(true); // @ignore JDK-8202624 checkExit(Exit.OK); html4Output(); diff --git a/test/langtools/jdk/javadoc/doclet/testModules/TestModules.java b/test/langtools/jdk/javadoc/doclet/testModules/TestModules.java index d557b1c4dc2..2a8a2372eda 100644 --- a/test/langtools/jdk/javadoc/doclet/testModules/TestModules.java +++ b/test/langtools/jdk/javadoc/doclet/testModules/TestModules.java @@ -244,11 +244,13 @@ public class TestModules extends JavadocTester { */ @Test void testAggregatorModuleSummary() { + setAutomaticCheckLinks(false); // @ignore JDK-8202628 javadoc("-d", "out-aggregatorModuleSummary", "-use", "--module-source-path", testSrc, "--expand-requires", "transitive", "--module", "moduleT"); + setAutomaticCheckLinks(true); // @ignore JDK-8202628 checkExit(Exit.OK); checkAggregatorModuleSummary(); } diff --git a/test/langtools/jdk/javadoc/doclet/testPackageDeprecation/TestPackageDeprecation.java b/test/langtools/jdk/javadoc/doclet/testPackageDeprecation/TestPackageDeprecation.java index 1d3f52d28e3..cefefca26f3 100644 --- a/test/langtools/jdk/javadoc/doclet/testPackageDeprecation/TestPackageDeprecation.java +++ b/test/langtools/jdk/javadoc/doclet/testPackageDeprecation/TestPackageDeprecation.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2011, 2017, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2011, 2018, 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 @@ -36,6 +36,7 @@ public class TestPackageDeprecation extends JavadocTester { public static void main(String... args) throws Exception { TestPackageDeprecation tester = new TestPackageDeprecation(); + tester.setAutomaticCheckLinks(false); // @ignore JDK-8202626 tester.runTests(); } diff --git a/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java b/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java index faa571826ad..ac5405b907e 100644 --- a/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java +++ b/test/langtools/jdk/javadoc/doclet/testSearch/TestSearch.java @@ -232,12 +232,14 @@ public class TestSearch extends JavadocTester { @Test void test7() { + setAutomaticCheckLinks(false); // @ignore JDK-8202627 javadoc("-d", "out-7", "-nodeprecated", "-Xdoclint:none", "-sourcepath", testSrc, "-use", "pkg", "pkg1", "pkg2", "pkg3"); + setAutomaticCheckLinks(true); // @ignore JDK-8202627 checkExit(Exit.OK); checkSearchOutput(true); checkIndexNoDeprecated(); diff --git a/test/langtools/jdk/javadoc/doclet/testUseOption/TestUseOption.java b/test/langtools/jdk/javadoc/doclet/testUseOption/TestUseOption.java index 1694dcc9b53..c4b463edc91 100644 --- a/test/langtools/jdk/javadoc/doclet/testUseOption/TestUseOption.java +++ b/test/langtools/jdk/javadoc/doclet/testUseOption/TestUseOption.java @@ -37,6 +37,7 @@ public class TestUseOption extends JavadocTester { public static void main(String... args) throws Exception { TestUseOption tester = new TestUseOption(); + tester.setAutomaticCheckLinks(false); // @ignore JDK-8202626 tester.runTests(); }