2021-05-27 15:01:03 +00:00
|
|
|
/*
|
2024-02-12 21:10:19 +00:00
|
|
|
* Copyright (c) 2021, 2024, Oracle and/or its affiliates. All rights reserved.
|
2021-05-27 15:01:03 +00:00
|
|
|
* 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 8267574
|
|
|
|
* @summary check stylesheet names against HtmlStyle
|
2024-08-02 17:53:23 +00:00
|
|
|
* @modules jdk.javadoc/jdk.javadoc.internal.html
|
|
|
|
* jdk.javadoc/jdk.javadoc.internal.doclets.formats.html.markup
|
2023-08-09 07:01:15 +00:00
|
|
|
* jdk.javadoc/jdk.javadoc.internal.doclets.formats.html.resources:open
|
2021-05-27 15:01:03 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.BufferedWriter;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.PrintWriter;
|
|
|
|
import java.net.URL;
|
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.nio.file.Path;
|
|
|
|
import java.util.Arrays;
|
|
|
|
import java.util.Set;
|
|
|
|
import java.util.TreeSet;
|
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
2024-08-02 17:53:23 +00:00
|
|
|
import jdk.javadoc.internal.html.HtmlStyle;
|
|
|
|
import jdk.javadoc.internal.doclets.formats.html.markup.HtmlStyles;
|
2021-05-27 15:01:03 +00:00
|
|
|
|
|
|
|
/**
|
2024-08-02 17:53:23 +00:00
|
|
|
* This test compares the set of CSS class names defined in HtmlStyles
|
2021-05-27 15:01:03 +00:00
|
|
|
* and other files (such as search.js) against the set of CSS class names
|
|
|
|
* defined in the main stylesheet.css provided by the doclet.
|
|
|
|
*
|
|
|
|
* The goal is to detect "unexpected" discrepancies between the two sets.
|
|
|
|
* "Expected" discrepancies are taken into account, but may indicate a
|
|
|
|
* need to resolve the discrepancy.
|
|
|
|
*
|
|
|
|
* The test does not take into direct account the recent introduction of
|
|
|
|
* CSS constructs like section {@code [class$="-details"]}
|
|
|
|
*/
|
|
|
|
public class CheckStylesheetClasses {
|
|
|
|
public static void main(String... args) throws Exception {
|
|
|
|
CheckStylesheetClasses c = new CheckStylesheetClasses();
|
|
|
|
c.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
int errors = 0;
|
|
|
|
|
|
|
|
void run() throws Exception {
|
|
|
|
Set<String> htmlStyleNames = getHtmlStyleNames();
|
|
|
|
Set<String> styleSheetNames = getStylesheetNames();
|
|
|
|
|
|
|
|
System.err.println("found " + htmlStyleNames.size() + " names in HtmlStyle");
|
|
|
|
System.err.println("found " + styleSheetNames.size() + " names in stylesheet");
|
|
|
|
|
|
|
|
// Write the lists to external files for the benefit of external diff tools:
|
|
|
|
// for example, to compare against the CSS class names used in generated documentation.
|
|
|
|
// To find the classes used in a directory containing HTML files, use something like
|
|
|
|
// find $DIRECTORY -name \*.html | \
|
|
|
|
// xargs grep -o 'class="[^"]*"' | \
|
|
|
|
// sed -e 's/^[^"]*"//' -e 's/".*$//' | \
|
|
|
|
// while read line ; do for w in $line ; do echo $w ; done ; done | \
|
|
|
|
// sort -u
|
|
|
|
|
|
|
|
try (BufferedWriter out = Files.newBufferedWriter(Path.of("htmlStyleNames.txt"));
|
|
|
|
PrintWriter pw = new PrintWriter(out)) {
|
|
|
|
htmlStyleNames.forEach(pw::println);
|
|
|
|
}
|
|
|
|
|
|
|
|
try (BufferedWriter out = Files.newBufferedWriter(Path.of("styleSheetNames.txt"));
|
|
|
|
PrintWriter pw = new PrintWriter(out)) {
|
|
|
|
styleSheetNames.forEach(pw::println);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove names from htmlStyleNames if they are valid names generated by the doclet,
|
|
|
|
// even if they do not by default require a style to be defined in the stylesheet.
|
|
|
|
// In general, names in these lists are worthy of attention to see if they *should*
|
|
|
|
// be defined in the stylesheet, especially when the names exist in a family of
|
|
|
|
// related items: related by name or by function.
|
|
|
|
|
|
|
|
// the page names are provided to override a style on a specific page;
|
|
|
|
// only some are used in the stylesheet
|
|
|
|
htmlStyleNames.removeIf(s -> s.endsWith("-page") && !styleSheetNames.contains(s));
|
|
|
|
|
|
|
|
// descriptions; class-description is used;
|
|
|
|
// surprisingly? module-description and package-description are not
|
|
|
|
htmlStyleNames.removeIf(s -> s.endsWith("-description") && !styleSheetNames.contains(s));
|
|
|
|
|
|
|
|
// help page
|
|
|
|
htmlStyleNames.removeIf(s -> s.startsWith("help-") && !styleSheetNames.contains(s));
|
|
|
|
|
|
|
|
// summary and details tables; styles for these may be present in the stylesheet
|
|
|
|
// using constructs like these:
|
|
|
|
// .summary section[class$="-summary"], .details section[class$="-details"],
|
2021-10-05 16:57:12 +00:00
|
|
|
htmlStyleNames.removeIf(s -> s.endsWith("-details") && !styleSheetNames.contains(s));
|
2021-05-27 15:01:03 +00:00
|
|
|
htmlStyleNames.removeIf(s -> s.endsWith("-summary") && !styleSheetNames.contains(s));
|
|
|
|
|
|
|
|
// signature classes
|
|
|
|
removeAll(htmlStyleNames, "annotations", "element-name", "extends-implements",
|
|
|
|
"modifiers", "permits", "return-type");
|
|
|
|
|
|
|
|
// misc: these are defined in HtmlStyle, and used by the doclet
|
2022-09-30 11:11:28 +00:00
|
|
|
removeAll(htmlStyleNames, "col-plain", "external-link", "header",
|
2021-05-27 15:01:03 +00:00
|
|
|
"hierarchy", "index", "package-uses", "packages", "permits-note",
|
|
|
|
"serialized-package-container", "source-container");
|
|
|
|
|
|
|
|
// Remove names from styleSheetNames if they are false positives,
|
|
|
|
// or used by other code (i.e. not HtmlStyle),
|
|
|
|
// or if they are unused and therefore candidates to be deleted.
|
|
|
|
|
|
|
|
// false positives: file extensions and URL components
|
2023-04-03 10:37:19 +00:00
|
|
|
removeAll(styleSheetNames, "css", "png", "w3", "org");
|
2021-05-27 15:01:03 +00:00
|
|
|
|
|
|
|
// for doc-comment authors; maybe worthy of inclusion in HtmlStyle, just to be documented
|
|
|
|
removeAll(styleSheetNames, "borderless", "plain", "striped");
|
|
|
|
|
2022-05-06 08:56:42 +00:00
|
|
|
// used in search.js and search-page.js; may be worth documenting in HtmlStyle
|
2023-05-25 07:30:03 +00:00
|
|
|
removeAll(styleSheetNames, "result-highlight", "result-item", "anchor-link",
|
2022-05-06 08:56:42 +00:00
|
|
|
"search-tag-desc-result", "search-tag-holder-result", "page-search-header",
|
2023-11-20 18:32:20 +00:00
|
|
|
"ui-autocomplete", "ui-autocomplete-category", "ui-state-active", "ui-menu",
|
|
|
|
"ui-menu-item-wrapper", "ui-static-link", "expanded", "search-result-link",
|
|
|
|
"two-column-search-results", "sort-asc", "sort-desc", "visible");
|
2021-05-27 15:01:03 +00:00
|
|
|
|
|
|
|
// very JDK specific
|
|
|
|
styleSheetNames.remove("module-graph");
|
2022-11-03 14:51:24 +00:00
|
|
|
styleSheetNames.remove("sealed-graph");
|
2021-05-27 15:01:03 +00:00
|
|
|
|
|
|
|
boolean ok = check(htmlStyleNames, "HtmlStyle", styleSheetNames, "stylesheet")
|
|
|
|
& check(styleSheetNames, "stylesheet", htmlStyleNames, "HtmlStyle");
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
throw new Exception("differences found");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (errors > 0) {
|
|
|
|
throw new Exception(errors + " errors found");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean check(Set<String> s1, String l1, Set<String> s2, String l2) {
|
|
|
|
boolean equal = true;
|
|
|
|
for (String s : s1) {
|
|
|
|
if (!s2.contains(s)) {
|
|
|
|
System.err.println("In " + l1 + " but not " + l2 + ": " + s);
|
|
|
|
equal = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return equal;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove all the names from the set, giving a message for any that were not found.
|
|
|
|
*/
|
|
|
|
void removeAll(Set<String> set, String... names) {
|
|
|
|
for (String name : names) {
|
|
|
|
if (!set.remove(name)) {
|
|
|
|
error("name not found in set: " + name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void error(String message) {
|
|
|
|
System.err.println("error: " + message);
|
|
|
|
errors++;
|
|
|
|
}
|
|
|
|
|
|
|
|
Set<String> getHtmlStyleNames() {
|
2024-08-02 17:53:23 +00:00
|
|
|
return Arrays.stream(HtmlStyles.values())
|
2021-05-27 15:01:03 +00:00
|
|
|
.map(HtmlStyle::cssName)
|
|
|
|
.collect(Collectors.toCollection(TreeSet::new));
|
|
|
|
}
|
|
|
|
|
|
|
|
Set<String> getStylesheetNames() throws IOException {
|
|
|
|
Set<String> names = new TreeSet<>();
|
2023-08-09 07:01:15 +00:00
|
|
|
String stylesheet = "/jdk/javadoc/internal/doclets/formats/html/resources/stylesheet.css";
|
2024-08-02 17:53:23 +00:00
|
|
|
URL url = HtmlStyles.class.getResource(stylesheet);
|
2021-05-27 15:01:03 +00:00
|
|
|
readStylesheet(url, names);
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void readStylesheet(URL resource, Set<String> names) throws IOException {
|
|
|
|
try (InputStream in = resource.openStream()) {
|
|
|
|
if (in == null) {
|
|
|
|
throw new AssertionError("Cannot find or access resource " + resource);
|
|
|
|
}
|
2024-02-12 21:10:19 +00:00
|
|
|
String s = new String(in.readAllBytes())
|
|
|
|
.replaceAll("(?s)/\\*.*?\\*/", ""); // remove comments
|
2023-04-03 10:37:19 +00:00
|
|
|
Pattern p = Pattern.compile("(?i)\\.(?<name>[a-z][a-z0-9-]+)\\b");
|
2021-05-27 15:01:03 +00:00
|
|
|
Matcher m = p.matcher(s);
|
|
|
|
while (m.find()) {
|
|
|
|
names.add(m.group("name"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-09 07:01:15 +00:00
|
|
|
}
|