Skip to content

Commit b52611b

Browse files
committedNov 30, 2022
8295401: Error recovery in module-info.java could be improved
Reviewed-by: vromero
1 parent 9e80cf9 commit b52611b

File tree

2 files changed

+81
-6
lines changed

2 files changed

+81
-6
lines changed
 

‎src/jdk.compiler/share/classes/com/sun/tools/javac/parser/JavacParser.java

+5-4
Original file line numberDiff line numberDiff line change
@@ -3846,15 +3846,16 @@ List<JCDirective> moduleDirectiveList() {
38463846
} else if (token.name() == names.provides) {
38473847
nextToken();
38483848
JCExpression serviceName = qualident(false);
3849+
List<JCExpression> implNames;
38493850
if (token.kind == IDENTIFIER && token.name() == names.with) {
38503851
nextToken();
3851-
List<JCExpression> implNames = qualidentList(false);
3852-
accept(SEMI);
3853-
defs.append(toP(F.at(pos).Provides(serviceName, implNames)));
3852+
implNames = qualidentList(false);
38543853
} else {
38553854
log.error(DiagnosticFlag.SYNTAX, token.pos, Errors.ExpectedStr("'" + names.with + "'"));
3856-
skip(false, false, false, false);
3855+
implNames = List.nil();
38573856
}
3857+
accept(SEMI);
3858+
defs.append(toP(F.at(pos).Provides(serviceName, implNames)));
38583859
} else if (token.name() == names.uses) {
38593860
nextToken();
38603861
JCExpression service = qualident(false);

‎test/langtools/tools/javac/parser/JavacParserTest.java

+76-2
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
/*
2525
* @test
26-
* @bug 7073631 7159445 7156633 8028235 8065753 8205418 8205913 8228451 8237041 8253584 8246774 8256411 8256149 8259050 8266436 8267221 8271928 8275097 8293897
26+
* @bug 7073631 7159445 7156633 8028235 8065753 8205418 8205913 8228451 8237041 8253584 8246774 8256411 8256149 8259050 8266436 8267221 8271928 8275097 8293897 8295401
2727
* @summary tests error and diagnostics positions
2828
* @author Jan Lahoda
2929
* @modules jdk.compiler/com.sun.tools.javac.api
@@ -84,7 +84,9 @@
8484

8585
import com.sun.source.tree.CaseTree;
8686
import com.sun.source.tree.DefaultCaseLabelTree;
87+
import com.sun.source.tree.ModuleTree;
8788
import com.sun.source.util.TreePathScanner;
89+
import com.sun.tools.javac.api.JavacTaskPool;
8890
import java.util.Objects;
8991

9092
public class JavacParserTest extends TestCase {
@@ -106,7 +108,11 @@ class MyFileObject extends SimpleJavaFileObject {
106108
private String text;
107109

108110
public MyFileObject(String text) {
109-
super(URI.create("myfo:/Test.java"), JavaFileObject.Kind.SOURCE);
111+
this("Test", text);
112+
}
113+
114+
public MyFileObject(String fileName, String text) {
115+
super(URI.create("myfo:/" + fileName + ".java"), JavaFileObject.Kind.SOURCE);
110116
this.text = text;
111117
}
112118

@@ -1989,6 +1995,74 @@ public Void visitModifiers(ModifiersTree node, Void p) {
19891995
}.scan(cut, null);
19901996
}
19911997

1998+
@Test //JDK-8295401
1999+
void testModuleInfoProvidesRecovery() throws IOException {
2000+
String code = """
2001+
module m {
2002+
$DIRECTIVE
2003+
}
2004+
""";
2005+
record Test(String directive, int prefix, Kind expectedKind) {}
2006+
Test[] tests = new Test[] {
2007+
new Test("uses api.api.API;", 4, Kind.USES),
2008+
new Test("opens api.api to other.module;", 5, Kind.OPENS),
2009+
new Test("exports api.api to other.module;", 7, Kind.EXPORTS),
2010+
new Test("provides java.util.spi.ToolProvider with impl.ToolProvider;", 8, Kind.PROVIDES),
2011+
};
2012+
JavacTaskPool pool = new JavacTaskPool(1);
2013+
for (Test test : tests) {
2014+
String directive = test.directive();
2015+
for (int i = test.prefix(); i < directive.length(); i++) {
2016+
String replaced = code.replace("$DIRECTIVE", directive.substring(0, i));
2017+
pool.getTask(null, null, d -> {}, List.of(), null, List.of(new MyFileObject(replaced)), task -> {
2018+
try {
2019+
CompilationUnitTree cut = task.parse().iterator().next();
2020+
new TreePathScanner<Void, Void>() {
2021+
@Override
2022+
public Void visitModule(ModuleTree node, Void p) {
2023+
assertEquals("Unexpected directives size: " + node.getDirectives().size(),
2024+
node.getDirectives().size(),
2025+
1);
2026+
assertEquals("Unexpected directive: " + node.getDirectives().get(0).getKind(),
2027+
node.getDirectives().get(0).getKind(),
2028+
test.expectedKind);
2029+
return super.visitModule(node, p);
2030+
}
2031+
}.scan(cut, null);
2032+
return null;
2033+
} catch (IOException ex) {
2034+
throw new IllegalStateException(ex);
2035+
}
2036+
});
2037+
}
2038+
}
2039+
String extendedCode = """
2040+
module m {
2041+
provides ;
2042+
provides java.;
2043+
provides java.util.spi.ToolProvider with ;
2044+
provides java.util.spi.ToolProvider with impl.;
2045+
""";
2046+
pool.getTask(null, null, d -> {}, List.of(), null, List.of(new MyFileObject("module-info", extendedCode)), task -> {
2047+
try {
2048+
CompilationUnitTree cut = task.parse().iterator().next();
2049+
task.analyze();
2050+
new TreePathScanner<Void, Void>() {
2051+
@Override
2052+
public Void visitModule(ModuleTree node, Void p) {
2053+
assertEquals("Unexpected directives size: " + node.getDirectives().size(),
2054+
node.getDirectives().size(),
2055+
4);
2056+
return super.visitModule(node, p);
2057+
}
2058+
}.scan(cut, null);
2059+
return null;
2060+
} catch (IOException ex) {
2061+
throw new IllegalStateException(ex);
2062+
}
2063+
});
2064+
}
2065+
19922066
void run(String[] args) throws Exception {
19932067
int passed = 0, failed = 0;
19942068
final Pattern p = (args != null && args.length > 0)

0 commit comments

Comments
 (0)
Please sign in to comment.