author | coleenp |
Fri, 31 Aug 2018 07:03:46 -0400 | |
changeset 51608 | 625a5bdde0c5 |
parent 50735 | 2f2af62dfac7 |
permissions | -rw-r--r-- |
33362 | 1 |
/* |
50735
2f2af62dfac7
8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents:
47216
diff
changeset
|
2 |
* Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved. |
33362 | 3 |
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
4 |
* |
|
5 |
* This code is free software; you can redistribute it and/or modify it |
|
6 |
* under the terms of the GNU General Public License version 2 only, as |
|
7 |
* published by the Free Software Foundation. |
|
8 |
* |
|
9 |
* This code is distributed in the hope that it will be useful, but WITHOUT |
|
10 |
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
|
11 |
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
|
12 |
* version 2 for more details (a copy is included in the LICENSE file that |
|
13 |
* accompanied this code). |
|
14 |
* |
|
15 |
* You should have received a copy of the GNU General Public License version |
|
16 |
* 2 along with this work; if not, write to the Free Software Foundation, |
|
17 |
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
|
18 |
* |
|
19 |
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
|
20 |
* or visit www.oracle.com if you need additional information or have any |
|
21 |
* questions. |
|
22 |
*/ |
|
23 |
||
24 |
/* |
|
50735
2f2af62dfac7
8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents:
47216
diff
changeset
|
25 |
* @test 8173232 8010319 |
43366 | 26 |
* @summary Test of forward referencing of snippets. |
33362 | 27 |
* @build KullaTesting TestingInputStream |
43366 | 28 |
* @run testng ForwardReferenceTest |
33362 | 29 |
*/ |
30 |
||
36499 | 31 |
import java.util.List; |
33362 | 32 |
import jdk.jshell.Snippet; |
33 |
import jdk.jshell.MethodSnippet; |
|
34 |
import jdk.jshell.VarSnippet; |
|
35 |
import jdk.jshell.DeclarationSnippet; |
|
36 |
import org.testng.annotations.Test; |
|
37 |
||
36499 | 38 |
import jdk.jshell.SnippetEvent; |
39 |
import jdk.jshell.UnresolvedReferenceException; |
|
40 |
import static org.testng.Assert.assertEquals; |
|
33362 | 41 |
import static jdk.jshell.Snippet.Status.*; |
42 |
import static org.testng.Assert.assertTrue; |
|
43 |
||
44 |
@Test |
|
43366 | 45 |
public class ForwardReferenceTest extends KullaTesting { |
33362 | 46 |
|
47 |
public void testOverwriteMethodForwardReferenceClass() { |
|
48 |
Snippet k1 = methodKey(assertEval("int q(Boo b) { return b.x; }", |
|
49 |
added(RECOVERABLE_NOT_DEFINED))); |
|
50 |
assertUnresolvedDependencies1((MethodSnippet) k1, RECOVERABLE_NOT_DEFINED, "class Boo"); |
|
51 |
assertEval("class Boo { int x = 55; }", |
|
52 |
added(VALID), |
|
53 |
ste(k1, RECOVERABLE_NOT_DEFINED, VALID, true, null)); |
|
54 |
assertMethodDeclSnippet((MethodSnippet) k1, "q", "(Boo)int", VALID, 0, 0); |
|
55 |
assertEval("q(new Boo());", "55"); |
|
56 |
assertActiveKeys(); |
|
57 |
} |
|
58 |
||
59 |
public void testOverwriteMethodForwardReferenceClassImport() { |
|
60 |
MethodSnippet k1 = methodKey(assertEval("int ff(List lis) { return lis.size(); }", |
|
61 |
added(RECOVERABLE_NOT_DEFINED))); |
|
62 |
assertUnresolvedDependencies1(k1, RECOVERABLE_NOT_DEFINED, "class List"); |
|
63 |
assertEval("import java.util.*;", |
|
64 |
added(VALID), |
|
65 |
ste(k1, RECOVERABLE_NOT_DEFINED, VALID, true, null)); |
|
66 |
assertMethodDeclSnippet(k1, "ff", "(List)int", VALID, 0, 0); |
|
67 |
assertEval("ff(new ArrayList());", "0"); |
|
68 |
assertActiveKeys(); |
|
69 |
} |
|
70 |
||
71 |
public void testForwardVarToMethod() { |
|
72 |
DeclarationSnippet t = methodKey(assertEval("int t() { return x; }", added(RECOVERABLE_DEFINED))); |
|
73 |
assertUnresolvedDependencies1(t, RECOVERABLE_DEFINED, "variable x"); |
|
74 |
assertEvalUnresolvedException("t();", "t", 1, 0); |
|
75 |
Snippet x = varKey(assertEval("int x = 33;", "33", |
|
76 |
added(VALID), |
|
77 |
ste(t, RECOVERABLE_DEFINED, VALID, false, null))); |
|
78 |
assertEval("t();", "33"); |
|
79 |
assertEval("double x = 0.88;", |
|
80 |
"0.88", null, |
|
81 |
DiagCheck.DIAG_OK, |
|
82 |
DiagCheck.DIAG_ERROR, |
|
83 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
84 |
ste(x, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
85 |
ste(t, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
|
86 |
assertEvalUnresolvedException("t();", "t", 0, 1); |
|
87 |
assertActiveKeys(); |
|
88 |
} |
|
89 |
||
90 |
public void testForwardMethodToMethod() { |
|
91 |
Snippet t = methodKey(assertEval("int t() { return f(); }", added(RECOVERABLE_DEFINED))); |
|
92 |
Snippet f = methodKey(assertEval("int f() { return g(); }", |
|
93 |
added(RECOVERABLE_DEFINED), |
|
94 |
ste(t, RECOVERABLE_DEFINED, VALID, false, null))); |
|
95 |
assertUnresolvedDependencies1((DeclarationSnippet) f, RECOVERABLE_DEFINED, "method g()"); |
|
96 |
assertEvalUnresolvedException("t();", "f", 1, 0); |
|
97 |
Snippet g = methodKey(assertEval("int g() { return 55; }", |
|
98 |
added(VALID), |
|
99 |
ste(f, RECOVERABLE_DEFINED, VALID, false, null))); |
|
100 |
assertEval("t();", "55"); |
|
101 |
assertEval("double g() { return 3.14159; }", |
|
102 |
DiagCheck.DIAG_OK, |
|
103 |
DiagCheck.DIAG_ERROR, |
|
104 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
105 |
ste(g, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
106 |
ste(f, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
|
36499 | 107 |
DeclarationSnippet exsn = assertEvalUnresolvedException("t();", "f", 0, 1); |
33362 | 108 |
assertTrue(exsn == f, "Identity must not change"); |
109 |
assertActiveKeys(); |
|
110 |
} |
|
111 |
||
112 |
public void testForwardClassToMethod() { |
|
113 |
DeclarationSnippet t = methodKey(assertEval("int t() { return new A().f(); }", added(RECOVERABLE_DEFINED))); |
|
114 |
assertUnresolvedDependencies1(t, RECOVERABLE_DEFINED, "class A"); |
|
115 |
assertEvalUnresolvedException("t();", "t", 1, 0); |
|
116 |
Snippet a = classKey(assertEval( |
|
117 |
"class A {\n" + |
|
118 |
" int f() { return 10; }\n" + |
|
119 |
"}", |
|
120 |
added(VALID), |
|
121 |
ste(t, RECOVERABLE_DEFINED, VALID, false, null))); |
|
122 |
assertEval("t();", "10"); |
|
123 |
assertEval( |
|
124 |
"class A {\n" + |
|
125 |
" double f() { return 88.0; }\n" + |
|
126 |
"}", |
|
127 |
DiagCheck.DIAG_OK, |
|
128 |
DiagCheck.DIAG_ERROR, |
|
129 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
130 |
ste(a, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
131 |
ste(t, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
|
132 |
assertEvalUnresolvedException("t();", "t", 0, 1); |
|
133 |
assertActiveKeys(); |
|
134 |
} |
|
135 |
||
136 |
public void testForwardVarToClass() { |
|
36499 | 137 |
DeclarationSnippet a = classKey(assertEval("class A { int f() { return g; } }", added(RECOVERABLE_DEFINED))); |
138 |
assertUnresolvedDependencies1(a, RECOVERABLE_DEFINED, "variable g"); |
|
139 |
Snippet g = varKey(assertEval("int g = 10;", "10", |
|
140 |
added(VALID), |
|
141 |
ste(a, RECOVERABLE_DEFINED, VALID, false, null))); |
|
142 |
assertEval("new A().f();", "10"); |
|
143 |
assertEval("double g = 10;", "10.0", null, |
|
144 |
DiagCheck.DIAG_OK, |
|
145 |
DiagCheck.DIAG_ERROR, |
|
146 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
147 |
ste(g, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
148 |
ste(a, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
|
149 |
assertUnresolvedDependencies(a, 0); |
|
150 |
assertActiveKeys(); |
|
151 |
} |
|
152 |
||
153 |
public void testForwardVarToClassGeneric() { |
|
154 |
DeclarationSnippet a = classKey(assertEval("class A<T> { final T x; A(T v) { this.x = v; } ; T get() { return x; } int core() { return g; } }", added(RECOVERABLE_DEFINED))); |
|
155 |
assertUnresolvedDependencies1(a, RECOVERABLE_DEFINED, "variable g"); |
|
156 |
||
157 |
List<SnippetEvent> events = assertEval("A<String> as = new A<>(\"hi\");", null, |
|
158 |
UnresolvedReferenceException.class, DiagCheck.DIAG_OK, DiagCheck.DIAG_OK, null); |
|
159 |
SnippetEvent ste = events.get(0); |
|
160 |
Snippet assn = ste.snippet(); |
|
161 |
DeclarationSnippet unsn = ((UnresolvedReferenceException) ste.exception()).getSnippet(); |
|
162 |
assertEquals(unsn.name(), "A", "Wrong with unresolved"); |
|
40304
0318f4e75c6d
8143964: JShell API: convert query responses to Stream instead of List
rfield
parents:
37644
diff
changeset
|
163 |
assertEquals(getState().unresolvedDependencies(unsn).count(), 1, "Wrong size unresolved"); |
0318f4e75c6d
8143964: JShell API: convert query responses to Stream instead of List
rfield
parents:
37644
diff
changeset
|
164 |
assertEquals(getState().diagnostics(unsn).count(), 0L, "Expected no diagnostics"); |
36499 | 165 |
|
33362 | 166 |
Snippet g = varKey(assertEval("int g = 10;", "10", |
167 |
added(VALID), |
|
36499 | 168 |
ste(a, RECOVERABLE_DEFINED, VALID, false, MAIN_SNIPPET))); |
169 |
assertEval("A<String> as = new A<>(\"low\");", |
|
170 |
ste(MAIN_SNIPPET, VALID, VALID, false, null), |
|
171 |
ste(assn, VALID, OVERWRITTEN, false, MAIN_SNIPPET)); |
|
172 |
assertEval("as.get();", "\"low\""); |
|
173 |
assertUnresolvedDependencies(a, 0); |
|
174 |
assertActiveKeys(); |
|
175 |
} |
|
176 |
||
177 |
public void testForwardVarToClassExtendsImplements() { |
|
178 |
DeclarationSnippet ik = classKey(assertEval("interface I { default int ii() { return 1; } }", added(VALID))); |
|
179 |
DeclarationSnippet jk = classKey(assertEval("interface J { default int jj() { return 2; } }", added(VALID))); |
|
180 |
DeclarationSnippet ck = classKey(assertEval("class C { int cc() { return 3; } }", added(VALID))); |
|
181 |
DeclarationSnippet dk = classKey(assertEval("class D extends C implements I,J { int dd() { return g; } }", added(RECOVERABLE_DEFINED))); |
|
182 |
DeclarationSnippet ek = classKey(assertEval("class E extends D { int ee() { return 5; } }", added(VALID))); |
|
183 |
assertUnresolvedDependencies1(dk, RECOVERABLE_DEFINED, "variable g"); |
|
184 |
assertEvalUnresolvedException("new D();", "D", 1, 0); |
|
185 |
assertEvalUnresolvedException("new E();", "D", 1, 0); |
|
186 |
VarSnippet g = varKey(assertEval("int g = 10;", "10", |
|
187 |
added(VALID), |
|
188 |
ste(dk, RECOVERABLE_DEFINED, VALID, false, MAIN_SNIPPET))); |
|
189 |
assertEval("E e = new E();"); |
|
190 |
assertDrop(g, |
|
191 |
ste(g, VALID, DROPPED, true, null), |
|
192 |
ste(dk, VALID, RECOVERABLE_DEFINED, false, g)); |
|
193 |
assertEvalUnresolvedException("new D();", "D", 1, 0); |
|
194 |
assertEvalUnresolvedException("new E();", "D", 1, 0); |
|
195 |
assertEval("e.ee();", "5"); |
|
196 |
assertEvalUnresolvedException("e.dd();", "D", 1, 0); |
|
197 |
assertEval("e.cc();", "3"); |
|
198 |
assertEval("e.jj();", "2"); |
|
199 |
assertEval("e.ii();", "1"); |
|
200 |
assertActiveKeys(); |
|
201 |
} |
|
202 |
||
203 |
public void testForwardVarToInterface() { |
|
204 |
DeclarationSnippet i = classKey(assertEval("interface I { default int f() { return x; } }", added(RECOVERABLE_DEFINED))); |
|
205 |
assertUnresolvedDependencies1(i, RECOVERABLE_DEFINED, "variable x"); |
|
206 |
DeclarationSnippet c = classKey(assertEval("class C implements I { int z() { return 2; } }", added(VALID))); |
|
207 |
assertEval("C c = new C();"); |
|
208 |
assertEval("c.z();", "2"); |
|
209 |
assertEvalUnresolvedException("c.f()", "I", 1, 0); |
|
210 |
Snippet g = varKey(assertEval("int x = 55;", "55", |
|
211 |
added(VALID), |
|
212 |
ste(i, RECOVERABLE_DEFINED, VALID, false, null))); |
|
213 |
assertEval("c.f();", "55"); |
|
214 |
assertUnresolvedDependencies(i, 0); |
|
215 |
assertActiveKeys(); |
|
216 |
} |
|
217 |
||
218 |
public void testForwardVarToEnum() { |
|
36780
6bf2bef08a91
8152925: JShell: enable corralling of any type declaration, including enum
rfield
parents:
36499
diff
changeset
|
219 |
DeclarationSnippet a = classKey(assertEval("enum E { Q, W, E; float ff() { return fff; } }", added(RECOVERABLE_DEFINED))); |
6bf2bef08a91
8152925: JShell: enable corralling of any type declaration, including enum
rfield
parents:
36499
diff
changeset
|
220 |
assertUnresolvedDependencies1(a, RECOVERABLE_DEFINED, "variable fff"); |
36499 | 221 |
Snippet g = varKey(assertEval("float fff = 4.5f;", "4.5", |
222 |
added(VALID), |
|
36780
6bf2bef08a91
8152925: JShell: enable corralling of any type declaration, including enum
rfield
parents:
36499
diff
changeset
|
223 |
ste(a, RECOVERABLE_DEFINED, VALID, false, null))); |
36499 | 224 |
assertEval("E.Q.ff();", "4.5"); |
225 |
assertEval("double fff = 3.3;", "3.3", null, |
|
33362 | 226 |
DiagCheck.DIAG_OK, |
227 |
DiagCheck.DIAG_ERROR, |
|
228 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
229 |
ste(g, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
36780
6bf2bef08a91
8152925: JShell: enable corralling of any type declaration, including enum
rfield
parents:
36499
diff
changeset
|
230 |
ste(a, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
33362 | 231 |
assertUnresolvedDependencies(a, 0); |
232 |
assertActiveKeys(); |
|
233 |
} |
|
234 |
||
235 |
public void testForwardMethodToClass() { |
|
36499 | 236 |
DeclarationSnippet a = classKey(assertEval("class A { int f() { return g(); } }", added(RECOVERABLE_DEFINED))); |
237 |
assertUnresolvedDependencies1(a, RECOVERABLE_DEFINED, "method g()"); |
|
238 |
assertEval("A foo() { return null; }"); |
|
239 |
assertEvalUnresolvedException("new A();", "A", 1, 0); |
|
33362 | 240 |
Snippet g = methodKey(assertEval("int g() { return 10; }", |
241 |
added(VALID), |
|
36499 | 242 |
ste(a, RECOVERABLE_DEFINED, VALID, false, null))); |
33362 | 243 |
assertEval("new A().f();", "10"); |
244 |
assertEval("double g() { return 10; }", |
|
245 |
DiagCheck.DIAG_OK, |
|
246 |
DiagCheck.DIAG_ERROR, |
|
247 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
248 |
ste(g, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
36499 | 249 |
ste(a, VALID, RECOVERABLE_DEFINED, false, MAIN_SNIPPET)); |
33362 | 250 |
assertUnresolvedDependencies(a, 0); |
251 |
assertActiveKeys(); |
|
252 |
} |
|
253 |
||
254 |
public void testForwardClassToClass1() { |
|
255 |
Snippet a = classKey(assertEval("class A { B b = new B(); }", added(RECOVERABLE_NOT_DEFINED))); |
|
256 |
assertDeclareFail("new A().b;", "compiler.err.cant.resolve.location"); |
|
257 |
||
258 |
Snippet b = classKey(assertEval("class B { public String toString() { return \"B\"; } }", |
|
259 |
added(VALID), |
|
260 |
ste(a, RECOVERABLE_NOT_DEFINED, VALID, true, null))); |
|
261 |
assertEval("new A().b;", "B"); |
|
262 |
assertEval("interface B { }", |
|
263 |
DiagCheck.DIAG_OK, |
|
264 |
DiagCheck.DIAG_ERROR, |
|
265 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
266 |
ste(b, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
36499 | 267 |
ste(a, VALID, RECOVERABLE_DEFINED, true, MAIN_SNIPPET)); |
268 |
assertEvalUnresolvedException("new A().b;", "A", 0, 1); |
|
33362 | 269 |
assertActiveKeys(); |
270 |
} |
|
271 |
||
272 |
public void testForwardClassToClass2() { |
|
273 |
Snippet a = classKey(assertEval("class A extends B { }", added(RECOVERABLE_NOT_DEFINED))); |
|
274 |
assertDeclareFail("new A();", "compiler.err.cant.resolve.location"); |
|
275 |
||
276 |
Snippet b = classKey(assertEval("class B { public String toString() { return \"B\"; } }", |
|
277 |
added(VALID), |
|
278 |
ste(a, RECOVERABLE_NOT_DEFINED, VALID, true, null))); |
|
279 |
assertEval("new A();", "B"); |
|
280 |
assertEval("interface B { }", |
|
281 |
DiagCheck.DIAG_OK, |
|
282 |
DiagCheck.DIAG_ERROR, |
|
283 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
284 |
ste(b, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
285 |
ste(a, VALID, RECOVERABLE_NOT_DEFINED, true, MAIN_SNIPPET)); |
|
286 |
assertDeclareFail("new A();", "compiler.err.cant.resolve.location"); |
|
287 |
assertActiveKeys(); |
|
288 |
} |
|
289 |
||
290 |
public void testForwardClassToClass3() { |
|
291 |
Snippet a = classKey(assertEval("interface A extends B { static int f() { return 10; } }", added(RECOVERABLE_NOT_DEFINED))); |
|
292 |
assertDeclareFail("A.f();", "compiler.err.cant.resolve.location"); |
|
293 |
||
294 |
Snippet b = classKey(assertEval("interface B { }", |
|
295 |
added(VALID), |
|
296 |
ste(a, RECOVERABLE_NOT_DEFINED, VALID, true, null))); |
|
297 |
assertEval("A.f();", "10"); |
|
298 |
assertEval("class B { }", |
|
299 |
DiagCheck.DIAG_OK, |
|
300 |
DiagCheck.DIAG_ERROR, |
|
301 |
ste(MAIN_SNIPPET, VALID, VALID, true, null), |
|
302 |
ste(b, VALID, OVERWRITTEN, false, MAIN_SNIPPET), |
|
303 |
ste(a, VALID, RECOVERABLE_NOT_DEFINED, true, MAIN_SNIPPET)); |
|
304 |
assertDeclareFail("A.f();", "compiler.err.cant.resolve.location"); |
|
305 |
assertActiveKeys(); |
|
306 |
} |
|
307 |
||
308 |
public void testForwardVariable() { |
|
309 |
assertEval("int f() { return x; }", added(RECOVERABLE_DEFINED)); |
|
310 |
assertEvalUnresolvedException("f();", "f", 1, 0); |
|
311 |
assertActiveKeys(); |
|
312 |
} |
|
313 |
||
314 |
public void testLocalClassInUnresolved() { |
|
315 |
Snippet f = methodKey(assertEval("void f() { class A {} g(); }", added(RECOVERABLE_DEFINED))); |
|
316 |
assertEval("void g() {}", |
|
317 |
added(VALID), |
|
50735
2f2af62dfac7
8010319: Implementation of JEP 181: Nest-Based Access Control
dholmes
parents:
47216
diff
changeset
|
318 |
ste(f, RECOVERABLE_DEFINED, VALID, true, null)); |
33362 | 319 |
assertEval("f();", ""); |
320 |
} |
|
321 |
} |