1 /*
2 * Copyright (c) 2021, 2023, Oracle and/or its affiliates. All rights reserved.
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 */
36 import java.util.stream.Stream;
37
38 import org.junit.jupiter.api.Test;
39 import org.junit.jupiter.params.ParameterizedTest;
40 import org.junit.jupiter.params.provider.MethodSource;
41 import static org.junit.jupiter.api.Assertions.*;
42
43 class WithScopedValue {
44
45 private static Stream<ThreadFactory> factories() {
46 return Stream.of(Thread.ofPlatform().factory(), Thread.ofVirtual().factory());
47 }
48
49 /**
50 * Test inheritance of a scoped value.
51 */
52 @ParameterizedTest
53 @MethodSource("factories")
54 void testInheritsScopedValue(ThreadFactory factory) throws Exception {
55 ScopedValue<String> name = ScopedValue.newInstance();
56 String value = ScopedValue.callWhere(name, "duke", () -> {
57 var result = new AtomicReference<String>();
58 try (var flock = ThreadFlock.open(null)) {
59 Thread thread = factory.newThread(() -> {
60 // child
61 result.set(name.get());
62 });
63 flock.start(thread);
64 }
65 return result.get();
66 });
67 assertEquals("duke", value);
68 }
69
70 /**
71 * Test exiting a dynamic scope with open thread flocks.
72 */
73 @Test
74 void testStructureViolation1() {
75 ScopedValue<String> name = ScopedValue.newInstance();
76 class Box {
77 ThreadFlock flock1;
78 ThreadFlock flock2;
79 }
80 var box = new Box();
81 try {
82 ScopedValue.runWhere(name, "x1", () -> {
83 box.flock1 = ThreadFlock.open(null);
84 box.flock2 = ThreadFlock.open(null);
85 });
86 fail();
87 } catch (StructureViolationException expected) { }
88 assertTrue(box.flock1.isClosed());
89 assertTrue(box.flock2.isClosed());
90 }
91
92 /**
93 * Test closing a thread flock while in a dynamic scope and with enclosing thread
94 * flocks. This test closes enclosing flock1.
95 */
96 @Test
97 void testStructureViolation2() {
98 ScopedValue<String> name = ScopedValue.newInstance();
99 try (var flock1 = ThreadFlock.open("flock1")) {
100 ScopedValue.runWhere(name, "x1", () -> {
101 try (var flock2 = ThreadFlock.open("flock2")) {
102 ScopedValue.runWhere(name, "x2", () -> {
103 try (var flock3 = ThreadFlock.open("flock3")) {
104 ScopedValue.runWhere(name, "x3", () -> {
105 var flock4 = ThreadFlock.open("flock4");
106
107 try {
108 flock1.close();
109 fail();
110 } catch (StructureViolationException expected) { }
111
112 assertTrue(flock1.isClosed());
113 assertTrue(flock2.isClosed());
114 assertTrue(flock3.isClosed());
115 assertTrue(flock4.isClosed());
116 });
117 }
118 });
119 }
120 });
121 }
122 }
123
124 /**
125 * Test closing a thread flock while in a dynamic scope and with enclosing thread
126 * flocks. This test closes enclosing flock2.
127 */
128 @Test
129 void testStructureViolation3() {
130 ScopedValue<String> name = ScopedValue.newInstance();
131 try (var flock1 = ThreadFlock.open("flock1")) {
132 ScopedValue.runWhere(name, "x1", () -> {
133 try (var flock2 = ThreadFlock.open("flock2")) {
134 ScopedValue.runWhere(name, "x2", () -> {
135 try (var flock3 = ThreadFlock.open("flock3")) {
136 ScopedValue.runWhere(name, "x3", () -> {
137 var flock4 = ThreadFlock.open("flock4");
138
139 try {
140 flock2.close();
141 fail();
142 } catch (StructureViolationException expected) { }
143
144 assertFalse(flock1.isClosed());
145 assertTrue(flock2.isClosed());
146 assertTrue(flock3.isClosed());
147 assertTrue(flock4.isClosed());
148 });
149 }
150 });
151 }
152 });
153 }
154 }
155
156 /**
157 * Test closing a thread flock while in a dynamic scope and with enclosing thread
158 * flocks. This test closes enclosing flock3.
159 */
160 @Test
161 void testStructureViolation4() {
162 ScopedValue<String> name = ScopedValue.newInstance();
163 try (var flock1 = ThreadFlock.open("flock1")) {
164 ScopedValue.runWhere(name, "x1", () -> {
165 try (var flock2 = ThreadFlock.open("flock2")) {
166 ScopedValue.runWhere(name, "x2", () -> {
167 try (var flock3 = ThreadFlock.open("flock3")) {
168 ScopedValue.runWhere(name, "x3", () -> {
169 var flock4 = ThreadFlock.open("flock4");
170
171 try {
172 flock3.close();
173 fail();
174 } catch (StructureViolationException expected) { }
175
176 assertFalse(flock1.isClosed());
177 assertFalse(flock2.isClosed());
178 assertTrue(flock3.isClosed());
179 assertTrue(flock4.isClosed());
180 });
181 }
182 });
183 }
184 });
185 }
186 }
187
188 /**
189 * Test start when a scoped value is bound after a thread flock is created.
190 */
191 @ParameterizedTest
192 @MethodSource("factories")
193 void testStructureViolation5(ThreadFactory factory) throws Exception {
194 ScopedValue<String> name = ScopedValue.newInstance();
195 try (var flock = ThreadFlock.open(null)) {
196 ScopedValue.runWhere(name, "duke", () -> {
197 Thread thread = factory.newThread(() -> { });
198 assertThrows(StructureViolationException.class, () -> flock.start(thread));
199 });
200 }
201 }
202
203 /**
204 * Test start when a scoped value is re-bound after a thread flock is created.
205 */
206 @ParameterizedTest
207 @MethodSource("factories")
208 void testStructureViolation6(ThreadFactory factory) throws Exception {
209 ScopedValue<String> name = ScopedValue.newInstance();
210 ScopedValue.runWhere(name, "duke", () -> {
211 try (var flock = ThreadFlock.open(null)) {
212 ScopedValue.runWhere(name, "duchess", () -> {
213 Thread thread = factory.newThread(() -> { });
214 assertThrows(StructureViolationException.class, () -> flock.start(thread));
215 });
216 }
217 });
218 }
219 }
|
1 /*
2 * Copyright (c) 2021, 2024, Oracle and/or its affiliates. All rights reserved.
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 */
36 import java.util.stream.Stream;
37
38 import org.junit.jupiter.api.Test;
39 import org.junit.jupiter.params.ParameterizedTest;
40 import org.junit.jupiter.params.provider.MethodSource;
41 import static org.junit.jupiter.api.Assertions.*;
42
43 class WithScopedValue {
44
45 private static Stream<ThreadFactory> factories() {
46 return Stream.of(Thread.ofPlatform().factory(), Thread.ofVirtual().factory());
47 }
48
49 /**
50 * Test inheritance of a scoped value.
51 */
52 @ParameterizedTest
53 @MethodSource("factories")
54 void testInheritsScopedValue(ThreadFactory factory) throws Exception {
55 ScopedValue<String> name = ScopedValue.newInstance();
56 String value = ScopedValue.where(name, "duke").call(() -> {
57 var result = new AtomicReference<String>();
58 try (var flock = ThreadFlock.open(null)) {
59 Thread thread = factory.newThread(() -> {
60 // child
61 result.set(name.get());
62 });
63 flock.start(thread);
64 }
65 return result.get();
66 });
67 assertEquals("duke", value);
68 }
69
70 /**
71 * Test exiting a dynamic scope with open thread flocks.
72 */
73 @Test
74 void testStructureViolation1() {
75 ScopedValue<String> name = ScopedValue.newInstance();
76 class Box {
77 ThreadFlock flock1;
78 ThreadFlock flock2;
79 }
80 var box = new Box();
81 try {
82 ScopedValue.where(name, "x1").run(() -> {
83 box.flock1 = ThreadFlock.open(null);
84 box.flock2 = ThreadFlock.open(null);
85 });
86 fail();
87 } catch (StructureViolationException expected) { }
88 assertTrue(box.flock1.isClosed());
89 assertTrue(box.flock2.isClosed());
90 }
91
92 /**
93 * Test closing a thread flock while in a dynamic scope and with enclosing thread
94 * flocks. This test closes enclosing flock1.
95 */
96 @Test
97 void testStructureViolation2() {
98 ScopedValue<String> name = ScopedValue.newInstance();
99 try (var flock1 = ThreadFlock.open("flock1")) {
100 ScopedValue.where(name, "x1").run(() -> {
101 try (var flock2 = ThreadFlock.open("flock2")) {
102 ScopedValue.where(name, "x2").run(() -> {
103 try (var flock3 = ThreadFlock.open("flock3")) {
104 ScopedValue.where(name, "x3").run(() -> {
105 var flock4 = ThreadFlock.open("flock4");
106
107 try {
108 flock1.close();
109 fail();
110 } catch (StructureViolationException expected) { }
111
112 assertTrue(flock1.isClosed());
113 assertTrue(flock2.isClosed());
114 assertTrue(flock3.isClosed());
115 assertTrue(flock4.isClosed());
116 });
117 }
118 });
119 }
120 });
121 }
122 }
123
124 /**
125 * Test closing a thread flock while in a dynamic scope and with enclosing thread
126 * flocks. This test closes enclosing flock2.
127 */
128 @Test
129 void testStructureViolation3() {
130 ScopedValue<String> name = ScopedValue.newInstance();
131 try (var flock1 = ThreadFlock.open("flock1")) {
132 ScopedValue.where(name, "x1").run(() -> {
133 try (var flock2 = ThreadFlock.open("flock2")) {
134 ScopedValue.where(name, "x2").run(() -> {
135 try (var flock3 = ThreadFlock.open("flock3")) {
136 ScopedValue.where(name, "x3").run(() -> {
137 var flock4 = ThreadFlock.open("flock4");
138
139 try {
140 flock2.close();
141 fail();
142 } catch (StructureViolationException expected) { }
143
144 assertFalse(flock1.isClosed());
145 assertTrue(flock2.isClosed());
146 assertTrue(flock3.isClosed());
147 assertTrue(flock4.isClosed());
148 });
149 }
150 });
151 }
152 });
153 }
154 }
155
156 /**
157 * Test closing a thread flock while in a dynamic scope and with enclosing thread
158 * flocks. This test closes enclosing flock3.
159 */
160 @Test
161 void testStructureViolation4() {
162 ScopedValue<String> name = ScopedValue.newInstance();
163 try (var flock1 = ThreadFlock.open("flock1")) {
164 ScopedValue.where(name, "x1").run(() -> {
165 try (var flock2 = ThreadFlock.open("flock2")) {
166 ScopedValue.where(name, "x2").run(() -> {
167 try (var flock3 = ThreadFlock.open("flock3")) {
168 ScopedValue.where(name, "x3").run(() -> {
169 var flock4 = ThreadFlock.open("flock4");
170
171 try {
172 flock3.close();
173 fail();
174 } catch (StructureViolationException expected) { }
175
176 assertFalse(flock1.isClosed());
177 assertFalse(flock2.isClosed());
178 assertTrue(flock3.isClosed());
179 assertTrue(flock4.isClosed());
180 });
181 }
182 });
183 }
184 });
185 }
186 }
187
188 /**
189 * Test start when a scoped value is bound after a thread flock is created.
190 */
191 @ParameterizedTest
192 @MethodSource("factories")
193 void testStructureViolation5(ThreadFactory factory) throws Exception {
194 ScopedValue<String> name = ScopedValue.newInstance();
195 try (var flock = ThreadFlock.open(null)) {
196 ScopedValue.where(name, "duke").run(() -> {
197 Thread thread = factory.newThread(() -> { });
198 assertThrows(StructureViolationException.class, () -> flock.start(thread));
199 });
200 }
201 }
202
203 /**
204 * Test start when a scoped value is re-bound after a thread flock is created.
205 */
206 @ParameterizedTest
207 @MethodSource("factories")
208 void testStructureViolation6(ThreadFactory factory) throws Exception {
209 ScopedValue<String> name = ScopedValue.newInstance();
210 ScopedValue.where(name, "duke").run(() -> {
211 try (var flock = ThreadFlock.open(null)) {
212 ScopedValue.where(name, "duchess").run(() -> {
213 Thread thread = factory.newThread(() -> { });
214 assertThrows(StructureViolationException.class, () -> flock.start(thread));
215 });
216 }
217 });
218 }
219 }
|