Skip to content

Commit 326dbb1

Browse files
author
Viktor Klang
committedJun 5, 2024
8312436: CompletableFuture never completes when 'Throwable.toString()' method throws Exception
Reviewed-by: alanb
1 parent 9a8096f commit 326dbb1

File tree

2 files changed

+61
-11
lines changed

2 files changed

+61
-11
lines changed
 

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

+52-8
Original file line numberDiff line numberDiff line change
@@ -306,13 +306,57 @@ final boolean completeValue(T t) {
306306
return RESULT.compareAndSet(this, null, (t == null) ? NIL : t);
307307
}
308308

309+
static CompletionException wrapInCompletionException(Throwable t) {
310+
if (t == null)
311+
return new CompletionException();
312+
313+
String message;
314+
Throwable suppressed;
315+
try {
316+
message = t.toString();
317+
suppressed = null;
318+
} catch (Throwable unknown) {
319+
message = "";
320+
suppressed = unknown;
321+
}
322+
323+
final CompletionException wrapping = new CompletionException(message, t);
324+
325+
if (suppressed != null)
326+
wrapping.addSuppressed(suppressed);
327+
328+
return wrapping;
329+
}
330+
331+
static ExecutionException wrapInExecutionException(Throwable t) {
332+
if (t == null)
333+
return new ExecutionException();
334+
335+
String message;
336+
Throwable suppressed;
337+
try {
338+
message = t.toString();
339+
suppressed = null;
340+
} catch (Throwable unknown) {
341+
message = "";
342+
suppressed = unknown;
343+
}
344+
345+
final ExecutionException wrapping = new ExecutionException(message, t);
346+
347+
if (suppressed != null)
348+
wrapping.addSuppressed(suppressed);
349+
350+
return wrapping;
351+
}
352+
309353
/**
310354
* Returns the encoding of the given (non-null) exception as a
311355
* wrapped CompletionException unless it is one already.
312356
*/
313357
static AltResult encodeThrowable(Throwable x) {
314358
return new AltResult((x instanceof CompletionException) ? x :
315-
new CompletionException(x));
359+
wrapInCompletionException(x));
316360
}
317361

318362
/** Completes with an exceptional result, unless already completed. */
@@ -329,7 +373,7 @@ final boolean completeThrowable(Throwable x) {
329373
*/
330374
static Object encodeThrowable(Throwable x, Object r) {
331375
if (!(x instanceof CompletionException))
332-
x = new CompletionException(x);
376+
x = wrapInCompletionException(x);
333377
else if (r instanceof AltResult && x == ((AltResult)r).ex)
334378
return r;
335379
return new AltResult(x);
@@ -365,7 +409,7 @@ static Object encodeRelay(Object r) {
365409
if (r instanceof AltResult
366410
&& (x = ((AltResult)r).ex) != null
367411
&& !(x instanceof CompletionException))
368-
r = new AltResult(new CompletionException(x));
412+
r = new AltResult(wrapInCompletionException(x));
369413
return r;
370414
}
371415

@@ -393,7 +437,7 @@ private static Object reportGet(Object r, String details)
393437
if ((x instanceof CompletionException) &&
394438
(cause = x.getCause()) != null)
395439
x = cause;
396-
throw new ExecutionException(x);
440+
throw wrapInExecutionException(x);
397441
}
398442
return r;
399443
}
@@ -410,7 +454,7 @@ private static Object reportJoin(Object r, String details) {
410454
throw new CancellationException(details, (CancellationException)x);
411455
if (x instanceof CompletionException)
412456
throw (CompletionException)x;
413-
throw new CompletionException(x);
457+
throw wrapInCompletionException(x);
414458
}
415459
return r;
416460
}
@@ -2605,8 +2649,8 @@ public int getNumberOfDependents() {
26052649
/**
26062650
* Returns a string identifying this CompletableFuture, as well as
26072651
* its completion state. The state, in brackets, contains the
2608-
* String {@code "Completed Normally"} or the String {@code
2609-
* "Completed Exceptionally"}, or the String {@code "Not
2652+
* String {@code "Completed normally"} or the String {@code
2653+
* "Completed exceptionally"}, or the String {@code "Not
26102654
* completed"} followed by the number of CompletableFutures
26112655
* dependent upon its completion, if any.
26122656
*
@@ -2623,7 +2667,7 @@ public String toString() {
26232667
? "[Not completed]"
26242668
: "[Not completed, " + count + " dependents]")
26252669
: (((r instanceof AltResult) && ((AltResult)r).ex != null)
2626-
? "[Completed exceptionally: " + ((AltResult)r).ex + "]"
2670+
? "[Completed exceptionally: " + ((AltResult)r).ex.getClass().getName() + "]"
26272671
: "[Completed normally]"));
26282672
}
26292673

‎test/jdk/java/util/concurrent/tck/CompletableFutureTest.java

+9-3
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,14 @@ public static Test suite() {
8080
return new TestSuite(CompletableFutureTest.class);
8181
}
8282

83-
static class CFException extends RuntimeException {}
83+
static class CFException extends RuntimeException {
84+
// This makes sure that CompletableFuture still behaves appropriately
85+
// even if thrown exceptions end up throwing exceptions from their String
86+
// representations.
87+
@Override public String getMessage() {
88+
throw new IllegalStateException("malformed");
89+
}
90+
}
8491

8592
void checkIncomplete(CompletableFuture<?> f) {
8693
assertFalse(f.isDone());
@@ -272,8 +279,8 @@ public void testComplete() {
272279
*/
273280
public void testCompleteExceptionally() {
274281
CompletableFuture<Item> f = new CompletableFuture<>();
275-
CFException ex = new CFException();
276282
checkIncomplete(f);
283+
CFException ex = new CFException();
277284
f.completeExceptionally(ex);
278285
checkCompletedExceptionally(f, ex);
279286
}
@@ -5142,5 +5149,4 @@ public void testDefaultExceptionallyComposeAsyncExecutor_actionFailed() {
51425149
checkCompletedWithWrappedException(g.toCompletableFuture(), r.ex);
51435150
r.assertInvoked();
51445151
}}
5145-
51465152
}

0 commit comments

Comments
 (0)
Please sign in to comment.