8317621: --add-script should support JavaScript modules
Reviewed-by: jjg
This commit is contained in:
parent
4eb1eaf044
commit
9bb6169a1c
@ -25,8 +25,11 @@
|
|||||||
|
|
||||||
package jdk.javadoc.internal.doclets.formats.html;
|
package jdk.javadoc.internal.doclets.formats.html;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
import java.nio.file.InvalidPathException;
|
import java.nio.file.InvalidPathException;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.time.ZonedDateTime;
|
import java.time.ZonedDateTime;
|
||||||
@ -41,6 +44,7 @@ import java.util.Map;
|
|||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
import javax.lang.model.element.Element;
|
import javax.lang.model.element.Element;
|
||||||
import javax.lang.model.element.PackageElement;
|
import javax.lang.model.element.PackageElement;
|
||||||
@ -64,6 +68,7 @@ import jdk.javadoc.internal.doclets.toolkit.Messages;
|
|||||||
import jdk.javadoc.internal.doclets.toolkit.Resources;
|
import jdk.javadoc.internal.doclets.toolkit.Resources;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DeprecatedAPIListBuilder;
|
import jdk.javadoc.internal.doclets.toolkit.util.DeprecatedAPIListBuilder;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DocFile;
|
import jdk.javadoc.internal.doclets.toolkit.util.DocFile;
|
||||||
|
import jdk.javadoc.internal.doclets.toolkit.util.DocFileIOException;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
|
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DocPaths;
|
import jdk.javadoc.internal.doclets.toolkit.util.DocPaths;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.NewAPIBuilder;
|
import jdk.javadoc.internal.doclets.toolkit.util.NewAPIBuilder;
|
||||||
@ -198,6 +203,18 @@ public class HtmlConfiguration extends BaseConfiguration {
|
|||||||
*/
|
*/
|
||||||
private final Set<PackageElement> containingPackagesSeen;
|
private final Set<PackageElement> containingPackagesSeen;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List of additional JavaScript files
|
||||||
|
*/
|
||||||
|
private List<JavaScriptFile> additionalScripts;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Record for JavaScript file and module flag.
|
||||||
|
* @param path file path
|
||||||
|
* @param isModule module flag
|
||||||
|
*/
|
||||||
|
public record JavaScriptFile(DocPath path, boolean isModule) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructs the full configuration needed by the doclet, including
|
* Constructs the full configuration needed by the doclet, including
|
||||||
* the format-specific part, defined in this class, and the format-independent
|
* the format-specific part, defined in this class, and the format-independent
|
||||||
@ -316,6 +333,9 @@ public class HtmlConfiguration extends BaseConfiguration {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
additionalScripts = options.additionalScripts().stream()
|
||||||
|
.map(this::detectJSModule)
|
||||||
|
.collect(Collectors.toList());
|
||||||
if (options.createIndex()) {
|
if (options.createIndex()) {
|
||||||
indexBuilder = new HtmlIndexBuilder(this);
|
indexBuilder = new HtmlIndexBuilder(this);
|
||||||
}
|
}
|
||||||
@ -326,6 +346,26 @@ public class HtmlConfiguration extends BaseConfiguration {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private JavaScriptFile detectJSModule(String fileName) {
|
||||||
|
DocFile file = DocFile.createFileForInput(this, fileName);
|
||||||
|
boolean isModule = fileName.toLowerCase(Locale.ROOT).endsWith(".mjs");
|
||||||
|
if (!isModule) {
|
||||||
|
// Regex to detect JavaScript modules
|
||||||
|
Pattern modulePattern = Pattern.compile("""
|
||||||
|
(?:^|[;}])\\s*(?:\
|
||||||
|
import\\s*["']|\
|
||||||
|
import[\\s{*][^()]*from\\s*["']|\
|
||||||
|
export(?:\\s+(?:let|const|function|class|var|default|async)|\\s*[{*]))""");
|
||||||
|
try (InputStream in = file.openInputStream();
|
||||||
|
BufferedReader reader = new BufferedReader(new InputStreamReader(in))) {
|
||||||
|
isModule = reader.lines().anyMatch(s -> modulePattern.matcher(s).find());
|
||||||
|
} catch (DocFileIOException | IOException e) {
|
||||||
|
// Errors are handled when copying resources
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new JavaScriptFile(DocPath.create(file.getName()), isModule);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@return the date to be recorded in generated files}
|
* {@return the date to be recorded in generated files}
|
||||||
*/
|
*/
|
||||||
@ -421,11 +461,8 @@ public class HtmlConfiguration extends BaseConfiguration {
|
|||||||
.collect(Collectors.toCollection(ArrayList::new));
|
.collect(Collectors.toCollection(ArrayList::new));
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<DocPath> getAdditionalScripts() {
|
public List<JavaScriptFile> getAdditionalScripts() {
|
||||||
return options.additionalScripts().stream()
|
return additionalScripts;
|
||||||
.map(sf -> DocFile.createFileForInput(this, sf))
|
|
||||||
.map(file -> DocPath.create(file.getName()))
|
|
||||||
.collect(Collectors.toCollection(ArrayList::new));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -35,6 +35,7 @@ import java.util.List;
|
|||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
|
||||||
import jdk.javadoc.internal.doclets.formats.html.Content;
|
import jdk.javadoc.internal.doclets.formats.html.Content;
|
||||||
|
import jdk.javadoc.internal.doclets.formats.html.HtmlConfiguration;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
|
import jdk.javadoc.internal.doclets.toolkit.util.DocPath;
|
||||||
import jdk.javadoc.internal.doclets.toolkit.util.DocPaths;
|
import jdk.javadoc.internal.doclets.toolkit.util.DocPaths;
|
||||||
|
|
||||||
@ -60,7 +61,7 @@ public class Head extends Content {
|
|||||||
private Script mainBodyScript;
|
private Script mainBodyScript;
|
||||||
private final List<Script> scripts;
|
private final List<Script> scripts;
|
||||||
// Scripts added via --add-script option
|
// Scripts added via --add-script option
|
||||||
private List<DocPath> additionalScripts = List.of();
|
private List<HtmlConfiguration.JavaScriptFile> additionalScripts = List.of();
|
||||||
private final List<Content> extraContent;
|
private final List<Content> extraContent;
|
||||||
private boolean addDefaultScript = true;
|
private boolean addDefaultScript = true;
|
||||||
private DocPath canonicalLink;
|
private DocPath canonicalLink;
|
||||||
@ -176,7 +177,7 @@ public class Head extends Content {
|
|||||||
* @param scripts the list of additional script files
|
* @param scripts the list of additional script files
|
||||||
* @return this object
|
* @return this object
|
||||||
*/
|
*/
|
||||||
public Head setAdditionalScripts(List<DocPath> scripts) {
|
public Head setAdditionalScripts(List<HtmlConfiguration.JavaScriptFile> scripts) {
|
||||||
this.additionalScripts = scripts;
|
this.additionalScripts = scripts;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@ -346,7 +347,7 @@ public class Head extends Content {
|
|||||||
|
|
||||||
private void addScripts(HtmlTree head) {
|
private void addScripts(HtmlTree head) {
|
||||||
if (addDefaultScript) {
|
if (addDefaultScript) {
|
||||||
addScriptElement(head, DocPaths.SCRIPT_FILES.resolve(DocPaths.SCRIPT_JS));
|
addScriptElement(head, DocPaths.SCRIPT_JS);
|
||||||
}
|
}
|
||||||
if (index) {
|
if (index) {
|
||||||
if (pathToRoot != null && mainBodyScript != null) {
|
if (pathToRoot != null && mainBodyScript != null) {
|
||||||
@ -356,11 +357,11 @@ public class Head extends Content {
|
|||||||
.append(";\n")
|
.append(";\n")
|
||||||
.append("loadScripts(document, 'script');");
|
.append("loadScripts(document, 'script');");
|
||||||
}
|
}
|
||||||
addScriptElement(head, DocPaths.SCRIPT_FILES.resolve(DocPaths.JQUERY_JS));
|
addScriptElement(head, DocPaths.JQUERY_JS);
|
||||||
addScriptElement(head, DocPaths.SCRIPT_FILES.resolve(DocPaths.JQUERY_UI_JS));
|
addScriptElement(head, DocPaths.JQUERY_UI_JS);
|
||||||
}
|
}
|
||||||
for (DocPath path : additionalScripts) {
|
for (HtmlConfiguration.JavaScriptFile javaScriptFile : additionalScripts) {
|
||||||
addScriptElement(head, DocPaths.SCRIPT_FILES.resolve(path));
|
addScriptElement(head, javaScriptFile);
|
||||||
}
|
}
|
||||||
for (Script script : scripts) {
|
for (Script script : scripts) {
|
||||||
head.add(script.asContent());
|
head.add(script.asContent());
|
||||||
@ -368,7 +369,13 @@ public class Head extends Content {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void addScriptElement(HtmlTree head, DocPath filePath) {
|
private void addScriptElement(HtmlTree head, DocPath filePath) {
|
||||||
DocPath scriptFile = pathToRoot.resolve(filePath);
|
DocPath scriptFile = pathToRoot.resolve(DocPaths.SCRIPT_FILES).resolve(filePath);
|
||||||
head.add(HtmlTree.SCRIPT(scriptFile.getPath()));
|
head.add(HtmlTree.SCRIPT(scriptFile.getPath()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addScriptElement(HtmlTree head, HtmlConfiguration.JavaScriptFile script) {
|
||||||
|
DocPath scriptFile = pathToRoot.resolve(DocPaths.SCRIPT_FILES).resolve(script.path());
|
||||||
|
HtmlTree scriptTag = HtmlTree.SCRIPT(scriptFile.getPath());
|
||||||
|
head.add(script.isModule() ? scriptTag.put(HtmlAttr.TYPE, "module") : scriptTag);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,149 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, 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 8317621
|
||||||
|
* @summary --add-script should support JavaScript modules
|
||||||
|
* @library /tools/lib ../../lib
|
||||||
|
* @modules jdk.javadoc/jdk.javadoc.internal.tool
|
||||||
|
* @build toolbox.ToolBox javadoc.tester.*
|
||||||
|
* @run main TestJavaScriptModules
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
|
||||||
|
import javadoc.tester.JavadocTester;
|
||||||
|
import toolbox.ToolBox;
|
||||||
|
|
||||||
|
public class TestJavaScriptModules extends JavadocTester {
|
||||||
|
|
||||||
|
public static void main(String... args) throws Exception {
|
||||||
|
var tester = new TestJavaScriptModules();
|
||||||
|
tester.setup().runTests();
|
||||||
|
}
|
||||||
|
|
||||||
|
private final ToolBox tb = new ToolBox();
|
||||||
|
Path src;
|
||||||
|
|
||||||
|
TestJavaScriptModules setup() throws IOException {
|
||||||
|
src = Path.of("src");
|
||||||
|
tb.writeJavaFiles(src, """
|
||||||
|
/**
|
||||||
|
* Simple dummy class.
|
||||||
|
*/
|
||||||
|
public class Test {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module.mjs", """
|
||||||
|
var x = 1;
|
||||||
|
""");
|
||||||
|
tb.writeFile("module1.js", """
|
||||||
|
const x = 1;
|
||||||
|
export class FooModule {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module2.js", """
|
||||||
|
const x = 1;
|
||||||
|
export function f() {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module3.js", """
|
||||||
|
const x = 1;
|
||||||
|
export async function a() {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module4.js", """
|
||||||
|
// Another JS module
|
||||||
|
export const c = 3;
|
||||||
|
""");
|
||||||
|
tb.writeFile("module5.js", """
|
||||||
|
export default class FooModule {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module6.js", """
|
||||||
|
const x = 1;
|
||||||
|
export class FooModule {}
|
||||||
|
""");
|
||||||
|
tb.writeFile("module7.js", """
|
||||||
|
function abc() {}
|
||||||
|
import * as foo from "module1.js";
|
||||||
|
""");
|
||||||
|
tb.writeFile("module8.js", """
|
||||||
|
var z = false;
|
||||||
|
import { _A_, $b, C0 } from "abc.js";
|
||||||
|
""");
|
||||||
|
tb.writeFile("script1.js", """
|
||||||
|
var z = false;
|
||||||
|
import(1, z);
|
||||||
|
""");
|
||||||
|
tb.writeFile("script2.js", """
|
||||||
|
export("foo");
|
||||||
|
""");
|
||||||
|
tb.writeFile("script3.js", """
|
||||||
|
var import = 1;
|
||||||
|
""");
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test(Path base) {
|
||||||
|
javadoc("-d", base.resolve("out").toString(),
|
||||||
|
"--add-script", "module.mjs",
|
||||||
|
"--add-script", "module1.js",
|
||||||
|
"--add-script", "module2.js",
|
||||||
|
"--add-script", "module3.js",
|
||||||
|
"--add-script", "module4.js",
|
||||||
|
"--add-script", "module5.js",
|
||||||
|
"--add-script", "module6.js",
|
||||||
|
"--add-script", "module7.js",
|
||||||
|
"--add-script", "module8.js",
|
||||||
|
"--add-script", "script1.js",
|
||||||
|
"--add-script", "script2.js",
|
||||||
|
"--add-script", "script3.js",
|
||||||
|
src.resolve("Test.java").toString());
|
||||||
|
checkExit(Exit.OK);
|
||||||
|
|
||||||
|
checkOutput("Test.html", true,
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module.mjs"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module1.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module2.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module3.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module4.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module5.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module6.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module7.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="module" src="script-files/module8.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="text/javascript" src="script-files/script1.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="text/javascript" src="script-files/script2.js"></script>""",
|
||||||
|
"""
|
||||||
|
<script type="text/javascript" src="script-files/script3.js"></script>""");
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user