/* * Copyright (c) 2019, 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 8226585 8250768 * @summary Verify behavior w.r.t. preview feature API errors and warnings * @library /tools/lib /tools/javac/lib * @modules * java.base/jdk.internal.javac * jdk.compiler/com.sun.tools.javac.api * jdk.compiler/com.sun.tools.javac.file * jdk.compiler/com.sun.tools.javac.main * jdk.compiler/com.sun.tools.javac.util * jdk.jdeps/com.sun.tools.classfile * @build toolbox.ToolBox toolbox.JavacTask * @build combo.ComboTestHelper * @run main PreviewErrors */ import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import combo.ComboInstance; import combo.ComboParameter; import combo.ComboTask; import combo.ComboTestHelper; import java.util.Arrays; import java.util.Map; import java.util.Set; import java.util.TreeMap; import java.util.stream.Collectors; import javax.tools.Diagnostic; import com.sun.tools.classfile.ClassFile; import com.sun.tools.classfile.ConstantPoolException; import java.io.FileWriter; import java.io.UncheckedIOException; import java.io.Writer; import java.util.Map.Entry; import javax.tools.JavaFileObject; import toolbox.JavacTask; import toolbox.ToolBox; public class PreviewErrors extends ComboInstance { protected ToolBox tb; PreviewErrors() { super(); tb = new ToolBox(); } private static String previewAPI(PreviewElementType elementType) { return """ package preview.api; public class ${name} { @jdk.internal.javac.PreviewFeature(feature=jdk.internal.javac.PreviewFeature.Feature.${preview} ${reflective}) public static void test() { } @jdk.internal.javac.PreviewFeature(feature=jdk.internal.javac.PreviewFeature.Feature.${preview} ${reflective}) public static class Clazz {} } """.replace("${name}", elementType.className) .replace("${preview}", "TEST") .replace("${reflective}", elementType.reflective); } private static final String SEALED_DECLARATION = """ package user; public sealed interface R {} final class C implements R {} """; static Map>>> parts = new TreeMap<>(); public static void main(String... args) throws Exception { new ComboTestHelper() .withDimension("PREVIEW", (x, preview) -> x.preview = preview, Preview.values()) .withDimension("LINT", (x, lint) -> x.lint = lint, Lint.values()) .withDimension("SUPPRESS", (x, suppress) -> x.suppress = suppress, Suppress.values()) .withDimension("ELEMENT_TYPE", (x, elementType) -> x.elementType = elementType, PreviewElementType.values()) .run(PreviewErrors::new); if (args.length == 1) { try (Writer out = new FileWriter(args[0])) { int petCount = 0; for (Entry>>> pet : parts.entrySet()) { petCount++; switch (pet.getKey()) { case API_REFLECTIVE_CLASS, API_CLASS -> { if (pet.getKey() == PreviewElementType.API_REFLECTIVE_CLASS) { out.write("

" + petCount + ". Reflective Preview API

\n"); } else { out.write("

" + petCount + ". Preview API

\n"); } out.write("API source (part of java.base):\n"); out.write("
\n");
                            String previewAPI = previewAPI(pet.getKey());
                            out.write(previewAPI);
                            out.write("\n
\n"); } case REFER_TO_DECLARATION_CLASS -> { out.write("

" + petCount + ". Using an element declared using a preview feature

\n"); out.write("Element declaration:\n"); out.write("
\n");
                            out.write(SEALED_DECLARATION);
                            out.write("\n
\n"); } case LANGUAGE -> { out.write("

" + petCount + ". Using preview language feature

\n"); } } int prevCount = 0; for (Entry>> prev : pet.getValue().entrySet()) { prevCount++; switch (prev.getKey()) { case YES -> { out.write("

" + petCount + "." + prevCount + ". With --enable-preview

\n"); } case NO -> { out.write("

" + petCount + "." + prevCount + ". Without --enable-preview

\n"); } } int supCount = 0; for (Entry> sup : prev.getValue().entrySet()) { supCount++; switch (sup.getKey()) { case YES -> { out.write("

" + petCount + "." + prevCount + "." + supCount + ". Usages suppressed with @SuppressWarnings

\n"); } case NO -> { out.write("

" + petCount + "." + prevCount + "." + supCount + ". Usages not suppressed with @SuppressWarnings

\n"); } } int lintCount = 0; for (Entry lint : sup.getValue().entrySet()) { lintCount++; switch (lint.getKey()) { case NONE -> { out.write("
" + petCount + "." + prevCount + "." + supCount + "." + lintCount + ". Neither -Xlint:preview nor -Xlint:-preview
\n"); } case ENABLE_PREVIEW -> { out.write("
" + petCount + "." + prevCount + "." + supCount + "." + lintCount + ". With -Xlint:preview
\n"); } case DISABLE_PREVIEW -> { out.write("
" + petCount + "." + prevCount + "." + supCount + "." + lintCount + ". With -Xlint:-preview
\n"); } } out.write(lint.getValue().toString()); out.write("\n"); } } } } } } } private Preview preview; private Lint lint; private Suppress suppress; private PreviewElementType elementType; @Override public void doWork() throws IOException { Path base = Paths.get("."); tb.cleanDirectory(base); Path src = base.resolve("src"); Path srcJavaBase = src.resolve("java.base"); Path classes = base.resolve("classes"); Path classesJavaBase = classes.resolve("java.base"); Files.createDirectories(classesJavaBase); String previewAPI = previewAPI(elementType); ComboTask task = newCompilationTask() .withOption("-XDrawDiagnostics") .withOption("-source") .withOption(String.valueOf(Runtime.version().feature())); switch (elementType) { case API_CLASS, API_REFLECTIVE_CLASS -> { tb.writeJavaFiles(srcJavaBase, previewAPI); new JavacTask(tb) .outdir(classesJavaBase) .options("--patch-module", "java.base=" + srcJavaBase.toString()) .files(tb.findJavaFiles(srcJavaBase)) .run() .writeAll(); task.withOption("--patch-module") .withOption("java.base=" + classesJavaBase.toString()) .withOption("--add-exports") .withOption("java.base/preview.api=ALL-UNNAMED"); } case API_SOURCE, API_REFLECTIVE_SOURCE -> { tb.writeJavaFiles(srcJavaBase, previewAPI); task.withOption("--patch-module") .withOption("java.base=" + srcJavaBase.toString()) .withOption("--add-exports") .withOption("java.base/preview.api=ALL-UNNAMED"); } case LANGUAGE -> { task.withOption("-XDforcePreview=true"); } case REFER_TO_DECLARATION_CLASS -> { tb.writeJavaFiles(srcJavaBase, SEALED_DECLARATION); new JavacTask(tb) .outdir(classesJavaBase) .options("--patch-module", "java.base=" + srcJavaBase.toString(), "--enable-preview", "-source", String.valueOf(Runtime.version().feature())) .files(tb.findJavaFiles(srcJavaBase)) .run() .writeAll(); task.withOption("--patch-module") .withOption("java.base=" + classesJavaBase.toString()) .withOption("--add-exports") .withOption("java.base/user=ALL-UNNAMED"); } case REFER_TO_DECLARATION_SOURCE -> { tb.writeJavaFiles(srcJavaBase, SEALED_DECLARATION); task.withOption("--patch-module") .withOption("java.base=" + srcJavaBase.toString()) .withOption("--add-exports") .withOption("java.base/user=ALL-UNNAMED"); } } task.withSourceFromTemplate(""" package test; public class Test { #{SUPPRESS} public void test(Object o) { #{ELEMENT_TYPE} } } """); if (preview.expand(null)!= null) { task.withOption(preview.expand(null)); } if (lint.expand(null) != null) { task.withOption(lint.expand(null)); } task.generate(result -> { Set actual = Arrays.stream(Diagnostic.Kind.values()) .flatMap(kind -> result.diagnosticsForKind(kind).stream()) .map(d -> d.getLineNumber() + ":" + d.getColumnNumber() + ":" + d.getCode()) .collect(Collectors.toSet()); boolean ok; boolean previewClass = true; Set expected = null; if (preview == Preview.NO) { switch (elementType) { case LANGUAGE -> { ok = false; expected = Set.of("5:41:compiler.err.preview.feature.disabled"); } case REFER_TO_DECLARATION_CLASS -> { ok = false; expected = Set.of("-1:-1:compiler.err.preview.feature.disabled.classfile"); } case REFER_TO_DECLARATION_SOURCE -> { ok = false; expected = Set.of("2:8:compiler.err.preview.feature.disabled.plural"); } case API_CLASS, API_SOURCE -> { ok = false; expected = Set.of("6:17:compiler.err.is.preview", "5:25:compiler.err.is.preview"); } case API_REFLECTIVE_CLASS, API_REFLECTIVE_SOURCE -> { ok = true; previewClass = false; if (suppress == Suppress.YES) { expected = Set.of(); } else if (lint == Lint.NONE || lint == Lint.ENABLE_PREVIEW) { expected = Set.of("6:20:compiler.warn.is.preview.reflective", "5:28:compiler.warn.is.preview.reflective"); } else {//-Xlint:-preview expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } } default -> { throw new IllegalStateException(elementType.name()); } } } else { ok = true; switch (elementType) { case LANGUAGE -> { if (lint == Lint.ENABLE_PREVIEW) { expected = Set.of("5:41:compiler.warn.preview.feature.use"); } else { expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } } case REFER_TO_DECLARATION_CLASS -> { if (suppress == Suppress.YES) { if (lint == Lint.ENABLE_PREVIEW) { expected = Set.of("-1:-1:compiler.warn.preview.feature.use.classfile"); } else { expected = Set.of(/*"-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"*/); } } else if (lint == Lint.ENABLE_PREVIEW) { expected = Set.of("5:13:compiler.warn.declared.using.preview", "5:24:compiler.warn.declared.using.preview", "-1:-1:compiler.warn.preview.feature.use.classfile"); } else { expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } } case REFER_TO_DECLARATION_SOURCE -> { if (lint == Lint.ENABLE_PREVIEW) { if (suppress == Suppress.YES) { expected = Set.of("2:8:compiler.warn.preview.feature.use.plural", "3:26:compiler.warn.declared.using.preview"); } else { expected = Set.of("5:13:compiler.warn.declared.using.preview", "5:24:compiler.warn.declared.using.preview", "2:8:compiler.warn.preview.feature.use.plural", "3:26:compiler.warn.declared.using.preview"); } } else { if (suppress == Suppress.YES) { expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } else { expected = Set.of("-1:-1:compiler.note.preview.plural", "-1:-1:compiler.note.preview.recompile"); } } } case API_CLASS, API_SOURCE -> { if (suppress == Suppress.YES) { expected = Set.of(); } else if (lint == Lint.ENABLE_PREVIEW) { expected = Set.of("6:17:compiler.warn.is.preview", "5:25:compiler.warn.is.preview"); } else { expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } } case API_REFLECTIVE_CLASS, API_REFLECTIVE_SOURCE -> { previewClass = false; if (suppress == Suppress.YES) { expected = Set.of(); } else if (lint == Lint.ENABLE_PREVIEW) { expected = Set.of("6:20:compiler.warn.is.preview.reflective", "5:28:compiler.warn.is.preview.reflective"); } else { expected = Set.of("-1:-1:compiler.note.preview.filename", "-1:-1:compiler.note.preview.recompile"); } } } } if (!elementType.isSource) { try { parts.computeIfAbsent(elementType, x -> new TreeMap<>()) .computeIfAbsent(preview, x -> new TreeMap<>()) .computeIfAbsent(suppress, x -> new TreeMap<>()) .computeIfAbsent(lint, x -> new StringBuilder()) .append("
\n")
                                .append(task.getSources().head.getCharContent(false))
                                .append("\n
\n") .append("
\n")
                                .append(Arrays.stream(Diagnostic.Kind.values())
                                              .flatMap(kind -> result.diagnosticsForKind(kind).reverse().stream())
                                              .filter(d -> d.getSource() == null || !d.getSource().getName().contains("R.java"))
                                              .map(d -> (d.getSource() != null ? d.getSource().getName() + ":" + d.getLineNumber() + ":" : "") + d.getKind()+ ": " + d.getMessage(null)).collect(Collectors.joining("\n")))
                                .append("\n
\n") .append(ok ? previewClass ? "Test.class is marked as a preview class file." : "Test.class is NOT marked as a preview class file." : "Does not compile."); } catch (IOException ex) { throw new UncheckedIOException(ex); } } if (ok) { if (!result.get().iterator().hasNext()) { throw new IllegalStateException("Did not succeed as expected for preview=" + preview + ", lint=" + lint + ", suppress=" + suppress + ", elementType=" + elementType + ": actual:\"" + actual + "\""); } ClassFile cf; try { JavaFileObject testClass = null; for (JavaFileObject classfile : result.get()) { if (classfile.isNameCompatible("Test", JavaFileObject.Kind.CLASS)){ testClass = classfile; } } if (testClass == null) { throw new IllegalStateException("Cannot find Test.class"); } cf = ClassFile.read(testClass.openInputStream()); } catch (IOException | ConstantPoolException ex) { throw new IllegalStateException(ex); } if (previewClass && cf.minor_version != 65535) { throw new IllegalStateException("Expected preview class, but got: " + cf.minor_version); } else if (!previewClass && cf.minor_version != 0) { throw new IllegalStateException("Expected minor version == 0 but got: " + cf.minor_version); } } else { if (result.get().iterator().hasNext()) { throw new IllegalStateException("Succeed unexpectedly for preview=" + preview + ", lint=" + lint + ", suppress=" + suppress + ", elementType=" + elementType); } } if (expected != null && !expected.equals(actual)) { throw new IllegalStateException("Unexpected output for preview=" + preview + ", lint=" + lint + ", suppress=" + suppress + ", elementType=" + elementType + ": actual: \"" + actual + "\", expected: \"" + expected + "\""); } }); } public enum Preview implements ComboParameter { YES("--enable-preview"), NO(null); private final String opt; private Preview(String opt) { this.opt = opt; } public String expand(String optParameter) { return opt; } } public enum Lint implements ComboParameter { NONE(null), ENABLE_PREVIEW("-Xlint:preview"), DISABLE_PREVIEW("-Xlint:-preview"); private final String opt; private Lint(String opt) { this.opt = opt; } public String expand(String optParameter) { return opt; } } public enum Suppress implements ComboParameter { YES("@SuppressWarnings(\"preview\")"), NO(""); private final String code; private Suppress(String code) { this.code = code; } public String expand(String optParameter) { return code; } } public enum PreviewElementType implements ComboParameter { LANGUAGE("boolean b = o instanceof String s;", ", reflective=false", "", false), REFER_TO_DECLARATION_SOURCE("user.R d1; user.R d2;", ", reflective=false", "", true), REFER_TO_DECLARATION_CLASS("user.R d1; user.R d2;", ", reflective=false", "", false), API_CLASS(""" preview.api.Core.test(); preview.api.Core.Clazz c; """, ", reflective=false", "Core", false), API_REFLECTIVE_CLASS(""" preview.api.Reflect.test(); preview.api.Reflect.Clazz c; """, ", reflective=true", "Reflect", false), API_SOURCE(""" preview.api.Core.test(); preview.api.Core.Clazz c; """, ", reflective=false", "Core", true), API_REFLECTIVE_SOURCE(""" preview.api.Reflect.test(); preview.api.Reflect.Clazz c; """, ", reflective=true", "Reflect", true); String code; String reflective; String className; boolean isSource; private PreviewElementType(String code, String reflective, String className, boolean isSource) { this.code = code; this.reflective = reflective; this.className = className; this.isSource = isSource; } public String expand(String optParameter) { return code; } } }