/* * Copyright (c) 2018, 2022, 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 8206986 * @summary Ensure rule cases can be parsed correctly for complex expressions. * @modules jdk.compiler */ import java.io.StringWriter; import java.net.URI; import java.util.AbstractMap.SimpleEntry; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.Map.Entry; import javax.tools.*; import com.sun.source.tree.CaseTree; import com.sun.source.tree.CompilationUnitTree; import com.sun.source.util.JavacTask; import com.sun.source.util.TreePathScanner; import com.sun.source.util.Trees; public class RuleParsingTest { public static void main(String[] args) throws Exception { String sourceVersion = Integer.toString(Runtime.version().feature()); new RuleParsingTest().testParseComplexExpressions(sourceVersion); } void testParseComplexExpressions(String sourceVersion) throws Exception { String[] expressions = { "(a)", "a", "a + a", "~a + a", "a = a", "a += a", "a + (a)", "a + (a) b", "true ? a : b", "m(() -> {})", "m(() -> 1)", "m(a -> 1)", "m((t a) -> 1)", }; StringBuilder code = new StringBuilder(); List> spans = new ArrayList<>(); code.append("class Test {\n" + " void t(int i) {\n"); for (boolean switchExpr : new boolean[] {false, true}) { if (switchExpr) { code.append(" int j = switch(i) {\n"); } else { code.append(" switch(i) {\n"); } for (String expr : expressions) { code.append("case "); int start = code.length(); code.append(expr); spans.add(new SimpleEntry<>((long) start, (long) code.length())); code.append(" -> {}"); } code.append(" };\n"); } code.append(" }\n" + "}\n"); final JavaCompiler tool = ToolProvider.getSystemJavaCompiler(); assert tool != null; DiagnosticListener noErrors = d -> { throw new AssertionError(d.getMessage(null)); }; String version = System.getProperty("java.specification.version"); StringWriter out = new StringWriter(); JavacTask ct = (JavacTask) tool.getTask(out, null, noErrors, List.of(), null, Arrays.asList(new MyFileObject(code.toString()))); CompilationUnitTree cut = ct.parse().iterator().next(); Trees trees = Trees.instance(ct); new TreePathScanner() { @Override public Void visitCase(CaseTree node, Void p) { long start = trees.getSourcePositions().getStartPosition(cut, node.getExpression()); long end = trees.getSourcePositions().getEndPosition(cut, node.getExpression()); if (!spans.remove(new SimpleEntry<>(start, end))) { throw new AssertionError("Did not find an expression span in expected spans: " + start + "-" + end + " '" + node.getExpression().toString() + "'"); } return super.visitCase(node, p); } }.scan(cut, null); if (!spans.isEmpty()) { throw new AssertionError("Remaning spans: " + spans); } } static class MyFileObject extends SimpleJavaFileObject { private String text; public MyFileObject(String text) { super(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE); this.text = text; } @Override public CharSequence getCharContent(boolean ignoreEncodingErrors) { return text; } } }