Skip to content

Commit 765ad0e

Browse files
committedJun 5, 2024
8331947: Preview creates checkbox for JEP-less preview feature
Reviewed-by: liach, prappo
1 parent 8e903ee commit 765ad0e

File tree

8 files changed

+86
-51
lines changed

8 files changed

+86
-51
lines changed
 

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

+8-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2024, 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
@@ -31,9 +31,9 @@
3131
import java.util.Map;
3232
import java.util.SortedSet;
3333
import java.util.TreeSet;
34+
import java.util.stream.Collectors;
3435

3536
import javax.lang.model.element.AnnotationMirror;
36-
import javax.lang.model.element.AnnotationValue;
3737
import javax.lang.model.element.Element;
3838
import javax.lang.model.element.ExecutableElement;
3939
import javax.lang.model.element.ModuleElement;
@@ -48,15 +48,12 @@
4848

4949
import com.sun.source.util.TreePath;
5050
import com.sun.tools.javac.code.Flags;
51-
import com.sun.tools.javac.code.Scope;
5251
import com.sun.tools.javac.code.Symbol;
5352
import com.sun.tools.javac.code.Symbol.ClassSymbol;
5453
import com.sun.tools.javac.code.Symbol.MethodSymbol;
5554
import com.sun.tools.javac.code.Symbol.ModuleSymbol;
5655
import com.sun.tools.javac.code.Symbol.PackageSymbol;
5756
import com.sun.tools.javac.code.Symbol.VarSymbol;
58-
import com.sun.tools.javac.comp.AttrContext;
59-
import com.sun.tools.javac.comp.Env;
6057
import com.sun.tools.javac.util.Names;
6158
import com.sun.tools.javac.util.Options;
6259

@@ -408,15 +405,19 @@ public boolean accessInternalAPI() {
408405
* @param feature the name of the PreviewFeature.Feature enum value
409406
* @return the map of PreviewFeature.JEP annotation element values, or an empty map
410407
*/
411-
public Map<? extends ExecutableElement, ? extends AnnotationValue> getJepInfo(String feature) {
408+
public Map<String, Object> getJepInfo(String feature) {
412409
TypeElement featureType = elementUtils.getTypeElement("jdk.internal.javac.PreviewFeature.Feature");
413410
TypeElement jepType = elementUtils.getTypeElement("jdk.internal.javac.PreviewFeature.JEP");
414411
var featureVar = featureType.getEnclosedElements().stream()
415412
.filter(e -> feature.equals(e.getSimpleName().toString())).findFirst();
416413
if (featureVar.isPresent()) {
417414
for (AnnotationMirror anno : featureVar.get().getAnnotationMirrors()) {
418415
if (anno.getAnnotationType().asElement().equals(jepType)) {
419-
return anno.getElementValues();
416+
return anno.getElementValues().entrySet()
417+
.stream()
418+
.collect(Collectors.toMap(
419+
e -> e.getKey().getSimpleName().toString(),
420+
e -> e.getValue().getValue()));
420421
}
421422
}
422423
}

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

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2024, 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
@@ -47,7 +47,7 @@ public class DeprecatedAPIListBuilder extends SummaryAPIListBuilder {
4747
* @param since list of releases passed via <code>--since</code> option
4848
*/
4949
public DeprecatedAPIListBuilder(BaseConfiguration configuration, List<String> since) {
50-
super(configuration, configuration.utils::isDeprecated);
50+
super(configuration);
5151
this.foundReleases = new HashSet<>();
5252
buildSummaryAPIInfo();
5353
// The releases list is set to the intersection of releases defined via `--since` option
@@ -73,6 +73,11 @@ public SortedSet<Element> getForRemoval() {
7373
return forRemoval;
7474
}
7575

76+
@Override
77+
protected boolean belongsToSummary(Element element) {
78+
return utils.isDeprecated(element);
79+
}
80+
7681
@Override
7782
protected void handleElement(Element e) {
7883
foundReleases.add(utils.getDeprecatedSince(e));

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

+5-4
Original file line numberDiff line numberDiff line change
@@ -46,16 +46,17 @@ public class NewAPIBuilder extends SummaryAPIListBuilder {
4646
public final List<String> releases;
4747

4848
public NewAPIBuilder(BaseConfiguration configuration, List<String> releases) {
49-
super(configuration, element -> isNewAPI(element, configuration.utils, releases));
49+
super(configuration);
5050
this.releases = releases;
5151
buildSummaryAPIInfo();
5252
}
5353

54-
private static boolean isNewAPI(Element e, Utils utils, List<String> releases) {
55-
if (!utils.hasDocCommentTree(e)) {
54+
@Override
55+
protected boolean belongsToSummary(Element element) {
56+
if (!utils.hasDocCommentTree(element)) {
5657
return false;
5758
}
58-
var sinceTrees = utils.getBlockTags(e, SINCE, SinceTree.class);
59+
var sinceTrees = utils.getBlockTags(element, SINCE, SinceTree.class);
5960
if (sinceTrees.isEmpty()) {
6061
return false;
6162
}

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

+34-23
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2024, 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
@@ -27,14 +27,13 @@
2727

2828
import jdk.javadoc.internal.doclets.toolkit.BaseConfiguration;
2929

30-
import javax.lang.model.element.AnnotationValue;
3130
import javax.lang.model.element.Element;
32-
import javax.lang.model.element.ExecutableElement;
3331
import java.util.HashMap;
3432
import java.util.Map;
3533
import java.util.Objects;
3634
import java.util.Set;
3735
import java.util.TreeSet;
36+
import java.util.stream.Collectors;
3837

3938
/**
4039
* Build list of all the preview packages, classes, constructors, fields and methods.
@@ -43,6 +42,7 @@ public class PreviewAPIListBuilder extends SummaryAPIListBuilder {
4342

4443
private final Map<Element, JEP> elementJeps = new HashMap<>();
4544
private final Map<String, JEP> jeps = new HashMap<>();
45+
private static final JEP NULL_SENTINEL = new JEP(0, "", "");
4646

4747
/**
4848
* The JEP for a preview feature in this release.
@@ -60,40 +60,51 @@ public int compareTo(JEP o) {
6060
* @param configuration the current configuration of the doclet
6161
*/
6262
public PreviewAPIListBuilder(BaseConfiguration configuration) {
63-
super(configuration, configuration.utils::isPreviewAPI);
63+
super(configuration);
6464
buildSummaryAPIInfo();
6565
}
6666

6767
@Override
68-
protected void handleElement(Element e) {
69-
String feature = Objects.requireNonNull(utils.getPreviewFeature(e),
68+
protected boolean belongsToSummary(Element element) {
69+
if (!utils.isPreviewAPI(element)) {
70+
return false;
71+
}
72+
String feature = Objects.requireNonNull(utils.getPreviewFeature(element),
7073
"Preview feature not specified").toString();
71-
JEP jep = jeps.computeIfAbsent(feature, (featureName) -> {
72-
Map<? extends ExecutableElement, ? extends AnnotationValue> anno = configuration.workArounds.getJepInfo(featureName);
73-
int number = 0;
74-
String title = "";
75-
String status = "Preview"; // Default value is not returned by the method we use above.
76-
for (var entry : anno.entrySet()) {
77-
if ("number".equals(entry.getKey().getSimpleName().toString())) {
78-
number = (int) entry.getValue().getValue();
79-
} else if ("title".equals(entry.getKey().getSimpleName().toString())) {
80-
title = (String) entry.getValue().getValue();
81-
} else if ("status".equals(entry.getKey().getSimpleName().toString())) {
82-
status = (String) entry.getValue().getValue();
83-
} else {
84-
throw new IllegalArgumentException(entry.getKey().getSimpleName().toString());
74+
JEP jep = jeps.computeIfAbsent(feature, featureName -> {
75+
Map<String, Object> jepInfo = configuration.workArounds.getJepInfo(featureName);
76+
if (!jepInfo.isEmpty()) {
77+
int number = 0;
78+
String title = "";
79+
String status = "Preview"; // Default value is not returned by the method we used above.
80+
for (var entry : jepInfo.entrySet()) {
81+
switch (entry.getKey()) {
82+
case "number" -> number = (int) entry.getValue();
83+
case "title" -> title = (String) entry.getValue();
84+
case "status" -> status = (String) entry.getValue();
85+
default -> throw new IllegalArgumentException(entry.getKey());
86+
}
8587
}
88+
return new JEP(number, title, status);
8689
}
87-
return new JEP(number, title, status);
90+
return NULL_SENTINEL;
8891
});
89-
elementJeps.put(e, jep);
92+
if (jep != NULL_SENTINEL) {
93+
elementJeps.put(element, jep);
94+
return true;
95+
}
96+
// Preview features without JEP are not included.
97+
return false;
9098
}
9199

92100
/**
93101
* {@return a sorted set of preview feature JEPs in this release}
94102
*/
95103
public Set<JEP> getJEPs() {
96-
return new TreeSet<>(jeps.values());
104+
return jeps.values()
105+
.stream()
106+
.filter(jep -> jep != NULL_SENTINEL)
107+
.collect(Collectors.toCollection(TreeSet::new));
97108
}
98109

99110
/**

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

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2023, 2024, 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
@@ -41,7 +41,12 @@ public class RestrictedAPIListBuilder extends SummaryAPIListBuilder {
4141
* @param configuration the current configuration of the doclet
4242
*/
4343
public RestrictedAPIListBuilder(BaseConfiguration configuration) {
44-
super(configuration, configuration.utils::isRestrictedAPI);
44+
super(configuration);
4545
buildSummaryAPIInfo();
4646
}
47+
48+
@Override
49+
protected boolean belongsToSummary(Element element) {
50+
return utils.isRestrictedAPI(element);
51+
}
4752
}

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

+16-12
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1998, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1998, 2024, 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
@@ -26,7 +26,6 @@
2626
package jdk.javadoc.internal.doclets.toolkit.util;
2727

2828
import java.util.*;
29-
import java.util.function.Predicate;
3029

3130
import javax.lang.model.element.Element;
3231
import javax.lang.model.element.ModuleElement;
@@ -39,14 +38,13 @@
3938
/**
4039
* Build list of all the summary packages, classes, constructors, fields and methods.
4140
*/
42-
public class SummaryAPIListBuilder {
41+
public abstract class SummaryAPIListBuilder {
4342
/**
4443
* List of summary type Lists.
4544
*/
4645
private final Map<SummaryElementKind, SortedSet<Element>> summaryMap;
4746
protected final BaseConfiguration configuration;
4847
protected final Utils utils;
49-
private final Predicate<Element> belongsToSummary;
5048

5149
public enum SummaryElementKind {
5250
MODULE,
@@ -69,11 +67,9 @@ public enum SummaryElementKind {
6967
*
7068
* @param configuration the current configuration of the doclet
7169
*/
72-
public SummaryAPIListBuilder(BaseConfiguration configuration,
73-
Predicate<Element> belongsToSummary) {
70+
protected SummaryAPIListBuilder(BaseConfiguration configuration) {
7471
this.configuration = configuration;
7572
this.utils = configuration.utils;
76-
this.belongsToSummary = belongsToSummary;
7773
summaryMap = new EnumMap<>(SummaryElementKind.class);
7874
for (SummaryElementKind kind : SummaryElementKind.values()) {
7975
summaryMap.put(kind, createSummarySet());
@@ -93,22 +89,22 @@ protected void buildSummaryAPIInfo() {
9389
SortedSet<ModuleElement> modules = configuration.modules;
9490
SortedSet<Element> mset = summaryMap.get(SummaryElementKind.MODULE);
9591
for (Element me : modules) {
96-
if (belongsToSummary.test(me)) {
92+
if (belongsToSummary(me)) {
9793
mset.add(me);
9894
handleElement(me);
9995
}
10096
}
10197
SortedSet<PackageElement> packages = configuration.packages;
10298
SortedSet<Element> pset = summaryMap.get(SummaryElementKind.PACKAGE);
10399
for (Element pe : packages) {
104-
if (belongsToSummary.test(pe)) {
100+
if (belongsToSummary(pe)) {
105101
pset.add(pe);
106102
handleElement(pe);
107103
}
108104
}
109105
for (TypeElement te : configuration.getIncludedTypeElements()) {
110106
SortedSet<Element> eset;
111-
if (belongsToSummary.test(te)) {
107+
if (belongsToSummary(te)) {
112108
switch (te.getKind()) {
113109
case ANNOTATION_TYPE -> {
114110
eset = summaryMap.get(SummaryElementKind.ANNOTATION_TYPE);
@@ -149,7 +145,7 @@ protected void buildSummaryAPIInfo() {
149145
}
150146
if (utils.isRecord(te)) {
151147
for (RecordComponentElement component : te.getRecordComponents()) {
152-
if (belongsToSummary.test(component)) {
148+
if (belongsToSummary(component)) {
153149
throw new AssertionError("record components not supported in summary builders: " +
154150
"component: " + component.getSimpleName() +
155151
" of record: " + te.getQualifiedName());
@@ -164,6 +160,14 @@ protected void buildSummaryAPIInfo() {
164160
}
165161
}
166162

163+
/**
164+
* This method decides whether Element {@code element} should be included in this summary list.
165+
*
166+
* @param element an element
167+
* @return true if the element should be included
168+
*/
169+
protected abstract boolean belongsToSummary(Element element);
170+
167171
/**
168172
* Add the members into a single list of summary members.
169173
*
@@ -172,7 +176,7 @@ protected void buildSummaryAPIInfo() {
172176
*/
173177
private void composeSummaryList(SortedSet<Element> sset, List<? extends Element> members) {
174178
for (Element member : members) {
175-
if (belongsToSummary.test(member)) {
179+
if (belongsToSummary(member)) {
176180
sset.add(member);
177181
handleElement(member);
178182
}

‎test/langtools/jdk/javadoc/doclet/testPreview/TestPreview.java

+4
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
/*
2525
* @test
2626
* @bug 8250768 8261976 8277300 8282452 8287597 8325325 8325874 8297879
27+
* 8331947
2728
* @summary test generated docs for items declared using preview
2829
* @library ../../lib
2930
* @modules jdk.javadoc/jdk.javadoc.internal.tool
@@ -156,6 +157,9 @@ public void testPreviewAPIJavadoc() {
156157
<li><a href="package-summary.html">preview</a></li>
157158
<li><a href="Core.html" class="current-selection">Core</a></li>
158159
</ol>""");
160+
161+
// 8331947: Support preview features without JEP should not be included in Preview API page
162+
checkOutput("preview-list.html", false, "supportMethod");
159163
}
160164

161165
@Test

‎test/langtools/jdk/javadoc/doclet/testPreview/api/preview/NoPreview.java

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, 2024, 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
@@ -34,4 +34,8 @@ public T get() {
3434

3535
@PreviewFeature(feature=Feature.TEST)
3636
public static class T {}
37+
38+
// Preview support feature without JEP should not be listed
39+
@PreviewFeature(feature=Feature.LANGUAGE_MODEL)
40+
public void supportMethod() {}
3741
}

0 commit comments

Comments
 (0)
Please sign in to comment.