Skip to content

Commit ef84eca

Browse files
committedAug 29, 2024
Merge
2 parents 42aef90 + f96747b commit ef84eca

File tree

2 files changed

+192
-141
lines changed

2 files changed

+192
-141
lines changed
 

‎src/java.base/share/classes/java/util/concurrent/StructuredTaskScope.java

+153-80
Original file line numberDiff line numberDiff line change
@@ -59,8 +59,10 @@
5959
* beyond the {@code close} method until all threads started to execute subtasks have finished.
6060
* To ensure correct usage, the {@code fork}, {@code join} and {@code close} methods may
6161
* only be invoked by the <em>owner thread</em> (the thread that opened the {@code
62-
* StructuredTaskScope}), and the {@code close} method throws an exception after closing
63-
* if the owner did not invoke the {@code join} method.
62+
* StructuredTaskScope}), the {@code fork} method may not be called after {@code join},
63+
* the {@code join} method may only be invoked once, and the {@code close} method throws
64+
* an exception after closing if the owner did not invoke the {@code join} method after
65+
* forking subtasks.
6466
*
6567
* <p> As a first example, consider a main task that splits into two subtasks to concurrently
6668
* fetch resources from two URL locations "left" and "right". Both subtasks may complete
@@ -89,7 +91,7 @@
8991
* completes and the main task uses the {@link Subtask#get() Subtask.get()} method to get
9092
* the result of each subtask. If one of the subtasks fails then the other subtask
9193
* is cancelled (this will interrupt the thread executing the other subtask) and the
92-
* {@code join} method throws {@link ExecutionException} with the exception from
94+
* {@code join} method throws {@link FailedException} with the exception from
9395
* the failed subtask as the {@linkplain Throwable#getCause() cause}.
9496
*
9597
* <p> A {@code StructuredTaskScope} may be opened with a {@link Joiner} that handles subtask
@@ -102,12 +104,16 @@
102104
* require the result of subtasks that are still executing. Cancelling execution prevents
103105
* new threads from being started to execute further subtasks, {@linkplain Thread#interrupt()
104106
* interrupts} the threads executing subtasks that have not completed, and causes the
105-
* {@code join} method to wakeup with a result (or exception). The {@link #close() close}
106-
* method always waits for threads executing subtasks to finish, even if execution is
107-
* cancelled, so it cannot continue beyond the {@code close} method until the interrupted
108-
* threads finish. Subtasks should be coded so that they finish as soon as possible when
109-
* interrupted. Subtasks that do not respond to interrupt, e.g. block on methods that are
110-
* not interruptible, may delay the closing of a task scope indefinitely.
107+
* {@code join} method to wakeup with a result (or exception). In the above example,
108+
* the no-arg {@link #open() open} method created the {@code StructuredTaskScope} with a
109+
* {@code Joiner} that cancelled execution when any subtask failed.
110+
*
111+
* <p> The {@link #close() close} method always waits for threads executing subtasks to
112+
* finish, even if execution is cancelled, so it cannot continue beyond the {@code close}
113+
* method until the interrupted threads finish. Subtasks should be coded so that they
114+
* finish as soon as possible when interrupted. Subtasks that do not respond to interrupt,
115+
* e.g. block on methods that are not interruptible, may delay the closing of a task scope
116+
* indefinitely.
111117
*
112118
* <p> Now consider another example that also splits into two subtasks to concurrently
113119
* fetch resources. In this example, the code in the main task is only interested in the
@@ -136,7 +142,7 @@
136142
* subtask to be cancelled (this will interrupt the thread executing the subtask), and
137143
* the {@code join} method returns the result from the first subtask. Cancelling the other
138144
* subtask avoids the main task waiting for a result that it doesn't care about. If both
139-
* subtasks fail then the {@code join} method throws {@link ExecutionException} with the
145+
* subtasks fail then the {@code join} method throws {@link FailedException} with the
140146
* exception from one of the subtasks as the {@linkplain Throwable#getCause() cause}.
141147
*
142148
* <p> Whether code uses the {@code Subtask} returned from {@code fork} will depend on
@@ -152,19 +158,34 @@
152158
* handles subtask completion and produces the outcome for the {@link #join() join} method.
153159
* In some cases, the outcome will be a result, in other cases it will be an exception.
154160
* If the outcome is an exception then the {@code join} method throws {@link
155-
* ExecutionException} with the exception as the {@linkplain Throwable#getCause()
161+
* FailedException} with the exception as the {@linkplain Throwable#getCause()
156162
* cause}. For many {@code Joiner} implementations, the exception will be an exception
157163
* thrown by a subtask that failed. In the case of {@link Joiner#allSuccessfulOrThrow()
158164
* allSuccessfulOrThrow} and {@link Joiner#awaitAllSuccessfulOrThrow() awaitAllSuccessfulOrThrow}
159165
* for example, the exception is from the first subtask to fail.
160166
*
161167
* <p> Many of the details for how exceptions are handled will depend on usage. In some
162-
* cases, the {@code join} method will be called in a {@code try-catch} block to catch
163-
* {@code ExecutionException} and handle the cause. The exception handling may use
164-
* {@code instanceof} with pattern matching to handle specific causes. In some cases it
165-
* may not be useful to catch {@code ExecutionException} but instead leave it to propagate
166-
* to the configured {@linkplain Thread.UncaughtExceptionHandler uncaught exception handler}
167-
* for logging purposes.
168+
* cases it may be useful t add a {@code catch} block to catch {@code FailedException}.
169+
* The exception handling may use {@code instanceof} with pattern matching to handle
170+
* specific causes.
171+
* {@snippet lang=java :
172+
* try (var scope = StructuredTaskScope.open()) {
173+
*
174+
* ..
175+
*
176+
* } catch (StructuredTaskScope.FailedException e) {
177+
*
178+
* Throwable cause = e.getCause();
179+
* switch (cause) {
180+
* case IOException ioe -> ..
181+
* default -> ..
182+
* }
183+
*
184+
* }
185+
* }
186+
* In some cases it may not be useful to catch {@code FailedException} but instead leave
187+
* it to propagate to the configured {@linkplain Thread.UncaughtExceptionHandler uncaught
188+
* exception handler} for logging purposes.
168189
*
169190
* <p> For cases where a specific exception triggers the use of a default result then it
170191
* may be more appropriate to handle this in the subtask itself rather than the subtask
@@ -212,7 +233,7 @@
212233
* starts when the new task scope is opened. If the timeout expires before the {@code join}
213234
* method has completed then <a href="#CancelExecution">execution is cancelled</a>. This
214235
* interrupts the threads executing the two subtasks and causes the {@link #join() join}
215-
* method to throw {@link ExecutionException} with {@link TimeoutException} as the cause.
236+
* method to throw {@link FailedException} with {@link TimeoutException} as the cause.
216237
* {@snippet lang=java :
217238
* Duration timeout = Duration.ofSeconds(10);
218239
*
@@ -337,10 +358,15 @@ public class StructuredTaskScope<T, R> implements AutoCloseable {
337358
private final ThreadFactory threadFactory;
338359
private final ThreadFlock flock;
339360

340-
// fields that are only accessed by owner thread
341-
private boolean needToJoin; // set by fork to indicate that owner must join
342-
private boolean joined; // set to true when owner joins
343-
private boolean closed;
361+
// state, only accessed by owner thread
362+
private static final int ST_NEW = 0;
363+
private static final int ST_FORKED = 1; // subtasks forked, need to join
364+
private static final int ST_JOIN_STARTED = 2; // join started, can no longer fork
365+
private static final int ST_JOIN_COMPLETED = 3; // join completed
366+
private static final int ST_CLOSED = 4; // closed
367+
private int state;
368+
369+
// timer task, only accessed by owner thread
344370
private Future<?> timerTask;
345371

346372
// set or read by any thread
@@ -349,22 +375,32 @@ public class StructuredTaskScope<T, R> implements AutoCloseable {
349375
// set by the timer thread, read by the owner thread
350376
private volatile boolean timeoutExpired;
351377

378+
/**
379+
* Throws WrongThreadException if the current thread is not the owner thread.
380+
*/
381+
private void ensureOwner() {
382+
if (Thread.currentThread() != flock.owner()) {
383+
throw new WrongThreadException("Current thread not owner");
384+
}
385+
}
386+
352387
/**
353388
* Throws IllegalStateException if the task scope is closed.
354389
*/
355390
private void ensureOpen() {
356391
assert Thread.currentThread() == flock.owner();
357-
if (closed) {
392+
if (state == ST_CLOSED) {
358393
throw new IllegalStateException("Task scope is closed");
359394
}
360395
}
361396

362397
/**
363-
* Throws WrongThreadException if the current thread is not the owner thread.
398+
* Throws IllegalStateException if the already joined or task scope is closed.
364399
*/
365-
private void ensureOwner() {
366-
if (Thread.currentThread() != flock.owner()) {
367-
throw new WrongThreadException("Current thread not owner");
400+
private void ensureNotJoined() {
401+
assert Thread.currentThread() == flock.owner();
402+
if (state > ST_FORKED) {
403+
throw new IllegalStateException("Already joined or task scope is closed");
368404
}
369405
}
370406

@@ -373,9 +409,8 @@ private void ensureOwner() {
373409
* has not joined.
374410
*/
375411
private void ensureJoinedIfOwner() {
376-
if (Thread.currentThread() == flock.owner() && !joined) {
377-
String msg = needToJoin ? "Owner did not join" : "join did not complete";
378-
throw new IllegalStateException(msg);
412+
if (Thread.currentThread() == flock.owner() && state <= ST_JOIN_STARTED) {
413+
throw new IllegalStateException("join not called");
379414
}
380415
}
381416

@@ -531,7 +566,8 @@ enum State {
531566
*
532567
* @return the possibly-null result
533568
* @throws IllegalStateException if the subtask has not completed, did not complete
534-
* successfully, or the current thread is the task scope owner and it has not joined
569+
* successfully, or the current thread is the task scope owner invoking this
570+
* method before {@linkplain #join() joining}
535571
* @see State#SUCCESS
536572
*/
537573
T get();
@@ -552,7 +588,8 @@ enum State {
552588
* {@link State#FAILED FAILED} before using this method to get the exception.
553589
*
554590
* @throws IllegalStateException if the subtask has not completed, completed with
555-
* a result, or the current thread is the task scope owner and it has not joined
591+
* a result, or the current thread is the task scope owner invoking this method
592+
* before {@linkplain #join() joining}
556593
* @see State#FAILED
557594
*/
558595
Throwable exception();
@@ -576,7 +613,7 @@ enum State {
576613
* {@code Joiner} that waits for all successful subtasks. It cancels execution and
577614
* causes {@code join} to throw if any subtask fails.
578615
* <li> {@link #awaitAll() awaitAll()} creates a {@code Joiner} that waits for all
579-
* subtasks. If does not cancel execution.
616+
* subtasks. It does not cancel execution or cause {@code join} to throw.
580617
* </ul>
581618
*
582619
* <p> In addition to the methods to create {@code Joiner} objects for common cases,
@@ -673,14 +710,14 @@ default boolean onComplete(Subtask<? extends T> subtask) {
673710
* Invoked by {@link #join()} to produce the result (or exception) after waiting
674711
* for all subtasks to complete or execution to be cancelled. The result from this
675712
* method is returned by the {@code join} method. If this method throws, then
676-
* {@code join} throws {@link ExecutionException} with the exception thrown by
713+
* {@code join} throws {@link FailedException} with the exception thrown by
677714
* this method as the cause.
678715
*
679-
* <p> In normal usage, this method will be called at most once to produce the
680-
* result (or exception). If the {@code join} method is called more than once
681-
* then this method may be called more than once to produce the result. An
682-
* implementation should return an equal result (or throw the same exception) on
683-
* second or subsequent calls to produce the outcome.
716+
* <p> In normal usage, this method will be called at most once by the {@code join}
717+
* method to produce the result (or exception). The behavior of this method when
718+
* invoked directly, and invoked more than once, is not specified. Where possible,
719+
* an implementation should return an equal result (or throw the same exception)
720+
* on second or subsequent calls to produce the outcome.
684721
*
685722
* @apiNote This method is invoked by the {@code join} method. It should not be
686723
* invoked directly.
@@ -823,8 +860,8 @@ static <T> Joiner<T, Stream<Subtask<T>>> all(Predicate<Subtask<? extends T>> isD
823860
* ThreadFactory} to create threads, an optional name for the purposes of monitoring
824861
* and management, and an optional timeout.
825862
*
826-
* <p> Creating a {@code StructuredTaskScope} with its 1-arg {@link #open(Joiner) open}
827-
* method uses the <a href="StructuredTaskScope.html#DefaultConfiguration">default
863+
* <p> Creating a {@code StructuredTaskScope} with {@link #open()} or {@link #open(Joiner)}
864+
* uses the <a href="StructuredTaskScope.html#DefaultConfiguration">default
828865
* configuration</a>. The default configuration consists of a thread factory that
829866
* creates unnamed <a href="{@docRoot}/java.base/java/lang/Thread.html#virtual-threads">
830867
* virtual threads</a>, no name for monitoring and management purposes, and no timeout.
@@ -881,6 +918,45 @@ public sealed interface Config permits ConfigImpl {
881918
Config withTimeout(Duration timeout);
882919
}
883920

921+
/**
922+
* Exception thrown by {@link #join()} when the outcome is an exception rather than a
923+
* result.
924+
*
925+
* @since 24
926+
*/
927+
@PreviewFeature(feature = PreviewFeature.Feature.STRUCTURED_CONCURRENCY)
928+
public static class FailedException extends RuntimeException {
929+
@java.io.Serial
930+
static final long serialVersionUID = -1533055100078459923L;
931+
932+
/**
933+
* Constructs a {@code FailedException} with the specified cause.
934+
*
935+
* @param cause the cause, can be {@code null}
936+
*/
937+
public FailedException(Throwable cause) {
938+
super(cause);
939+
}
940+
}
941+
942+
/**
943+
* Exception thrown by {@link #join()} if the task scope is created with the timeout
944+
* expires before or while waiting in {@code join}.
945+
*
946+
* @since 24
947+
* @see Config#withTimeout(Duration)
948+
*/
949+
@PreviewFeature(feature = PreviewFeature.Feature.STRUCTURED_CONCURRENCY)
950+
public static class TimeoutException extends RuntimeException {
951+
@java.io.Serial
952+
static final long serialVersionUID = 705788143955048766L;
953+
954+
/**
955+
* Constructs a {@code TimeoutException} with no detail message.
956+
*/
957+
public TimeoutException() { }
958+
}
959+
884960
/**
885961
* Opens a new structured task scope to use the given {@code Joiner} object and with
886962
* configuration that is the result of applying the given function to the
@@ -898,7 +974,7 @@ public sealed interface Config permits ConfigImpl {
898974
* <p> If a {@linkplain Config#withTimeout(Duration) timeout} is set then it starts
899975
* when the task scope is opened. If the timeout expires before the task scope has
900976
* {@linkplain #join() joined} then execution is cancelled and the {@code join} method
901-
* throws {@link ExecutionException} with {@link TimeoutException} as the cause.
977+
* throws {@link FailedException} with {@link TimeoutException} as the cause.
902978
*
903979
* <p> The new task scope is owned by the current thread. Only code executing in this
904980
* thread can {@linkplain #fork(Callable) fork}, {@linkplain #join() join}, or
@@ -965,7 +1041,7 @@ public static <T, R> StructuredTaskScope<T, R> open(Joiner<? super T, ? extends
9651041
* or any subtask to fail.
9661042
*
9671043
* <p> The {@code join} method returns {@code null} if all subtasks complete successfully.
968-
* It throws {@link ExecutionException} if any subtask fails, with the exception from
1044+
* It throws {@link FailedException} if any subtask fails, with the exception from
9691045
* the first subtask to fail as the cause.
9701046
*
9711047
* <p> The task scope is created with the <a href="#DefaultConfiguration">default
@@ -1023,9 +1099,9 @@ public static <T> StructuredTaskScope<T, Void> open() {
10231099
* @param task the value-returning task for the thread to execute
10241100
* @param <U> the result type
10251101
* @return the subtask
1026-
* @throws IllegalStateException if this task scope is closed or the owner has already
1027-
* joined
10281102
* @throws WrongThreadException if the current thread is not the task scope owner
1103+
* @throws IllegalStateException if the owner has already {@linkplain #join() joined}
1104+
* or the task scope is closed
10291105
* @throws StructureViolationException if the current scoped value bindings are not
10301106
* the same as when the task scope was created
10311107
* @throws RejectedExecutionException if the thread factory rejected creating a
@@ -1034,10 +1110,7 @@ public static <T> StructuredTaskScope<T, Void> open() {
10341110
public <U extends T> Subtask<U> fork(Callable<? extends U> task) {
10351111
Objects.requireNonNull(task);
10361112
ensureOwner();
1037-
ensureOpen();
1038-
if (joined) {
1039-
throw new IllegalStateException("Already joined");
1040-
}
1113+
ensureNotJoined();
10411114

10421115
var subtask = new SubtaskImpl<U>(this, task);
10431116

@@ -1062,7 +1135,7 @@ public <U extends T> Subtask<U> fork(Callable<? extends U> task) {
10621135
}
10631136
}
10641137

1065-
needToJoin = true;
1138+
state = ST_FORKED;
10661139
return subtask;
10671140
}
10681141

@@ -1076,9 +1149,9 @@ public <U extends T> Subtask<U> fork(Callable<? extends U> task) {
10761149
*
10771150
* @param task the task for the thread to execute
10781151
* @return the subtask
1079-
* @throws IllegalStateException if this task scope is closed or the owner has already
1080-
* joined
10811152
* @throws WrongThreadException if the current thread is not the task scope owner
1153+
* @throws IllegalStateException if the owner has already {@linkplain #join() joined}
1154+
* or the task scope is closed
10821155
* @throws StructureViolationException if the current scoped value bindings are not
10831156
* the same as when the task scope was created
10841157
* @throws RejectedExecutionException if the thread factory rejected creating a
@@ -1094,10 +1167,10 @@ public Subtask<? extends T> fork(Runnable task) {
10941167
* Waits for all subtasks started in this task scope to complete or execution to be
10951168
* cancelled. If a {@linkplain Config#withTimeout(Duration) timeout} has been set
10961169
* then execution will be cancelled if the timeout expires before or while waiting.
1097-
* Once finished waiting, the {@code Joiner}'s {@link Joiner#result() result}
1098-
* method is invoked to get the result or throw an exception. If the {@code result}
1099-
* method throws then this method throws {@code ExecutionException} with the
1100-
* exception thrown by the {@code result()} method as the cause.
1170+
* Once finished waiting, the {@code Joiner}'s {@link Joiner#result() result} method
1171+
* is invoked to get the result or throw an exception. If the {@code result} method
1172+
* throws then this method throws {@code FailedException} with the exception thrown
1173+
* by the {@code result()} method as the cause.
11011174
*
11021175
* <p> This method waits for all subtasks by waiting for all threads {@linkplain
11031176
* #fork(Callable) started} in this task scope to finish execution. It stops waiting
@@ -1106,42 +1179,42 @@ public Subtask<? extends T> fork(Runnable task) {
11061179
* to cancel execution, the timeout (if set) expires, or the current thread is
11071180
* {@linkplain Thread#interrupt() interrupted}.
11081181
*
1109-
* <p> This method may only be invoked by the task scope owner.
1182+
* <p> This method may only be invoked by the task scope owner, and only once.
11101183
*
11111184
* @return the {@link Joiner#result() result}
1112-
* @throws IllegalStateException if this task scope is closed
11131185
* @throws WrongThreadException if the current thread is not the task scope owner
1114-
* @throws ExecutionException if the joiner's {@code result} method throws, or with
1115-
* cause {@link TimeoutException} if a timeout is set and the timeout expires
1186+
* @throws IllegalStateException if already joined or this task scope is closed
1187+
* @throws FailedException if the <i>outcome</i> is an exception, thrown with the
1188+
* exception from {@link Joiner#result()} as the cause
1189+
* @throws TimeoutException if a timeout is set and the timeout expires before or
1190+
* while waiting
11161191
* @throws InterruptedException if interrupted while waiting
11171192
* @since 24
11181193
*/
1119-
public R join() throws ExecutionException, InterruptedException {
1194+
public R join() throws InterruptedException {
11201195
ensureOwner();
1121-
ensureOpen();
1196+
ensureNotJoined();
11221197

1123-
if (!joined) {
1124-
// owner has attempted to join
1125-
needToJoin = false;
1198+
// join started
1199+
state = ST_JOIN_STARTED;
11261200

1127-
// wait for all subtasks, execution to be cancelled, or interrupt
1128-
flock.awaitAll();
1129-
1130-
// subtasks are done or execution is cancelled
1131-
joined = true;
1132-
}
1201+
// wait for all subtasks, execution to be cancelled, or interrupt
1202+
flock.awaitAll();
11331203

11341204
// throw if timeout expired
11351205
if (timeoutExpired) {
1136-
throw new ExecutionException(new TimeoutException());
1206+
throw new TimeoutException();
11371207
}
11381208
cancelTimeout();
11391209

1210+
// all subtasks completed or cancelled
1211+
state = ST_JOIN_COMPLETED;
1212+
11401213
// invoke joiner to get result
11411214
try {
11421215
return joiner.result();
11431216
} catch (Throwable e) {
1144-
throw new ExecutionException(e);
1217+
throw new FailedException(e);
11451218
}
11461219
}
11471220

@@ -1244,12 +1317,13 @@ public boolean isCancelled() {
12441317
@Override
12451318
public void close() {
12461319
ensureOwner();
1247-
if (closed) {
1320+
int s = state;
1321+
if (s == ST_CLOSED) {
12481322
return;
12491323
}
12501324

1251-
// cancel execution if not already joined
1252-
if (!joined) {
1325+
// cancel execution if join did not complete
1326+
if (s < ST_JOIN_COMPLETED) {
12531327
cancelExecution();
12541328
cancelTimeout();
12551329
}
@@ -1258,13 +1332,12 @@ public void close() {
12581332
try {
12591333
flock.close();
12601334
} finally {
1261-
closed = true;
1335+
state = ST_CLOSED;
12621336
}
12631337

12641338
// throw ISE if the owner didn't join after forking
1265-
if (needToJoin) {
1266-
needToJoin = false;
1267-
throw new IllegalStateException("Owner did not join");
1339+
if (s == ST_FORKED) {
1340+
throw new IllegalStateException("Owner did not join after forking");
12681341
}
12691342
}
12701343

‎test/jdk/java/util/concurrent/StructuredTaskScope/StructuredTaskScopeTest.java

+39-61
Original file line numberDiff line numberDiff line change
@@ -45,16 +45,16 @@
4545
import java.util.concurrent.ConcurrentHashMap;
4646
import java.util.concurrent.CountDownLatch;
4747
import java.util.concurrent.Executors;
48-
import java.util.concurrent.ExecutionException;
4948
import java.util.concurrent.Future;
5049
import java.util.concurrent.LinkedTransferQueue;
5150
import java.util.concurrent.ThreadFactory;
52-
import java.util.concurrent.TimeoutException;
5351
import java.util.concurrent.TimeUnit;
5452
import java.util.concurrent.RejectedExecutionException;
5553
import java.util.concurrent.ScheduledExecutorService;
5654
import java.util.concurrent.StructuredTaskScope;
55+
import java.util.concurrent.StructuredTaskScope.TimeoutException;
5756
import java.util.concurrent.StructuredTaskScope.Config;
57+
import java.util.concurrent.StructuredTaskScope.FailedException;
5858
import java.util.concurrent.StructuredTaskScope.Joiner;
5959
import java.util.concurrent.StructuredTaskScope.Subtask;
6060
import java.util.concurrent.StructureViolationException;
@@ -248,14 +248,8 @@ void testForkAfterJoinThrows(ThreadFactory factory) throws Exception {
248248
Thread.currentThread().interrupt();
249249
assertThrows(InterruptedException.class, scope::join);
250250

251-
// allow subtask1 to finish
252-
latch.countDown();
253-
254-
// continue to fork
255-
var subtask2 = scope.fork(() -> "bar");
256-
scope.join();
257-
assertEquals("foo", subtask1.get());
258-
assertEquals("bar", subtask2.get());
251+
// fork should throw
252+
assertThrows(IllegalStateException.class, () -> scope.fork(() -> "bar"));
259253
}
260254
}
261255

@@ -384,38 +378,42 @@ void testJoinWithRemainingSubtasks(ThreadFactory factory) throws Exception {
384378
}
385379

386380
/**
387-
* Test repeated calls to join when there is a result.
381+
* Test join after join completed with a result.
388382
*/
389383
@Test
390384
void testJoinAfterJoin1() throws Exception {
391385
var results = new LinkedTransferQueue<>(List.of("foo", "bar", "baz"));
392386
Joiner<Object, String> joiner = results::take;
393387
try (var scope = StructuredTaskScope.open(joiner)) {
394388
scope.fork(() -> "foo");
395-
396-
// each call to join should invoke Joiner::result
397389
assertEquals("foo", scope.join());
398-
assertEquals("bar", scope.join());
399-
assertEquals("baz", scope.join());
390+
391+
// join already called
392+
for (int i = 0 ; i < 3; i++) {
393+
assertThrows(IllegalStateException.class, scope::join);
394+
}
400395
}
401396
}
402397

403398
/**
404-
* Test repeated calls to join when there is an exception.
399+
* Test join after join completed with an exception.
405400
*/
406401
@Test
407402
void testJoinAfterJoin2() throws Exception {
408403
try (var scope = StructuredTaskScope.open(Joiner.anySuccessfulResultOrThrow())) {
409404
scope.fork(() -> { throw new FooException(); });
405+
Throwable ex = assertThrows(FailedException.class, scope::join);
406+
assertTrue(ex.getCause() instanceof FooException);
407+
408+
// join already called
410409
for (int i = 0 ; i < 3; i++) {
411-
Throwable ex = assertThrows(ExecutionException.class, scope::join);
412-
assertTrue(ex.getCause() instanceof FooException);
410+
assertThrows(IllegalStateException.class, scope::join);
413411
}
414412
}
415413
}
416414

417415
/**
418-
* Test repeated calls to join when scope cancelled due to timeout.
416+
* Test join after join completed with a timeout.
419417
*/
420418
@Test
421419
void testJoinAfterJoin3() throws Exception {
@@ -425,9 +423,11 @@ void testJoinAfterJoin3() throws Exception {
425423
while (!scope.isCancelled()) {
426424
Thread.sleep(20);
427425
}
426+
assertThrows(TimeoutException.class, scope::join);
427+
428+
// join already called
428429
for (int i = 0 ; i < 3; i++) {
429-
Throwable ex = assertThrows(ExecutionException.class, scope::join);
430-
assertTrue(ex.getCause() instanceof TimeoutException);
430+
assertThrows(IllegalStateException.class, scope::join);
431431
}
432432
}
433433
}
@@ -469,10 +469,8 @@ void testInterruptJoin1(ThreadFactory factory) throws Exception {
469469
try (var scope = StructuredTaskScope.open(Joiner.awaitAll(),
470470
cf -> cf.withThreadFactory(factory))) {
471471

472-
var latch = new CountDownLatch(1);
473-
474472
Subtask<String> subtask = scope.fork(() -> {
475-
latch.await();
473+
Thread.sleep(60_000);
476474
return "foo";
477475
});
478476

@@ -483,14 +481,7 @@ void testInterruptJoin1(ThreadFactory factory) throws Exception {
483481
fail("join did not throw");
484482
} catch (InterruptedException expected) {
485483
assertFalse(Thread.interrupted()); // interrupt status should be cleared
486-
} finally {
487-
// let task continue
488-
latch.countDown();
489484
}
490-
491-
// join should complete
492-
scope.join();
493-
assertEquals("foo", subtask.get());
494485
}
495486
}
496487

@@ -505,7 +496,7 @@ void testInterruptJoin2(ThreadFactory factory) throws Exception {
505496

506497
var latch = new CountDownLatch(1);
507498
Subtask<String> subtask = scope.fork(() -> {
508-
latch.await();
499+
Thread.sleep(60_000);
509500
return "foo";
510501
});
511502

@@ -516,14 +507,7 @@ void testInterruptJoin2(ThreadFactory factory) throws Exception {
516507
fail("join did not throw");
517508
} catch (InterruptedException expected) {
518509
assertFalse(Thread.interrupted()); // interrupt status should be clear
519-
} finally {
520-
// let task continue
521-
latch.countDown();
522510
}
523-
524-
// join should complete
525-
scope.join();
526-
assertEquals("foo", subtask.get());
527511
}
528512
}
529513

@@ -607,12 +591,7 @@ void testJoinWithTimeout2(ThreadFactory factory) throws Exception {
607591
return null;
608592
});
609593

610-
try {
611-
scope.join();
612-
fail();
613-
} catch (ExecutionException e) {
614-
assertTrue(e.getCause() instanceof TimeoutException);
615-
}
594+
assertThrows(TimeoutException.class, scope::join);
616595
expectDuration(startMillis, /*min*/1900, /*max*/20_000);
617596

618597
assertTrue(scope.isCancelled());
@@ -635,12 +614,8 @@ void testJoinWithTimeout3(ThreadFactory factory) throws Exception {
635614
return null;
636615
});
637616

638-
try {
639-
scope.join();
640-
fail();
641-
} catch (ExecutionException e) {
642-
assertTrue(e.getCause() instanceof TimeoutException);
643-
}
617+
assertThrows(TimeoutException.class, scope::join);
618+
644619
assertTrue(scope.isCancelled());
645620
assertEquals(Subtask.State.UNAVAILABLE, subtask.state());
646621
}
@@ -750,12 +725,8 @@ void testTimeoutInterruptsThreads(ThreadFactory factory) throws Exception {
750725
interrupted.await();
751726
}
752727

753-
try {
754-
scope.join();
755-
fail();
756-
} catch (ExecutionException e) {
757-
assertTrue(e.getCause() instanceof TimeoutException);
758-
}
728+
assertThrows(TimeoutException.class, scope::join);
729+
759730
assertEquals(Subtask.State.UNAVAILABLE, subtask.state());
760731
}
761732
}
@@ -782,8 +753,15 @@ void testCloseWithoutJoin2(ThreadFactory factory) {
782753
Thread.sleep(Duration.ofDays(1));
783754
return null;
784755
});
756+
757+
// first call to close should throw
785758
assertThrows(IllegalStateException.class, scope::close);
786759

760+
// subsequent calls to close should not throw
761+
for (int i = 0; i < 3; i++) {
762+
scope.close();
763+
}
764+
787765
// subtask result/exception not available
788766
assertEquals(Subtask.State.UNAVAILABLE, subtask.state());
789767
assertThrows(IllegalStateException.class, subtask::get);
@@ -1280,7 +1258,7 @@ void testAllSuccessfulOrThrow3(ThreadFactory factory) throws Throwable {
12801258
scope.fork(() -> { throw new FooException(); });
12811259
try {
12821260
scope.join();
1283-
} catch (ExecutionException e) {
1261+
} catch (FailedException e) {
12841262
assertTrue(e.getCause() instanceof FooException);
12851263
}
12861264
}
@@ -1294,7 +1272,7 @@ void testAnySuccessfulResultOrThrow1() throws Exception {
12941272
try (var scope = StructuredTaskScope.open(Joiner.anySuccessfulResultOrThrow())) {
12951273
try {
12961274
scope.join();
1297-
} catch (ExecutionException e) {
1275+
} catch (FailedException e) {
12981276
assertTrue(e.getCause() instanceof NoSuchElementException);
12991277
}
13001278
}
@@ -1354,7 +1332,7 @@ void testAnySuccessfulResultOrThrow5(ThreadFactory factory) throws Exception {
13541332
try (var scope = StructuredTaskScope.open(Joiner.anySuccessfulResultOrThrow(),
13551333
cf -> cf.withThreadFactory(factory))) {
13561334
scope.fork(() -> { throw new FooException(); });
1357-
Throwable ex = assertThrows(ExecutionException.class, scope::join);
1335+
Throwable ex = assertThrows(FailedException.class, scope::join);
13581336
assertTrue(ex.getCause() instanceof FooException);
13591337
}
13601338
}
@@ -1400,7 +1378,7 @@ void testAwaitSuccessfulOrThrow3(ThreadFactory factory) throws Throwable {
14001378
scope.fork(() -> { throw new FooException(); });
14011379
try {
14021380
scope.join();
1403-
} catch (ExecutionException e) {
1381+
} catch (FailedException e) {
14041382
assertTrue(e.getCause() instanceof FooException);
14051383
}
14061384
}

0 commit comments

Comments
 (0)
Please sign in to comment.