Skip to content

Commit b9a29eb

Browse files
Roman MarchenkoPaul Hohensee
Roman Marchenko
authored and
Paul Hohensee
committedJul 5, 2023
8297437: javadoc cannot link to old docs (with old style anchors)
Reviewed-by: phh Backport-of: 15a14884013a975707008f648b8e4864e16006ed
1 parent dbf8820 commit b9a29eb

File tree

6 files changed

+104
-61
lines changed

6 files changed

+104
-61
lines changed
 

‎src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/formats/html/markup/Links.java

+3-39
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1997, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1997, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -29,6 +29,7 @@
2929
import jdk.javadoc.internal.doclets.toolkit.Content;
3030
import jdk.javadoc.internal.doclets.toolkit.util.DocLink;
3131
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
32+
import jdk.javadoc.internal.doclets.toolkit.util.Extern;
3233

3334
/**
3435
* Factory for HTML A elements, both links (with a {@code href} attribute)
@@ -334,44 +335,7 @@ public String getName(String name) {
334335
return name.replaceAll(" +", "");
335336
}
336337

337-
StringBuilder sb = new StringBuilder();
338-
for (int i = 0; i < name.length(); i++) {
339-
char ch = name.charAt(i);
340-
switch (ch) {
341-
case '(':
342-
case ')':
343-
case '<':
344-
case '>':
345-
case ',':
346-
sb.append('-');
347-
break;
348-
case ' ':
349-
case '[':
350-
break;
351-
case ']':
352-
sb.append(":A");
353-
break;
354-
// Any appearance of $ needs to be substituted with ":D" and not with hyphen
355-
// since a field name "P$$ and a method P(), both valid member names, can end
356-
// up as "P--". A member name beginning with $ needs to be substituted with
357-
// "Z:Z:D".
358-
case '$':
359-
if (i == 0)
360-
sb.append("Z:Z");
361-
sb.append(":D");
362-
break;
363-
// A member name beginning with _ needs to be prefixed with "Z:Z" since valid anchor
364-
// names can only begin with a letter.
365-
case '_':
366-
if (i == 0)
367-
sb.append("Z:Z");
368-
sb.append(ch);
369-
break;
370-
default:
371-
sb.append(ch);
372-
}
373-
}
374-
return sb.toString();
338+
return Extern.getOldFormHtmlName(name);
375339
}
376340

377341
}

‎src/jdk.javadoc/share/classes/jdk/javadoc/internal/doclets/toolkit/util/Extern.java

+91-12
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -98,6 +98,11 @@ private static class Item {
9898
*/
9999
final boolean relative;
100100

101+
/**
102+
* Indicates that docs use old-form of anchors.
103+
*/
104+
final boolean useOldFormId;
105+
101106
/**
102107
* Constructor to build a Extern Item object and map it with the element name.
103108
* If the same element name is found in the map, then the first mapped
@@ -108,10 +113,11 @@ private static class Item {
108113
* file is picked.
109114
* @param relative True if path is URL, false if directory path.
110115
*/
111-
Item(String elementName, DocPath path, boolean relative) {
116+
Item(String elementName, DocPath path, boolean relative, boolean useOldFormId) {
112117
this.elementName = elementName;
113118
this.path = path;
114119
this.relative = relative;
120+
this.useOldFormId = useOldFormId;
115121
}
116122

117123
/**
@@ -179,7 +185,7 @@ public DocLink getExternalLink(Element element, DocPath relativepath, String fil
179185
DocPath p = fnd.relative ?
180186
relativepath.resolve(fnd.path).resolve(filename) :
181187
fnd.path.resolve(filename);
182-
return new DocLink(p, "is-external=true", memberName);
188+
return new DocLink(p, "is-external=true", fnd.useOldFormId ? getOldFormHtmlName(memberName) : memberName);
183189
}
184190

185191
/**
@@ -212,6 +218,18 @@ public boolean link(String url, String elemlisturl, Reporter reporter) throws Do
212218
return link(url, elemlisturl, reporter, true);
213219
}
214220

221+
/**
222+
* Checks if platform docs for the specified version use old-form anchors.
223+
* Old-form anchors are used by Oracle docs for JDKs 8 and 9.
224+
* It can be checked on https://docs.oracle.com/javase/<version>/docs/api
225+
*
226+
* @param version
227+
* @return True if docs use old-form anchors
228+
*/
229+
private boolean isOldFormPlatformDocs(int version) {
230+
return 8 == version || 9 == version;
231+
}
232+
215233
/*
216234
* Build the extern element list from given URL or the directory path.
217235
* Flag error if the "-link" or "-linkoffline" option is already used.
@@ -292,7 +310,7 @@ private String adjustEndFileSeparator(String url) {
292310
private void readElementListFromURL(String urlpath, URL elemlisturlpath) throws Fault {
293311
try {
294312
URL link = elemlisturlpath.toURI().resolve(DocPaths.ELEMENT_LIST.getPath()).toURL();
295-
readElementList(link.openStream(), urlpath, false);
313+
readElementList(link.openStream(), urlpath, false, false);
296314
} catch (URISyntaxException | MalformedURLException exc) {
297315
throw new Fault(configuration.getText("doclet.MalformedURL", elemlisturlpath.toString()), exc);
298316
} catch (IOException exc) {
@@ -309,7 +327,7 @@ private void readElementListFromURL(String urlpath, URL elemlisturlpath) throws
309327
private void readAlternateURL(String urlpath, URL elemlisturlpath) throws Fault {
310328
try {
311329
URL link = elemlisturlpath.toURI().resolve(DocPaths.PACKAGE_LIST.getPath()).toURL();
312-
readElementList(link.openStream(), urlpath, false);
330+
readElementList(link.openStream(), urlpath, false, true);
313331
} catch (URISyntaxException | MalformedURLException exc) {
314332
throw new Fault(configuration.getText("doclet.MalformedURL", elemlisturlpath.toString()), exc);
315333
} catch (IOException exc) {
@@ -332,27 +350,27 @@ private void readElementListFromFile(String path, DocFile elemListPath)
332350
file = file.resolveAgainst(DocumentationTool.Location.DOCUMENTATION_OUTPUT);
333351
}
334352
if (file.exists()) {
335-
readElementList(file, path);
353+
readElementList(file, path, false);
336354
} else {
337355
DocFile file1 = elemListPath.resolve(DocPaths.PACKAGE_LIST);
338356
if (!(file1.isAbsolute() || linkoffline)) {
339357
file1 = file1.resolveAgainst(DocumentationTool.Location.DOCUMENTATION_OUTPUT);
340358
}
341359
if (file1.exists()) {
342-
readElementList(file1, path);
360+
readElementList(file1, path, true);
343361
} else {
344362
throw new Fault(configuration.getText("doclet.File_error", file.getPath()), null);
345363
}
346364
}
347365
}
348366

349-
private void readElementList(DocFile file, String path) throws Fault, DocFileIOException {
367+
private void readElementList(DocFile file, String path, boolean isOldFormDoc) throws Fault, DocFileIOException {
350368
try {
351369
if (file.canRead()) {
352370
boolean pathIsRelative
353371
= !isUrl(path)
354372
&& !DocFile.createFileForInput(configuration, path).isAbsolute();
355-
readElementList(file.openInputStream(), path, pathIsRelative);
373+
readElementList(file.openInputStream(), path, pathIsRelative, isOldFormDoc);
356374
} else {
357375
throw new Fault(configuration.getText("doclet.File_error", file.getPath()), null);
358376
}
@@ -370,7 +388,7 @@ private void readElementList(DocFile file, String path) throws Fault, DocFileIOE
370388
* @param relative Is path relative?
371389
* @throws IOException if there is a problem reading or closing the stream
372390
*/
373-
private void readElementList(InputStream input, String path, boolean relative)
391+
private void readElementList(InputStream input, String path, boolean relative, boolean isOldFormDoc)
374392
throws IOException {
375393
try (BufferedReader in = new BufferedReader(new InputStreamReader(input))) {
376394
String elemname = null;
@@ -382,7 +400,7 @@ private void readElementList(InputStream input, String path, boolean relative)
382400
elempath = basePath;
383401
if (elemname.startsWith(DocletConstants.MODULE_PREFIX)) {
384402
moduleName = elemname.replace(DocletConstants.MODULE_PREFIX, "");
385-
Item item = new Item(moduleName, elempath, relative);
403+
Item item = new Item(moduleName, elempath, relative, isOldFormDoc);
386404
moduleItems.put(moduleName, item);
387405
} else {
388406
DocPath pkgPath = DocPath.create(elemname.replace('.', '/'));
@@ -392,7 +410,7 @@ private void readElementList(InputStream input, String path, boolean relative)
392410
elempath = elempath.resolve(pkgPath);
393411
}
394412
String actualModuleName = checkLinkCompatibility(elemname, moduleName, path);
395-
Item item = new Item(elemname, elempath, relative);
413+
Item item = new Item(elemname, elempath, relative, isOldFormDoc);
396414
packageItems.computeIfAbsent(actualModuleName, k -> new TreeMap<>())
397415
.put(elemname, item);
398416
}
@@ -455,4 +473,65 @@ private String checkLinkCompatibility(String packageName, String moduleName, Str
455473
}
456474
return moduleName == null ? DocletConstants.DEFAULT_ELEMENT_NAME : moduleName;
457475
}
476+
477+
/**
478+
* Converts a name to an old-form HTML name (old-form id).
479+
*
480+
* @param name the string that needs to be converted to a valid HTML name
481+
* @return old-form HTML name
482+
*/
483+
public static String getOldFormHtmlName(String name) {
484+
/* The HTML 4 spec at http://www.w3.org/TR/html4/types.html#h-6.2 mentions
485+
* that the name/id should begin with a letter followed by other valid characters.
486+
* The HTML 5 spec (draft) is more permissive on names/ids where the only restriction
487+
* is that it should be at least one character long and should not contain spaces.
488+
* The spec draft is @ http://www.w3.org/html/wg/drafts/html/master/dom.html#the-id-attribute.
489+
*
490+
* For HTML 4, we need to check for non-characters at the beginning of the name and
491+
* substitute it accordingly, "_" and "$" can appear at the beginning of a member name.
492+
* The method substitutes "$" with "Z:Z:D" and will prefix "_" with "Z:Z".
493+
*/
494+
495+
if (null == name)
496+
return name;
497+
498+
StringBuilder sb = new StringBuilder();
499+
for (int i = 0; i < name.length(); i++) {
500+
char ch = name.charAt(i);
501+
switch (ch) {
502+
case '(':
503+
case ')':
504+
case '<':
505+
case '>':
506+
case ',':
507+
sb.append('-');
508+
break;
509+
case ' ':
510+
case '[':
511+
break;
512+
case ']':
513+
sb.append(":A");
514+
break;
515+
// Any appearance of $ needs to be substituted with ":D" and not with hyphen
516+
// since a field name "P$$ and a method P(), both valid member names, can end
517+
// up as "P--". A member name beginning with $ needs to be substituted with
518+
// "Z:Z:D".
519+
case '$':
520+
if (i == 0)
521+
sb.append("Z:Z");
522+
sb.append(":D");
523+
break;
524+
// A member name beginning with _ needs to be prefixed with "Z:Z" since valid anchor
525+
// names can only begin with a letter.
526+
case '_':
527+
if (i == 0)
528+
sb.append("Z:Z");
529+
sb.append(ch);
530+
break;
531+
default:
532+
sb.append(ch);
533+
}
534+
}
535+
return sb.toString();
536+
}
458537
}

‎test/langtools/jdk/javadoc/doclet/testClassCrossReferences/TestClassCrossReferences.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2002, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2002, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -59,7 +59,7 @@ void test() {
5959
+ "title=\"class or interface in javax.swing.text\" class=\"externalLink\"><code>Link to AttributeContext innerclass</code></a>",
6060
"<a href=\"" + uri + "java/math/BigDecimal.html?is-external=true\" "
6161
+ "title=\"class or interface in java.math\" class=\"externalLink\"><code>Link to external class BigDecimal</code></a>",
62-
"<a href=\"" + uri + "java/math/BigInteger.html?is-external=true#gcd(java.math.BigInteger)\" "
62+
"<a href=\"" + uri + "java/math/BigInteger.html?is-external=true#gcd-java.math.BigInteger-\" "
6363
+ "title=\"class or interface in java.math\" class=\"externalLink\"><code>Link to external member gcd</code></a>",
6464
"<a href=\"" + uri + "javax/tools/SimpleJavaFileObject.html?is-external=true#uri\" "
6565
+ "title=\"class or interface in javax.tools\" class=\"externalLink\"><code>Link to external member URI</code></a>",
@@ -90,7 +90,7 @@ void test_warning() {
9090
+ "title=\"class or interface in javax.swing.text\" class=\"externalLink\"><code>Link to AttributeContext innerclass</code></a>",
9191
"<a href=\"" + uri + "java/math/BigDecimal.html?is-external=true\" "
9292
+ "title=\"class or interface in java.math\" class=\"externalLink\"><code>Link to external class BigDecimal</code></a>",
93-
"<a href=\"" + uri + "java/math/BigInteger.html?is-external=true#gcd(java.math.BigInteger)\" "
93+
"<a href=\"" + uri + "java/math/BigInteger.html?is-external=true#gcd-java.math.BigInteger-\" "
9494
+ "title=\"class or interface in java.math\" class=\"externalLink\"><code>Link to external member gcd</code></a>",
9595
"<a href=\"" + uri + "javax/tools/SimpleJavaFileObject.html?is-external=true#uri\" "
9696
+ "title=\"class or interface in javax.tools\" class=\"externalLink\"><code>Link to external member URI</code></a>",

‎test/langtools/jdk/javadoc/doclet/testExternalOverridenMethod/TestExternalOverridenMethod.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2003, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -53,12 +53,12 @@ void test() {
5353

5454
checkOutput("pkg/XReader.html", true,
5555
"<dt><span class=\"overrideSpecifyLabel\">Overrides:</span></dt>\n"
56-
+ "<dd><code><a href=\"" + uri + "/java/io/FilterReader.html?is-external=true#read()\" "
56+
+ "<dd><code><a href=\"" + uri + "/java/io/FilterReader.html?is-external=true#read--\" "
5757
+ "title=\"class or interface in java.io\" class=\"externalLink\">read</a></code>&nbsp;in class&nbsp;<code>"
5858
+ "<a href=\"" + uri + "/java/io/FilterReader.html?is-external=true\" "
5959
+ "title=\"class or interface in java.io\" class=\"externalLink\">FilterReader</a></code></dd>",
6060
"<dt><span class=\"overrideSpecifyLabel\">Specified by:</span></dt>\n"
61-
+ "<dd><code><a href=\"" + uri + "/java/io/DataInput.html?is-external=true#readInt()\" "
61+
+ "<dd><code><a href=\"" + uri + "/java/io/DataInput.html?is-external=true#readInt--\" "
6262
+ "title=\"class or interface in java.io\" class=\"externalLink\">readInt</a></code>&nbsp;in interface&nbsp;<code>"
6363
+ "<a href=\"" + uri + "/java/io/DataInput.html?is-external=true\" "
6464
+ "title=\"class or interface in java.io\" class=\"externalLink\">DataInput</a></code></dd>"

‎test/langtools/jdk/javadoc/doclet/testHref/TestHref.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2003, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -51,7 +51,7 @@ void test() {
5151

5252
checkOutput("pkg/C1.html", true,
5353
//External link.
54-
"href=\"http://java.sun.com/j2se/1.4/docs/api/java/lang/Object.html?is-external=true#wait(long,int)\"",
54+
"href=\"http://java.sun.com/j2se/1.4/docs/api/java/lang/Object.html?is-external=true#wait-long-int-\"",
5555
//Member summary table link.
5656
"href=\"#method(int,int,java.util.ArrayList)\"",
5757
//Anchor test.

‎test/langtools/jdk/javadoc/doclet/testLinkOption/TestLinkOption.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2002, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2002, 2023, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -85,7 +85,7 @@ void test() {
8585

8686
checkOutput("pkg/B.html", true,
8787
"<div class=\"block\">A method with html tag the method "
88-
+ "<a href=\"" + url + "java/lang/ClassLoader.html?is-external=true#getSystemClassLoader()\""
88+
+ "<a href=\"" + url + "java/lang/ClassLoader.html?is-external=true#getSystemClassLoader--\""
8989
+ " title=\"class or interface in java.lang\" class=\"externalLink\"><code><tt>getSystemClassLoader()</tt>"
9090
+ "</code></a> as the parent class loader.</div>",
9191
"<div class=\"block\">is equivalent to invoking <code>"

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Jul 5, 2023

@openjdk-notifier[bot]
Please sign in to comment.