ViewVC Help
View File | Revision Log | Show Annotations | Download File | Root Listing
root/jsr166/jsr166/src/test/tck/CompletableFutureTest.java
(Generate patch)

Comparing jsr166/src/test/tck/CompletableFutureTest.java (file contents):
Revision 1.36 by jsr166, Sun Jun 1 23:12:45 2014 UTC vs.
Revision 1.45 by jsr166, Mon Jun 2 06:07:34 2014 UTC

# Line 318 | Line 318 | public class CompletableFutureTest exten
318  
319      // Choose non-commutative actions for better coverage
320  
321 <    // A non-commutative function that handles null values as well
322 <    public static int subtract(Integer x, Integer y) {
323 <        return ((x == null) ? 42 : x.intValue())
321 >    // A non-commutative function that handles and produces null values as well.
322 >    static Integer subtract(Integer x, Integer y) {
323 >        return (x == null && y == null) ? null :
324 >            ((x == null) ? 42 : x.intValue())
325              - ((y == null) ? 99 : y.intValue());
326      }
327  
328 +    // A function that handles and produces null values as well.
329 +    static Integer inc(Integer x) {
330 +        return (x == null) ? null : x + 1;
331 +    }
332 +
333      static final Supplier<Integer> supplyOne =
334          () -> Integer.valueOf(1);
335      static final Function<Integer, Integer> inc =
# Line 331 | Line 337 | public class CompletableFutureTest exten
337      static final BiFunction<Integer, Integer, Integer> subtract =
338          (Integer x, Integer y) -> subtract(x, y);
339      static final class IncAction implements Consumer<Integer> {
340 <        int value;
341 <        public void accept(Integer x) { value = x.intValue() + 1; }
340 >        int invocationCount = 0;
341 >        Integer value;
342 >        public void accept(Integer x) {
343 >            invocationCount++;
344 >            value = inc(x);
345 >        }
346 >    }
347 >    static final class IncFunction implements Function<Integer,Integer> {
348 >        int invocationCount = 0;
349 >        Integer value;
350 >        public Integer apply(Integer x) {
351 >            invocationCount++;
352 >            return value = inc(x);
353 >        }
354      }
355      static final class SubtractAction implements BiConsumer<Integer, Integer> {
356 <        volatile int invocationCount = 0;
357 <        int value;
356 >        int invocationCount = 0;
357 >        Integer value;
358          // Check this action was invoked exactly once when result is computed.
341        public boolean ran() { return invocationCount == 1; }
359          public void accept(Integer x, Integer y) {
360              invocationCount++;
361              value = subtract(x, y);
362          }
363      }
364      static final class SubtractFunction implements BiFunction<Integer, Integer, Integer> {
365 <        volatile int invocationCount = 0;
366 <        int value;
365 >        int invocationCount = 0;
366 >        Integer value;
367          // Check this action was invoked exactly once when result is computed.
351        public boolean ran() { return invocationCount == 1; }
368          public Integer apply(Integer x, Integer y) {
369              invocationCount++;
370 <            return subtract(x, y);
370 >            return value = subtract(x, y);
371          }
372      }
373      static final class Noop implements Runnable {
374 <        boolean ran;
375 <        public void run() { ran = true; }
374 >        int invocationCount = 0;
375 >        public void run() {
376 >            invocationCount++;
377 >        }
378      }
379  
380      static final class FailingSupplier implements Supplier<Integer> {
381 <        boolean ran;
382 <        public Integer get() { ran = true; throw new CFException(); }
381 >        int invocationCount = 0;
382 >        public Integer get() {
383 >            invocationCount++;
384 >            throw new CFException();
385 >        }
386      }
387      static final class FailingConsumer implements Consumer<Integer> {
388 <        boolean ran;
389 <        public void accept(Integer x) { ran = true; throw new CFException(); }
388 >        int invocationCount = 0;
389 >        public void accept(Integer x) {
390 >            invocationCount++;
391 >            throw new CFException();
392 >        }
393      }
394      static final class FailingBiConsumer implements BiConsumer<Integer, Integer> {
395 <        boolean ran;
396 <        public void accept(Integer x, Integer y) { ran = true; throw new CFException(); }
395 >        int invocationCount = 0;
396 >        public void accept(Integer x, Integer y) {
397 >            invocationCount++;
398 >            throw new CFException();
399 >        }
400      }
401      static final class FailingFunction implements Function<Integer, Integer> {
402 <        boolean ran;
403 <        public Integer apply(Integer x) { ran = true; throw new CFException(); }
402 >        int invocationCount = 0;
403 >        public Integer apply(Integer x) {
404 >            invocationCount++;
405 >            throw new CFException();
406 >        }
407      }
408      static final class FailingBiFunction implements BiFunction<Integer, Integer, Integer> {
409 <        boolean ran;
410 <        public Integer apply(Integer x, Integer y) { ran = true; throw new CFException(); }
409 >        int invocationCount = 0;
410 >        public Integer apply(Integer x, Integer y) {
411 >            invocationCount++;
412 >            throw new CFException();
413 >        }
414      }
415      static final class FailingNoop implements Runnable {
416 <        boolean ran;
417 <        public void run() { ran = true; throw new CFException(); }
416 >        int invocationCount = 0;
417 >        public void run() {
418 >            invocationCount++;
419 >            throw new CFException();
420 >        }
421      }
422  
423      static final class CompletableFutureInc
424          implements Function<Integer, CompletableFuture<Integer>> {
425 <        boolean ran;
425 >        int invocationCount = 0;
426          public CompletableFuture<Integer> apply(Integer x) {
427 <            ran = true;
427 >            invocationCount++;
428              CompletableFuture<Integer> f = new CompletableFuture<>();
429 <            f.complete(Integer.valueOf(x.intValue() + 1));
429 >            f.complete(inc(x));
430              return f;
431          }
432      }
433  
434      static final class FailingCompletableFutureFunction
435          implements Function<Integer, CompletableFuture<Integer>> {
436 <        boolean ran;
436 >        int invocationCount = 0;
437          public CompletableFuture<Integer> apply(Integer x) {
438 <            ran = true; throw new CFException();
438 >            invocationCount++;
439 >            throw new CFException();
440          }
441      }
442  
# Line 418 | Line 455 | public class CompletableFutureTest exten
455      }
456  
457      static final class IntegerHandler implements BiFunction<Integer, Throwable, Integer> {
458 <        boolean ran;
458 >        int invocationCount = 0;
459          public Integer apply(Integer x, Throwable t) {
460 <            ran = true;
460 >            invocationCount++;
461              return (t == null) ? two : three;
462          }
463      }
# Line 447 | Line 484 | public class CompletableFutureTest exten
484                   BiFunction<? super T,? super U,? extends V> a) {
485                  return f.thenCombine(g, a);
486              }
487 +            public <T,U> CompletableFuture<U> applyToEither
488 +                (CompletableFuture<T> f,
489 +                 CompletionStage<? extends T> g,
490 +                 Function<? super T,U> a) {
491 +                return f.applyToEither(g, a);
492 +            }
493 +            public <T> CompletableFuture<Void> acceptEither
494 +                (CompletableFuture<T> f,
495 +                 CompletionStage<? extends T> g,
496 +                 Consumer<? super T> a) {
497 +                return f.acceptEither(g, a);
498 +            }
499 +            public <T> CompletableFuture<Void> runAfterEither
500 +                (CompletableFuture<T> f,
501 +                 CompletionStage<?> g,
502 +                 java.lang.Runnable a) {
503 +                return f.runAfterEither(g, a);
504 +            }
505 +            public <T,U> CompletableFuture<U> thenCompose
506 +                (CompletableFuture<T> f,
507 +                 Function<? super T,? extends CompletionStage<U>> a) {
508 +                return f.thenCompose(a);
509 +            }
510 +            public <T> CompletableFuture<T> whenComplete
511 +                (CompletableFuture<T> f,
512 +                 BiConsumer<? super T,? super Throwable> a) {
513 +                return f.whenComplete(a);
514 +            }
515          },
516  
452 //             /** Experimental way to do more testing */
453 //         REVERSE_DEFAULT {
454 //             public <T,U> CompletableFuture<Void> runAfterBoth
455 //                 (CompletableFuture<T> f, CompletableFuture<U> g, Runnable a) {
456 //                 return g.runAfterBoth(f, a);
457 //             }
458 //             public <T,U> CompletableFuture<Void> thenAcceptBoth
459 //                 (CompletableFuture<T> f,
460 //                  CompletionStage<? extends U> g,
461 //                  BiConsumer<? super T,? super U> a) {
462 //                 return DEFAULT.thenAcceptBoth(f, g, a);
463 //             }
464 //         },
465
517          DEFAULT_ASYNC {
518              public <T,U> CompletableFuture<Void> runAfterBoth
519                  (CompletableFuture<T> f, CompletableFuture<U> g, Runnable a) {
# Line 480 | Line 531 | public class CompletableFutureTest exten
531                   BiFunction<? super T,? super U,? extends V> a) {
532                  return f.thenCombineAsync(g, a);
533              }
534 +            public <T,U> CompletableFuture<U> applyToEither
535 +                (CompletableFuture<T> f,
536 +                 CompletionStage<? extends T> g,
537 +                 Function<? super T,U> a) {
538 +                return f.applyToEitherAsync(g, a);
539 +            }
540 +            public <T> CompletableFuture<Void> acceptEither
541 +                (CompletableFuture<T> f,
542 +                 CompletionStage<? extends T> g,
543 +                 Consumer<? super T> a) {
544 +                return f.acceptEitherAsync(g, a);
545 +            }
546 +            public <T> CompletableFuture<Void> runAfterEither
547 +                (CompletableFuture<T> f,
548 +                 CompletionStage<?> g,
549 +                 java.lang.Runnable a) {
550 +                return f.runAfterEitherAsync(g, a);
551 +            }
552 +            public <T,U> CompletableFuture<U> thenCompose
553 +                (CompletableFuture<T> f,
554 +                 Function<? super T,? extends CompletionStage<U>> a) {
555 +                return f.thenComposeAsync(a);
556 +            }
557 +            public <T> CompletableFuture<T> whenComplete
558 +                (CompletableFuture<T> f,
559 +                 BiConsumer<? super T,? super Throwable> a) {
560 +                return f.whenCompleteAsync(a);
561 +            }
562          },
563  
485 //         REVERSE_DEFAULT_ASYNC {
486 //             public <T,U> CompletableFuture<Void> runAfterBoth
487 //                 (CompletableFuture<T> f, CompletableFuture<U> g, Runnable a) {
488 //                 return f.runAfterBothAsync(g, a);
489 //             }
490 //             public <T,U> CompletableFuture<Void> thenAcceptBoth
491 //                 (CompletableFuture<T> f,
492 //                  CompletionStage<? extends U> g,
493 //                  BiConsumer<? super T,? super U> a) {
494 //                 return DEFAULT_ASYNC.thenAcceptBoth(f, g, a);
495 //             }
496 //         },
497
564          EXECUTOR {
565              public <T,U> CompletableFuture<Void> runAfterBoth
566                  (CompletableFuture<T> f, CompletableFuture<U> g, Runnable a) {
# Line 512 | Line 578 | public class CompletableFutureTest exten
578                   BiFunction<? super T,? super U,? extends V> a) {
579                  return f.thenCombineAsync(g, a, new ThreadExecutor());
580              }
581 +            public <T,U> CompletableFuture<U> applyToEither
582 +                (CompletableFuture<T> f,
583 +                 CompletionStage<? extends T> g,
584 +                 Function<? super T,U> a) {
585 +                return f.applyToEitherAsync(g, a, new ThreadExecutor());
586 +            }
587 +            public <T> CompletableFuture<Void> acceptEither
588 +                (CompletableFuture<T> f,
589 +                 CompletionStage<? extends T> g,
590 +                 Consumer<? super T> a) {
591 +                return f.acceptEitherAsync(g, a, new ThreadExecutor());
592 +            }
593 +            public <T> CompletableFuture<Void> runAfterEither
594 +                (CompletableFuture<T> f,
595 +                 CompletionStage<?> g,
596 +                 java.lang.Runnable a) {
597 +                return f.runAfterEitherAsync(g, a, new ThreadExecutor());
598 +            }
599 +            public <T,U> CompletableFuture<U> thenCompose
600 +                (CompletableFuture<T> f,
601 +                 Function<? super T,? extends CompletionStage<U>> a) {
602 +                return f.thenComposeAsync(a, new ThreadExecutor());
603 +            }
604 +            public <T> CompletableFuture<T> whenComplete
605 +                (CompletableFuture<T> f,
606 +                 BiConsumer<? super T,? super Throwable> a) {
607 +                return f.whenCompleteAsync(a, new ThreadExecutor());
608 +            }
609          };
610  
611          public abstract <T,U> CompletableFuture<Void> runAfterBoth
# Line 524 | Line 618 | public class CompletableFutureTest exten
618              (CompletableFuture<T> f,
619               CompletionStage<? extends U> g,
620               BiFunction<? super T,? super U,? extends V> a);
621 +        public abstract <T,U> CompletableFuture<U> applyToEither
622 +            (CompletableFuture<T> f,
623 +             CompletionStage<? extends T> g,
624 +             Function<? super T,U> a);
625 +        public abstract <T> CompletableFuture<Void> acceptEither
626 +            (CompletableFuture<T> f,
627 +             CompletionStage<? extends T> g,
628 +             Consumer<? super T> a);
629 +        public abstract <T> CompletableFuture<Void> runAfterEither
630 +            (CompletableFuture<T> f,
631 +             CompletionStage<?> g,
632 +             java.lang.Runnable a);
633 +        public abstract <T,U> CompletableFuture<U> thenCompose
634 +            (CompletableFuture<T> f,
635 +             Function<? super T,? extends CompletionStage<U>> a);
636 +        public abstract <T> CompletableFuture<T> whenComplete
637 +            (CompletableFuture<T> f,
638 +             BiConsumer<? super T,? super Throwable> a);
639 +
640 +
641      }
642  
643      /**
# Line 555 | Line 669 | public class CompletableFutureTest exten
669          f = new CompletableFuture<>();
670          f.completeExceptionally(new CFException());
671          g = f.handle(r = new IntegerHandler());
672 <        assertTrue(r.ran);
672 >        assertEquals(1, r.invocationCount);
673 >        assertEquals(1, r.invocationCount);
674          checkCompletedNormally(g, three);
675  
676          f = new CompletableFuture<>();
677          g = f.handle(r = new IntegerHandler());
678 <        assertFalse(r.ran);
678 >        assertEquals(0, r.invocationCount);
679          f.completeExceptionally(new CFException());
680          checkCompletedNormally(g, three);
681 <        assertTrue(r.ran);
681 >        assertEquals(1, r.invocationCount);
682  
683          f = new CompletableFuture<>();
684          f.complete(one);
685          g = f.handle(r = new IntegerHandler());
686 <        assertTrue(r.ran);
686 >        assertEquals(1, r.invocationCount);
687          checkCompletedNormally(g, two);
688  
689          f = new CompletableFuture<>();
690          g = f.handle(r = new IntegerHandler());
691 <        assertFalse(r.ran);
691 >        assertEquals(0, r.invocationCount);
692          f.complete(one);
693 <        assertTrue(r.ran);
693 >        assertEquals(1, r.invocationCount);
694          checkCompletedNormally(g, two);
695      }
696  
# Line 586 | Line 701 | public class CompletableFutureTest exten
701          Noop r = new Noop();
702          CompletableFuture<Void> f = CompletableFuture.runAsync(r);
703          assertNull(f.join());
704 <        assertTrue(r.ran);
704 >        assertEquals(1, r.invocationCount);
705          checkCompletedNormally(f, null);
706      }
707  
# Line 598 | Line 713 | public class CompletableFutureTest exten
713          ThreadExecutor exec = new ThreadExecutor();
714          CompletableFuture<Void> f = CompletableFuture.runAsync(r, exec);
715          assertNull(f.join());
716 <        assertTrue(r.ran);
716 >        assertEquals(1, r.invocationCount);
717          checkCompletedNormally(f, null);
718          assertEquals(1, exec.count.get());
719      }
# Line 610 | Line 725 | public class CompletableFutureTest exten
725          FailingNoop r = new FailingNoop();
726          CompletableFuture<Void> f = CompletableFuture.runAsync(r);
727          checkCompletedWithWrappedCFException(f);
728 <        assertTrue(r.ran);
728 >        assertEquals(1, r.invocationCount);
729      }
730  
731      /**
# Line 640 | Line 755 | public class CompletableFutureTest exten
755          FailingSupplier r = new FailingSupplier();
756          CompletableFuture<Integer> f = CompletableFuture.supplyAsync(r);
757          checkCompletedWithWrappedCFException(f);
758 <        assertTrue(r.ran);
758 >        assertEquals(1, r.invocationCount);
759      }
760  
761      // seq completion methods
# Line 657 | Line 772 | public class CompletableFutureTest exten
772          g = f.thenRun(r = new Noop());
773          f.complete(null);
774          checkCompletedNormally(g, null);
775 <        assertTrue(r.ran);
775 >        assertEquals(1, r.invocationCount);
776  
777          f = new CompletableFuture<>();
778          f.complete(null);
779          g = f.thenRun(r = new Noop());
780          checkCompletedNormally(g, null);
781 <        assertTrue(r.ran);
781 >        assertEquals(1, r.invocationCount);
782      }
783  
784      /**
# Line 679 | Line 794 | public class CompletableFutureTest exten
794          g = f.thenRun(r = new Noop());
795          f.completeExceptionally(new CFException());
796          checkCompletedWithWrappedCFException(g);
797 <        assertFalse(r.ran);
797 >        assertEquals(0, r.invocationCount);
798  
799          f = new CompletableFuture<>();
800          f.completeExceptionally(new CFException());
801          g = f.thenRun(r = new Noop());
802          checkCompletedWithWrappedCFException(g);
803 <        assertFalse(r.ran);
803 >        assertEquals(0, r.invocationCount);
804      }
805  
806      /**
# Line 776 | Line 891 | public class CompletableFutureTest exten
891          CompletableFuture<Void> g = f.thenAccept(r);
892          f.complete(one);
893          checkCompletedNormally(g, null);
894 <        assertEquals(r.value, 2);
894 >        assertEquals(r.value, (Integer) 2);
895      }
896  
897      /**
# Line 800 | Line 915 | public class CompletableFutureTest exten
915          CompletableFuture<Void> g = f.thenAccept(r);
916          f.complete(one);
917          checkCompletedWithWrappedCFException(g);
918 <        assertTrue(r.ran);
918 >        assertEquals(1, r.invocationCount);
919      }
920  
921      /**
# Line 819 | Line 934 | public class CompletableFutureTest exten
934       * of sources
935       */
936      public void testThenCombine_normalCompletion1() {
937 +        for (boolean createIncomplete : new boolean[] { true, false })
938 +        for (boolean fFirst : new boolean[] { true, false })
939          for (ExecutionMode m : ExecutionMode.values())
940          for (Integer v1 : new Integer[] { 1, null })
941          for (Integer v2 : new Integer[] { 2, null }) {
# Line 826 | Line 943 | public class CompletableFutureTest exten
943          final CompletableFuture<Integer> f = new CompletableFuture<>();
944          final CompletableFuture<Integer> g = new CompletableFuture<>();
945          final SubtractFunction r = new SubtractFunction();
946 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
947 <
831 <        f.complete(v1);
832 <        checkIncomplete(h);
833 <        assertFalse(r.ran());
834 <        g.complete(v2);
835 <
836 <        checkCompletedNormally(h, subtract(v1, v2));
837 <        checkCompletedNormally(f, v1);
838 <        checkCompletedNormally(g, v2);
839 <        }
840 <    }
841 <
842 <    public void testThenCombine_normalCompletion2() {
843 <        for (ExecutionMode m : ExecutionMode.values())
844 <        for (Integer v1 : new Integer[] { 1, null })
845 <        for (Integer v2 : new Integer[] { 2, null }) {
846 <
847 <        final CompletableFuture<Integer> f = new CompletableFuture<>();
848 <        final CompletableFuture<Integer> g = new CompletableFuture<>();
849 <        final SubtractFunction r = new SubtractFunction();
850 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
851 <
852 <        g.complete(v2);
853 <        checkIncomplete(h);
854 <        assertFalse(r.ran());
855 <        f.complete(v1);
856 <
857 <        checkCompletedNormally(h, subtract(v1, v2));
858 <        checkCompletedNormally(f, v1);
859 <        checkCompletedNormally(g, v2);
860 <        }
861 <    }
862 <
863 <    public void testThenCombine_normalCompletion3() {
864 <        for (ExecutionMode m : ExecutionMode.values())
865 <        for (Integer v1 : new Integer[] { 1, null })
866 <        for (Integer v2 : new Integer[] { 2, null }) {
867 <
868 <        final CompletableFuture<Integer> f = new CompletableFuture<>();
869 <        final CompletableFuture<Integer> g = new CompletableFuture<>();
870 <        final SubtractFunction r = new SubtractFunction();
871 <
872 <        g.complete(v2);
873 <        f.complete(v1);
874 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
946 >        CompletableFuture<Integer> h = null;
947 >        if (createIncomplete) h = m.thenCombine(f, g, r);
948  
949 <        checkCompletedNormally(h, subtract(v1, v2));
950 <        checkCompletedNormally(f, v1);
951 <        checkCompletedNormally(g, v2);
952 <        }
953 <    }
954 <
955 <    public void testThenCombine_normalCompletion4() {
956 <        for (ExecutionMode m : ExecutionMode.values())
957 <        for (Integer v1 : new Integer[] { 1, null })
958 <        for (Integer v2 : new Integer[] { 2, null }) {
959 <
887 <        final CompletableFuture<Integer> f = new CompletableFuture<>();
888 <        final CompletableFuture<Integer> g = new CompletableFuture<>();
889 <        final SubtractFunction r = new SubtractFunction();
890 <
891 <        f.complete(v1);
892 <        g.complete(v2);
893 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
949 >        if (fFirst)
950 >            f.complete(v1);
951 >        else
952 >            g.complete(v2);
953 >        if (createIncomplete) checkIncomplete(h);
954 >        assertEquals(0, r.invocationCount);
955 >        if (!fFirst)
956 >            f.complete(v1);
957 >        else
958 >            g.complete(v2);
959 >        if (!createIncomplete) h = m.thenCombine(f, g, r);
960  
961          checkCompletedNormally(h, subtract(v1, v2));
962          checkCompletedNormally(f, v1);
963          checkCompletedNormally(g, v2);
964 +        assertEquals(1, r.invocationCount);
965          }
966      }
967  
# Line 918 | Line 985 | public class CompletableFutureTest exten
985  
986          checkCompletedWithWrappedCFException(h, ex);
987          checkCompletedWithWrappedCFException(f, ex);
988 <        assertFalse(r.ran());
988 >        assertEquals(0, r.invocationCount);
989          checkCompletedNormally(g, v1);
990          }
991      }
# Line 930 | Line 997 | public class CompletableFutureTest exten
997          final CompletableFuture<Integer> f = new CompletableFuture<>();
998          final CompletableFuture<Integer> g = new CompletableFuture<>();
999          final SubtractFunction r = new SubtractFunction();
1000 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, subtract);
1000 >        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
1001          final CFException ex = new CFException();
1002  
1003          g.completeExceptionally(ex);
# Line 939 | Line 1006 | public class CompletableFutureTest exten
1006  
1007          checkCompletedWithWrappedCFException(h, ex);
1008          checkCompletedWithWrappedCFException(g, ex);
1009 <        assertFalse(r.ran());
1009 >        assertEquals(0, r.invocationCount);
1010          checkCompletedNormally(f, v1);
1011          }
1012      }
# Line 959 | Line 1026 | public class CompletableFutureTest exten
1026  
1027          checkCompletedWithWrappedCFException(h, ex);
1028          checkCompletedWithWrappedCFException(g, ex);
1029 <        assertFalse(r.ran());
1029 >        assertEquals(0, r.invocationCount);
1030          checkCompletedNormally(f, v1);
1031          }
1032      }
# Line 975 | Line 1042 | public class CompletableFutureTest exten
1042  
1043          f.completeExceptionally(ex);
1044          g.complete(v1);
1045 <        final CompletableFuture<Integer> h = m.thenCombine(f, g, subtract);
1045 >        final CompletableFuture<Integer> h = m.thenCombine(f, g, r);
1046  
1047          checkCompletedWithWrappedCFException(h, ex);
1048          checkCompletedWithWrappedCFException(f, ex);
1049 <        assertFalse(r.ran());
1049 >        assertEquals(0, r.invocationCount);
1050          checkCompletedNormally(g, v1);
1051          }
1052      }
# Line 1046 | Line 1113 | public class CompletableFutureTest exten
1113  
1114          checkCompletedWithWrappedCancellationException(h);
1115          checkCancelled(f);
1116 <        assertFalse(r.ran());
1116 >        assertEquals(0, r.invocationCount);
1117          checkCompletedNormally(g, v1);
1118          }
1119      }
# Line 1067 | Line 1134 | public class CompletableFutureTest exten
1134  
1135          checkCompletedWithWrappedCancellationException(h);
1136          checkCancelled(g);
1137 <        assertFalse(r.ran());
1137 >        assertEquals(0, r.invocationCount);
1138          checkCompletedNormally(f, v1);
1139          }
1140      }
# Line 1087 | Line 1154 | public class CompletableFutureTest exten
1154  
1155          checkCompletedWithWrappedCancellationException(h);
1156          checkCancelled(g);
1157 <        assertFalse(r.ran());
1157 >        assertEquals(0, r.invocationCount);
1158          checkCompletedNormally(f, v1);
1159          }
1160      }
# Line 1107 | Line 1174 | public class CompletableFutureTest exten
1174  
1175          checkCompletedWithWrappedCancellationException(h);
1176          checkCancelled(f);
1177 <        assertFalse(r.ran());
1177 >        assertEquals(0, r.invocationCount);
1178          checkCompletedNormally(g, v1);
1179          }
1180      }
# Line 1128 | Line 1195 | public class CompletableFutureTest exten
1195  
1196          f.complete(v1);
1197          checkIncomplete(h);
1198 <        assertEquals(r.value, 0);
1198 >        assertEquals(0, r.invocationCount);
1199          g.complete(v2);
1200  
1201          checkCompletedNormally(h, null);
# Line 1150 | Line 1217 | public class CompletableFutureTest exten
1217  
1218          g.complete(v2);
1219          checkIncomplete(h);
1220 <        assertEquals(r.value, 0);
1220 >        assertEquals(0, r.invocationCount);
1221          f.complete(v1);
1222  
1223          checkCompletedNormally(h, null);
# Line 1220 | Line 1287 | public class CompletableFutureTest exten
1287  
1288          checkCompletedWithWrappedCFException(h, ex);
1289          checkCompletedWithWrappedCFException(f, ex);
1290 <        assertFalse(r.ran());
1290 >        assertEquals(0, r.invocationCount);
1291          checkCompletedNormally(g, v1);
1292          }
1293      }
# Line 1241 | Line 1308 | public class CompletableFutureTest exten
1308  
1309          checkCompletedWithWrappedCFException(h, ex);
1310          checkCompletedWithWrappedCFException(g, ex);
1311 <        assertFalse(r.ran());
1311 >        assertEquals(0, r.invocationCount);
1312          checkCompletedNormally(f, v1);
1313          }
1314      }
# Line 1261 | Line 1328 | public class CompletableFutureTest exten
1328  
1329          checkCompletedWithWrappedCFException(h, ex);
1330          checkCompletedWithWrappedCFException(g, ex);
1331 <        assertFalse(r.ran());
1331 >        assertEquals(0, r.invocationCount);
1332          checkCompletedNormally(f, v1);
1333          }
1334      }
# Line 1281 | Line 1348 | public class CompletableFutureTest exten
1348  
1349          checkCompletedWithWrappedCFException(h, ex);
1350          checkCompletedWithWrappedCFException(f, ex);
1351 <        assertFalse(r.ran());
1351 >        assertEquals(0, r.invocationCount);
1352          checkCompletedNormally(g, v1);
1353          }
1354      }
# Line 1348 | Line 1415 | public class CompletableFutureTest exten
1415  
1416          checkCompletedWithWrappedCancellationException(h);
1417          checkCancelled(f);
1418 <        assertFalse(r.ran());
1418 >        assertEquals(0, r.invocationCount);
1419          checkCompletedNormally(g, v1);
1420          }
1421      }
# Line 1369 | Line 1436 | public class CompletableFutureTest exten
1436  
1437          checkCompletedWithWrappedCancellationException(h);
1438          checkCancelled(g);
1439 <        assertFalse(r.ran());
1439 >        assertEquals(0, r.invocationCount);
1440          checkCompletedNormally(f, v1);
1441          }
1442      }
# Line 1389 | Line 1456 | public class CompletableFutureTest exten
1456  
1457          checkCompletedWithWrappedCancellationException(h);
1458          checkCancelled(g);
1459 <        assertFalse(r.ran());
1459 >        assertEquals(0, r.invocationCount);
1460          checkCompletedNormally(f, v1);
1461          }
1462      }
# Line 1409 | Line 1476 | public class CompletableFutureTest exten
1476  
1477          checkCompletedWithWrappedCancellationException(h);
1478          checkCancelled(f);
1479 <        assertFalse(r.ran());
1479 >        assertEquals(0, r.invocationCount);
1480          checkCompletedNormally(g, v1);
1481          }
1482      }
# Line 1430 | Line 1497 | public class CompletableFutureTest exten
1497  
1498          f.complete(v1);
1499          checkIncomplete(h);
1500 <        assertFalse(r.ran);
1500 >        assertEquals(0, r.invocationCount);
1501          g.complete(v2);
1502  
1503          checkCompletedNormally(h, null);
1504 <        assertTrue(r.ran);
1504 >        assertEquals(1, r.invocationCount);
1505          checkCompletedNormally(f, v1);
1506          checkCompletedNormally(g, v2);
1507          }
# Line 1452 | Line 1519 | public class CompletableFutureTest exten
1519  
1520          g.complete(v2);
1521          checkIncomplete(h);
1522 <        assertFalse(r.ran);
1522 >        assertEquals(0, r.invocationCount);
1523          f.complete(v1);
1524  
1525          checkCompletedNormally(h, null);
1526 <        assertTrue(r.ran);
1526 >        assertEquals(1, r.invocationCount);
1527          checkCompletedNormally(f, v1);
1528          checkCompletedNormally(g, v2);
1529          }
# Line 1476 | Line 1543 | public class CompletableFutureTest exten
1543          final CompletableFuture<Void> h = m.runAfterBoth(f, g, r);
1544  
1545          checkCompletedNormally(h, null);
1546 <        assertTrue(r.ran);
1546 >        assertEquals(1, r.invocationCount);
1547          checkCompletedNormally(f, v1);
1548          checkCompletedNormally(g, v2);
1549          }
# Line 1496 | Line 1563 | public class CompletableFutureTest exten
1563          final CompletableFuture<Void> h = m.runAfterBoth(f, g, r);
1564  
1565          checkCompletedNormally(h, null);
1566 <        assertTrue(r.ran);
1566 >        assertEquals(1, r.invocationCount);
1567          checkCompletedNormally(f, v1);
1568          checkCompletedNormally(g, v2);
1569          }
# Line 1522 | Line 1589 | public class CompletableFutureTest exten
1589  
1590          checkCompletedWithWrappedCFException(h, ex);
1591          checkCompletedWithWrappedCFException(f, ex);
1592 <        assertFalse(r.ran);
1592 >        assertEquals(0, r.invocationCount);
1593          checkCompletedNormally(g, v1);
1594          }
1595      }
# Line 1543 | Line 1610 | public class CompletableFutureTest exten
1610  
1611          checkCompletedWithWrappedCFException(h, ex);
1612          checkCompletedWithWrappedCFException(g, ex);
1613 <        assertFalse(r.ran);
1613 >        assertEquals(0, r.invocationCount);
1614          checkCompletedNormally(f, v1);
1615          }
1616      }
# Line 1563 | Line 1630 | public class CompletableFutureTest exten
1630  
1631          checkCompletedWithWrappedCFException(h, ex);
1632          checkCompletedWithWrappedCFException(g, ex);
1633 <        assertFalse(r.ran);
1633 >        assertEquals(0, r.invocationCount);
1634          checkCompletedNormally(f, v1);
1635          }
1636      }
# Line 1583 | Line 1650 | public class CompletableFutureTest exten
1650  
1651          checkCompletedWithWrappedCFException(h, ex);
1652          checkCompletedWithWrappedCFException(f, ex);
1653 <        assertFalse(r.ran);
1653 >        assertEquals(0, r.invocationCount);
1654          checkCompletedNormally(g, v1);
1655          }
1656      }
# Line 1650 | Line 1717 | public class CompletableFutureTest exten
1717  
1718          checkCompletedWithWrappedCancellationException(h);
1719          checkCancelled(f);
1720 <        assertFalse(r.ran);
1720 >        assertEquals(0, r.invocationCount);
1721          checkCompletedNormally(g, v1);
1722          }
1723      }
# Line 1671 | Line 1738 | public class CompletableFutureTest exten
1738  
1739          checkCompletedWithWrappedCancellationException(h);
1740          checkCancelled(g);
1741 <        assertFalse(r.ran);
1741 >        assertEquals(0, r.invocationCount);
1742          checkCompletedNormally(f, v1);
1743          }
1744      }
# Line 1691 | Line 1758 | public class CompletableFutureTest exten
1758  
1759          checkCompletedWithWrappedCancellationException(h);
1760          checkCancelled(g);
1761 <        assertFalse(r.ran);
1761 >        assertEquals(0, r.invocationCount);
1762          checkCompletedNormally(f, v1);
1763          }
1764      }
# Line 1711 | Line 1778 | public class CompletableFutureTest exten
1778  
1779          checkCompletedWithWrappedCancellationException(h);
1780          checkCancelled(f);
1781 <        assertFalse(r.ran);
1781 >        assertEquals(0, r.invocationCount);
1782          checkCompletedNormally(g, v1);
1783          }
1784      }
# Line 1720 | Line 1787 | public class CompletableFutureTest exten
1787       * applyToEither result completes normally after normal completion
1788       * of either source
1789       */
1790 <    public void testApplyToEither() {
1791 <        CompletableFuture<Integer> f = new CompletableFuture<>();
1792 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1793 <        CompletableFuture<Integer> g = f.applyToEither(f2, inc);
1727 <        f.complete(one);
1728 <        checkCompletedNormally(g, two);
1729 <        f2.complete(one);
1730 <        checkCompletedNormally(g, two);
1790 >    public void testApplyToEither_normalCompletion1() {
1791 >        for (ExecutionMode m : ExecutionMode.values())
1792 >        for (Integer v1 : new Integer[] { 1, null })
1793 >        for (Integer v2 : new Integer[] { 2, null }) {
1794  
1795 <        f = new CompletableFuture<>();
1796 <        f.complete(one);
1797 <        f2 = new CompletableFuture<>();
1798 <        g = f.applyToEither(f2, inc);
1799 <        checkCompletedNormally(g, two);
1795 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1796 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1797 >        final IncFunction r = new IncFunction();
1798 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1799 >
1800 >        f.complete(v1);
1801 >        checkCompletedNormally(h, inc(v1));
1802 >        g.complete(v2);
1803 >
1804 >        checkCompletedNormally(f, v1);
1805 >        checkCompletedNormally(g, v2);
1806 >        checkCompletedNormally(h, inc(v1));
1807 >        }
1808 >    }
1809 >
1810 >    public void testApplyToEither_normalCompletion2() {
1811 >        for (ExecutionMode m : ExecutionMode.values())
1812 >        for (Integer v1 : new Integer[] { 1, null })
1813 >        for (Integer v2 : new Integer[] { 2, null }) {
1814 >
1815 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1816 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1817 >        final IncFunction r = new IncFunction();
1818 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1819 >
1820 >        g.complete(v2);
1821 >        checkCompletedNormally(h, inc(v2));
1822 >        f.complete(v1);
1823 >
1824 >        checkCompletedNormally(f, v1);
1825 >        checkCompletedNormally(g, v2);
1826 >        checkCompletedNormally(h, inc(v2));
1827 >        }
1828 >    }
1829 >    public void testApplyToEither_normalCompletion3() {
1830 >        for (ExecutionMode m : ExecutionMode.values())
1831 >        for (Integer v1 : new Integer[] { 1, null })
1832 >        for (Integer v2 : new Integer[] { 2, null }) {
1833 >
1834 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1835 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1836 >        final IncFunction r = new IncFunction();
1837 >
1838 >        f.complete(v1);
1839 >        g.complete(v2);
1840 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1841 >
1842 >        checkCompletedNormally(f, v1);
1843 >        checkCompletedNormally(g, v2);
1844 >
1845 >        // unspecified behavior
1846 >        assertTrue(Objects.equals(h.join(), inc(v1)) ||
1847 >                   Objects.equals(h.join(), inc(v2)));
1848 >        assertEquals(1, r.invocationCount);
1849 >        }
1850      }
1851  
1852      /**
1853       * applyToEither result completes exceptionally after exceptional
1854       * completion of either source
1855       */
1856 <    public void testApplyToEither2() {
1857 <        CompletableFuture<Integer> f = new CompletableFuture<>();
1858 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1746 <        CompletableFuture<Integer> g = f.applyToEither(f2, inc);
1747 <        f.completeExceptionally(new CFException());
1748 <        f2.complete(one);
1749 <        checkCompletedWithWrappedCFException(g);
1856 >    public void testApplyToEither_exceptionalCompletion1() {
1857 >        for (ExecutionMode m : ExecutionMode.values())
1858 >        for (Integer v1 : new Integer[] { 1, null }) {
1859  
1860 <        f = new CompletableFuture<>();
1861 <        f2 = new CompletableFuture<>();
1862 <        f2.completeExceptionally(new CFException());
1863 <        g = f.applyToEither(f2, inc);
1864 <        checkCompletedWithWrappedCFException(g);
1860 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1861 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1862 >        final IncFunction r = new IncFunction();
1863 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1864 >        final CFException ex = new CFException();
1865 >
1866 >        f.completeExceptionally(ex);
1867 >        checkCompletedWithWrappedCFException(h, ex);
1868 >        g.complete(v1);
1869 >
1870 >        assertEquals(0, r.invocationCount);
1871 >        checkCompletedNormally(g, v1);
1872 >        checkCompletedWithWrappedCFException(f, ex);
1873 >        checkCompletedWithWrappedCFException(h, ex);
1874 >        }
1875 >    }
1876 >
1877 >    public void testApplyToEither_exceptionalCompletion2() {
1878 >        for (ExecutionMode m : ExecutionMode.values())
1879 >        for (Integer v1 : new Integer[] { 1, null }) {
1880 >
1881 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1882 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1883 >        final IncFunction r = new IncFunction();
1884 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1885 >        final CFException ex = new CFException();
1886 >
1887 >        g.completeExceptionally(ex);
1888 >        checkCompletedWithWrappedCFException(h, ex);
1889 >        f.complete(v1);
1890 >
1891 >        assertEquals(0, r.invocationCount);
1892 >        checkCompletedNormally(f, v1);
1893 >        checkCompletedWithWrappedCFException(g, ex);
1894 >        checkCompletedWithWrappedCFException(h, ex);
1895 >        }
1896 >    }
1897 >
1898 >    public void testApplyToEither_exceptionalCompletion3() {
1899 >        for (ExecutionMode m : ExecutionMode.values())
1900 >        for (Integer v1 : new Integer[] { 1, null }) {
1901 >
1902 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1903 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1904 >        final IncFunction r = new IncFunction();
1905 >        final CFException ex = new CFException();
1906 >
1907 >        g.completeExceptionally(ex);
1908 >        f.complete(v1);
1909 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1910 >
1911 >        // unspecified behavior
1912 >        Integer v;
1913 >        try {
1914 >            assertEquals(h.join(), inc(v1));
1915 >            assertEquals(1, r.invocationCount);
1916 >        } catch (CompletionException ok) {
1917 >            checkCompletedWithWrappedCFException(h, ex);
1918 >            assertEquals(0, r.invocationCount);
1919 >        }
1920 >
1921 >        checkCompletedWithWrappedCFException(g, ex);
1922 >        checkCompletedNormally(f, v1);
1923 >        }
1924 >    }
1925 >
1926 >    public void testApplyToEither_exceptionalCompletion4() {
1927 >        for (ExecutionMode m : ExecutionMode.values())
1928 >        for (Integer v1 : new Integer[] { 1, null }) {
1929 >
1930 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1931 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1932 >        final IncFunction r = new IncFunction();
1933 >        final CFException ex = new CFException();
1934 >
1935 >        f.completeExceptionally(ex);
1936 >        g.complete(v1);
1937 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1938 >
1939 >        // unspecified behavior
1940 >        Integer v;
1941 >        try {
1942 >            assertEquals(h.join(), inc(v1));
1943 >            assertEquals(1, r.invocationCount);
1944 >        } catch (CompletionException ok) {
1945 >            checkCompletedWithWrappedCFException(h, ex);
1946 >            assertEquals(0, r.invocationCount);
1947 >        }
1948 >
1949 >        checkCompletedWithWrappedCFException(f, ex);
1950 >        checkCompletedNormally(g, v1);
1951 >        }
1952      }
1953  
1954      /**
1955       * applyToEither result completes exceptionally if action does
1956       */
1957 <    public void testApplyToEither3() {
1958 <        CompletableFuture<Integer> f = new CompletableFuture<>();
1959 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1960 <        FailingFunction r = new FailingFunction();
1961 <        CompletableFuture<Integer> g = f.applyToEither(f2, r);
1962 <        f2.complete(two);
1963 <        checkCompletedWithWrappedCFException(g);
1957 >    public void testApplyToEither_actionFailed1() {
1958 >        for (ExecutionMode m : ExecutionMode.values())
1959 >        for (Integer v1 : new Integer[] { 1, null })
1960 >        for (Integer v2 : new Integer[] { 2, null }) {
1961 >
1962 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1963 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1964 >        final FailingFunction r = new FailingFunction();
1965 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1966 >
1967 >        f.complete(v1);
1968 >        checkCompletedWithWrappedCFException(h);
1969 >        g.complete(v2);
1970 >        checkCompletedNormally(f, v1);
1971 >        checkCompletedNormally(g, v2);
1972 >        }
1973 >    }
1974 >
1975 >    public void testApplyToEither_actionFailed2() {
1976 >        for (ExecutionMode m : ExecutionMode.values())
1977 >        for (Integer v1 : new Integer[] { 1, null })
1978 >        for (Integer v2 : new Integer[] { 2, null }) {
1979 >
1980 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
1981 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
1982 >        final FailingFunction r = new FailingFunction();
1983 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
1984 >
1985 >        g.complete(v2);
1986 >        checkCompletedWithWrappedCFException(h);
1987 >        f.complete(v1);
1988 >        checkCompletedNormally(f, v1);
1989 >        checkCompletedNormally(g, v2);
1990 >        }
1991      }
1992  
1993      /**
1994       * applyToEither result completes exceptionally if either source cancelled
1995       */
1996 <    public void testApplyToEither4() {
1997 <        CompletableFuture<Integer> f = new CompletableFuture<>();
1998 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1999 <        CompletableFuture<Integer> g = f.applyToEither(f2, inc);
2000 <        assertTrue(f.cancel(true));
2001 <        checkCompletedWithWrappedCancellationException(g);
2002 <        f = new CompletableFuture<>();
2003 <        f2 = new CompletableFuture<>();
2004 <        assertTrue(f2.cancel(true));
2005 <        checkCompletedWithWrappedCancellationException(g);
1996 >    public void testApplyToEither_sourceCancelled1() {
1997 >        for (ExecutionMode m : ExecutionMode.values())
1998 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
1999 >        for (Integer v1 : new Integer[] { 1, null }) {
2000 >
2001 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2002 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2003 >        final IncFunction r = new IncFunction();
2004 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
2005 >
2006 >        assertTrue(f.cancel(mayInterruptIfRunning));
2007 >        checkCompletedWithWrappedCancellationException(h);
2008 >        g.complete(v1);
2009 >
2010 >        checkCancelled(f);
2011 >        assertEquals(0, r.invocationCount);
2012 >        checkCompletedNormally(g, v1);
2013 >        checkCompletedWithWrappedCancellationException(h);
2014 >        }
2015 >    }
2016 >
2017 >    public void testApplyToEither_sourceCancelled2() {
2018 >        for (ExecutionMode m : ExecutionMode.values())
2019 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2020 >        for (Integer v1 : new Integer[] { 1, null }) {
2021 >
2022 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2023 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2024 >        final IncFunction r = new IncFunction();
2025 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
2026 >
2027 >        assertTrue(g.cancel(mayInterruptIfRunning));
2028 >        checkCompletedWithWrappedCancellationException(h);
2029 >        f.complete(v1);
2030 >
2031 >        checkCancelled(g);
2032 >        assertEquals(0, r.invocationCount);
2033 >        checkCompletedNormally(f, v1);
2034 >        checkCompletedWithWrappedCancellationException(h);
2035 >        }
2036 >    }
2037 >
2038 >    public void testApplyToEither_sourceCancelled3() {
2039 >        for (ExecutionMode m : ExecutionMode.values())
2040 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2041 >        for (Integer v1 : new Integer[] { 1, null }) {
2042 >
2043 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2044 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2045 >        final IncFunction r = new IncFunction();
2046 >
2047 >        assertTrue(g.cancel(mayInterruptIfRunning));
2048 >        f.complete(v1);
2049 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
2050 >
2051 >        // unspecified behavior
2052 >        Integer v;
2053 >        try {
2054 >            assertEquals(h.join(), inc(v1));
2055 >            assertEquals(1, r.invocationCount);
2056 >        } catch (CompletionException ok) {
2057 >            checkCompletedWithWrappedCancellationException(h);
2058 >            assertEquals(0, r.invocationCount);
2059 >        }
2060 >
2061 >        checkCancelled(g);
2062 >        checkCompletedNormally(f, v1);
2063 >        }
2064 >    }
2065 >
2066 >    public void testApplyToEither_sourceCancelled4() {
2067 >        for (ExecutionMode m : ExecutionMode.values())
2068 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2069 >        for (Integer v1 : new Integer[] { 1, null }) {
2070 >
2071 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2072 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2073 >        final IncFunction r = new IncFunction();
2074 >
2075 >        assertTrue(f.cancel(mayInterruptIfRunning));
2076 >        g.complete(v1);
2077 >        final CompletableFuture<Integer> h = m.applyToEither(f, g, r);
2078 >
2079 >        // unspecified behavior
2080 >        Integer v;
2081 >        try {
2082 >            assertEquals(h.join(), inc(v1));
2083 >            assertEquals(1, r.invocationCount);
2084 >        } catch (CompletionException ok) {
2085 >            checkCompletedWithWrappedCancellationException(h);
2086 >            assertEquals(0, r.invocationCount);
2087 >        }
2088 >
2089 >        checkCancelled(f);
2090 >        checkCompletedNormally(g, v1);
2091 >        }
2092      }
2093  
2094      /**
2095       * acceptEither result completes normally after normal completion
2096       * of either source
2097       */
2098 <    public void testAcceptEither() {
2099 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2100 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2101 <        IncAction r = new IncAction();
1793 <        CompletableFuture<Void> g = f.acceptEither(f2, r);
1794 <        f.complete(one);
1795 <        checkCompletedNormally(g, null);
1796 <        f2.complete(one);
1797 <        checkCompletedNormally(g, null);
1798 <        assertEquals(r.value, 2);
2098 >    public void testAcceptEither_normalCompletion1() {
2099 >        for (ExecutionMode m : ExecutionMode.values())
2100 >        for (Integer v1 : new Integer[] { 1, null })
2101 >        for (Integer v2 : new Integer[] { 2, null }) {
2102  
2103 <        r = new IncAction();
2104 <        f = new CompletableFuture<>();
2105 <        f.complete(one);
2106 <        f2 = new CompletableFuture<>();
2107 <        g = f.acceptEither(f2, r);
2108 <        checkCompletedNormally(g, null);
2109 <        assertEquals(r.value, 2);
2103 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2104 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2105 >        final IncAction r = new IncAction();
2106 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2107 >
2108 >        f.complete(v1);
2109 >        checkCompletedNormally(h, null);
2110 >        assertEquals(r.value, inc(v1));
2111 >        g.complete(v2);
2112 >
2113 >        checkCompletedNormally(f, v1);
2114 >        checkCompletedNormally(g, v2);
2115 >        checkCompletedNormally(h, null);
2116 >        }
2117 >    }
2118 >
2119 >    public void testAcceptEither_normalCompletion2() {
2120 >        for (ExecutionMode m : ExecutionMode.values())
2121 >        for (Integer v1 : new Integer[] { 1, null })
2122 >        for (Integer v2 : new Integer[] { 2, null }) {
2123 >
2124 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2125 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2126 >        final IncAction r = new IncAction();
2127 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2128 >
2129 >        g.complete(v2);
2130 >        checkCompletedNormally(h, null);
2131 >        assertEquals(r.value, inc(v2));
2132 >        f.complete(v1);
2133 >
2134 >        checkCompletedNormally(f, v1);
2135 >        checkCompletedNormally(g, v2);
2136 >        checkCompletedNormally(h, null);
2137 >        }
2138 >    }
2139 >    public void testAcceptEither_normalCompletion3() {
2140 >        for (ExecutionMode m : ExecutionMode.values())
2141 >        for (Integer v1 : new Integer[] { 1, null })
2142 >        for (Integer v2 : new Integer[] { 2, null }) {
2143 >
2144 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2145 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2146 >        final IncAction r = new IncAction();
2147 >
2148 >        f.complete(v1);
2149 >        g.complete(v2);
2150 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2151 >
2152 >        checkCompletedNormally(h, null);
2153 >        checkCompletedNormally(f, v1);
2154 >        checkCompletedNormally(g, v2);
2155 >
2156 >        // unspecified behavior
2157 >        assertTrue(Objects.equals(r.value, inc(v1)) ||
2158 >                   Objects.equals(r.value, inc(v2)));
2159 >        }
2160      }
2161  
2162      /**
2163       * acceptEither result completes exceptionally after exceptional
2164       * completion of either source
2165       */
2166 <    public void testAcceptEither2() {
2167 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2168 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1816 <        IncAction r = new IncAction();
1817 <        CompletableFuture<Void> g = f.acceptEither(f2, r);
1818 <        f.completeExceptionally(new CFException());
1819 <        f2.complete(one);
1820 <        checkCompletedWithWrappedCFException(g);
2166 >    public void testAcceptEither_exceptionalCompletion1() {
2167 >        for (ExecutionMode m : ExecutionMode.values())
2168 >        for (Integer v1 : new Integer[] { 1, null }) {
2169  
2170 <        r = new IncAction();
2171 <        f = new CompletableFuture<>();
2172 <        f2 = new CompletableFuture<>();
2173 <        f2.completeExceptionally(new CFException());
2174 <        g = f.acceptEither(f2, r);
2175 <        checkCompletedWithWrappedCFException(g);
2170 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2171 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2172 >        final IncAction r = new IncAction();
2173 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2174 >        final CFException ex = new CFException();
2175 >
2176 >        f.completeExceptionally(ex);
2177 >        checkCompletedWithWrappedCFException(h, ex);
2178 >        g.complete(v1);
2179 >
2180 >        assertEquals(0, r.invocationCount);
2181 >        checkCompletedNormally(g, v1);
2182 >        checkCompletedWithWrappedCFException(f, ex);
2183 >        checkCompletedWithWrappedCFException(h, ex);
2184 >        }
2185 >    }
2186 >
2187 >    public void testAcceptEither_exceptionalCompletion2() {
2188 >        for (ExecutionMode m : ExecutionMode.values())
2189 >        for (Integer v1 : new Integer[] { 1, null }) {
2190 >
2191 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2192 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2193 >        final IncAction r = new IncAction();
2194 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2195 >        final CFException ex = new CFException();
2196 >
2197 >        g.completeExceptionally(ex);
2198 >        checkCompletedWithWrappedCFException(h, ex);
2199 >        f.complete(v1);
2200 >
2201 >        assertEquals(0, r.invocationCount);
2202 >        checkCompletedNormally(f, v1);
2203 >        checkCompletedWithWrappedCFException(g, ex);
2204 >        checkCompletedWithWrappedCFException(h, ex);
2205 >        }
2206 >    }
2207 >
2208 >    public void testAcceptEither_exceptionalCompletion3() {
2209 >        for (ExecutionMode m : ExecutionMode.values())
2210 >        for (Integer v1 : new Integer[] { 1, null }) {
2211 >
2212 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2213 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2214 >        final IncAction r = new IncAction();
2215 >        final CFException ex = new CFException();
2216 >
2217 >        g.completeExceptionally(ex);
2218 >        f.complete(v1);
2219 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2220 >
2221 >        // unspecified behavior
2222 >        Integer v;
2223 >        try {
2224 >            assertEquals(h.join(), null);
2225 >            assertEquals(1, r.invocationCount);
2226 >            assertEquals(inc(v1), r.value);
2227 >        } catch (CompletionException ok) {
2228 >            checkCompletedWithWrappedCFException(h, ex);
2229 >            assertEquals(0, r.invocationCount);
2230 >        }
2231 >
2232 >        checkCompletedWithWrappedCFException(g, ex);
2233 >        checkCompletedNormally(f, v1);
2234 >        }
2235 >    }
2236 >
2237 >    public void testAcceptEither_exceptionalCompletion4() {
2238 >        for (ExecutionMode m : ExecutionMode.values())
2239 >        for (Integer v1 : new Integer[] { 1, null }) {
2240 >
2241 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2242 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2243 >        final IncAction r = new IncAction();
2244 >        final CFException ex = new CFException();
2245 >
2246 >        f.completeExceptionally(ex);
2247 >        g.complete(v1);
2248 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2249 >
2250 >        // unspecified behavior
2251 >        Integer v;
2252 >        try {
2253 >            assertEquals(h.join(), null);
2254 >            assertEquals(1, r.invocationCount);
2255 >            assertEquals(inc(v1), r.value);
2256 >        } catch (CompletionException ok) {
2257 >            checkCompletedWithWrappedCFException(h, ex);
2258 >            assertEquals(0, r.invocationCount);
2259 >        }
2260 >
2261 >        checkCompletedWithWrappedCFException(f, ex);
2262 >        checkCompletedNormally(g, v1);
2263 >        }
2264      }
2265  
2266      /**
2267       * acceptEither result completes exceptionally if action does
2268       */
2269 <    public void testAcceptEither3() {
2270 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2271 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2272 <        FailingConsumer r = new FailingConsumer();
2273 <        CompletableFuture<Void> g = f.acceptEither(f2, r);
2274 <        f2.complete(two);
2275 <        checkCompletedWithWrappedCFException(g);
2269 >    public void testAcceptEither_actionFailed1() {
2270 >        for (ExecutionMode m : ExecutionMode.values())
2271 >        for (Integer v1 : new Integer[] { 1, null })
2272 >        for (Integer v2 : new Integer[] { 2, null }) {
2273 >
2274 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2275 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2276 >        final FailingConsumer r = new FailingConsumer();
2277 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2278 >
2279 >        f.complete(v1);
2280 >        checkCompletedWithWrappedCFException(h);
2281 >        g.complete(v2);
2282 >        checkCompletedNormally(f, v1);
2283 >        checkCompletedNormally(g, v2);
2284 >        }
2285 >    }
2286 >
2287 >    public void testAcceptEither_actionFailed2() {
2288 >        for (ExecutionMode m : ExecutionMode.values())
2289 >        for (Integer v1 : new Integer[] { 1, null })
2290 >        for (Integer v2 : new Integer[] { 2, null }) {
2291 >
2292 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2293 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2294 >        final FailingConsumer r = new FailingConsumer();
2295 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2296 >
2297 >        g.complete(v2);
2298 >        checkCompletedWithWrappedCFException(h);
2299 >        f.complete(v1);
2300 >        checkCompletedNormally(f, v1);
2301 >        checkCompletedNormally(g, v2);
2302 >        }
2303      }
2304  
2305      /**
2306       * acceptEither result completes exceptionally if either source cancelled
2307       */
2308 <    public void testAcceptEither4() {
2309 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2310 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2311 <        IncAction r = new IncAction();
2312 <        CompletableFuture<Void> g = f.acceptEither(f2, r);
2313 <        assertTrue(f.cancel(true));
2314 <        checkCompletedWithWrappedCancellationException(g);
2315 <        f = new CompletableFuture<>();
2316 <        f2 = new CompletableFuture<>();
2317 <        assertTrue(f2.cancel(true));
2318 <        checkCompletedWithWrappedCancellationException(g);
2308 >    public void testAcceptEither_sourceCancelled1() {
2309 >        for (ExecutionMode m : ExecutionMode.values())
2310 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2311 >        for (Integer v1 : new Integer[] { 1, null }) {
2312 >
2313 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2314 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2315 >        final IncAction r = new IncAction();
2316 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2317 >
2318 >        assertTrue(f.cancel(mayInterruptIfRunning));
2319 >        checkCompletedWithWrappedCancellationException(h);
2320 >        g.complete(v1);
2321 >
2322 >        checkCancelled(f);
2323 >        assertEquals(0, r.invocationCount);
2324 >        checkCompletedNormally(g, v1);
2325 >        checkCompletedWithWrappedCancellationException(h);
2326 >        }
2327 >    }
2328 >
2329 >    public void testAcceptEither_sourceCancelled2() {
2330 >        for (ExecutionMode m : ExecutionMode.values())
2331 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2332 >        for (Integer v1 : new Integer[] { 1, null }) {
2333 >
2334 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2335 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2336 >        final IncAction r = new IncAction();
2337 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2338 >
2339 >        assertTrue(g.cancel(mayInterruptIfRunning));
2340 >        checkCompletedWithWrappedCancellationException(h);
2341 >        f.complete(v1);
2342 >
2343 >        checkCancelled(g);
2344 >        assertEquals(0, r.invocationCount);
2345 >        checkCompletedNormally(f, v1);
2346 >        checkCompletedWithWrappedCancellationException(h);
2347 >        }
2348 >    }
2349 >
2350 >    public void testAcceptEither_sourceCancelled3() {
2351 >        for (ExecutionMode m : ExecutionMode.values())
2352 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2353 >        for (Integer v1 : new Integer[] { 1, null }) {
2354 >
2355 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2356 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2357 >        final IncAction r = new IncAction();
2358 >
2359 >        assertTrue(g.cancel(mayInterruptIfRunning));
2360 >        f.complete(v1);
2361 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2362 >
2363 >        // unspecified behavior
2364 >        Integer v;
2365 >        try {
2366 >            assertEquals(h.join(), null);
2367 >            assertEquals(1, r.invocationCount);
2368 >            assertEquals(inc(v1), r.value);
2369 >        } catch (CompletionException ok) {
2370 >            checkCompletedWithWrappedCancellationException(h);
2371 >            assertEquals(0, r.invocationCount);
2372 >        }
2373 >
2374 >        checkCancelled(g);
2375 >        checkCompletedNormally(f, v1);
2376 >        }
2377 >    }
2378 >
2379 >    public void testAcceptEither_sourceCancelled4() {
2380 >        for (ExecutionMode m : ExecutionMode.values())
2381 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2382 >        for (Integer v1 : new Integer[] { 1, null }) {
2383 >
2384 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2385 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2386 >        final IncAction r = new IncAction();
2387 >
2388 >        assertTrue(f.cancel(mayInterruptIfRunning));
2389 >        g.complete(v1);
2390 >        final CompletableFuture<Void> h = m.acceptEither(f, g, r);
2391 >
2392 >        // unspecified behavior
2393 >        Integer v;
2394 >        try {
2395 >            assertEquals(h.join(), null);
2396 >            assertEquals(1, r.invocationCount);
2397 >            assertEquals(inc(v1), r.value);
2398 >        } catch (CompletionException ok) {
2399 >            checkCompletedWithWrappedCancellationException(h);
2400 >            assertEquals(0, r.invocationCount);
2401 >        }
2402 >
2403 >        checkCancelled(f);
2404 >        checkCompletedNormally(g, v1);
2405 >        }
2406      }
2407  
2408      /**
2409       * runAfterEither result completes normally after normal completion
2410       * of either source
2411       */
2412 <    public void testRunAfterEither() {
2413 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2414 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2415 <        Noop r = new Noop();
1866 <        CompletableFuture<Void> g = f.runAfterEither(f2, r);
1867 <        f.complete(one);
1868 <        checkCompletedNormally(g, null);
1869 <        f2.complete(one);
1870 <        checkCompletedNormally(g, null);
1871 <        assertTrue(r.ran);
2412 >    public void testRunAfterEither_normalCompletion1() {
2413 >        for (ExecutionMode m : ExecutionMode.values())
2414 >        for (Integer v1 : new Integer[] { 1, null })
2415 >        for (Integer v2 : new Integer[] { 2, null }) {
2416  
2417 <        r = new Noop();
2418 <        f = new CompletableFuture<>();
2419 <        f.complete(one);
2420 <        f2 = new CompletableFuture<>();
2421 <        g = f.runAfterEither(f2, r);
2422 <        checkCompletedNormally(g, null);
2423 <        assertTrue(r.ran);
2417 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2418 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2419 >        final Noop r = new Noop();
2420 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2421 >
2422 >        f.complete(v1);
2423 >        checkCompletedNormally(h, null);
2424 >        assertEquals(1, r.invocationCount);
2425 >        g.complete(v2);
2426 >
2427 >        checkCompletedNormally(f, v1);
2428 >        checkCompletedNormally(g, v2);
2429 >        checkCompletedNormally(h, null);
2430 >        assertEquals(1, r.invocationCount);
2431 >        }
2432 >    }
2433 >
2434 >    public void testRunAfterEither_normalCompletion2() {
2435 >        for (ExecutionMode m : ExecutionMode.values())
2436 >        for (Integer v1 : new Integer[] { 1, null })
2437 >        for (Integer v2 : new Integer[] { 2, null }) {
2438 >
2439 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2440 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2441 >        final Noop r = new Noop();
2442 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2443 >
2444 >        g.complete(v2);
2445 >        checkCompletedNormally(h, null);
2446 >        assertEquals(1, r.invocationCount);
2447 >        f.complete(v1);
2448 >
2449 >        checkCompletedNormally(f, v1);
2450 >        checkCompletedNormally(g, v2);
2451 >        checkCompletedNormally(h, null);
2452 >        assertEquals(1, r.invocationCount);
2453 >        }
2454 >    }
2455 >    public void testRunAfterEither_normalCompletion3() {
2456 >        for (ExecutionMode m : ExecutionMode.values())
2457 >        for (Integer v1 : new Integer[] { 1, null })
2458 >        for (Integer v2 : new Integer[] { 2, null }) {
2459 >
2460 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2461 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2462 >        final Noop r = new Noop();
2463 >
2464 >        f.complete(v1);
2465 >        g.complete(v2);
2466 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2467 >
2468 >        checkCompletedNormally(h, null);
2469 >        checkCompletedNormally(f, v1);
2470 >        checkCompletedNormally(g, v2);
2471 >        assertEquals(1, r.invocationCount);
2472 >        }
2473      }
2474  
2475      /**
2476       * runAfterEither result completes exceptionally after exceptional
2477       * completion of either source
2478       */
2479 <    public void testRunAfterEither2() {
2480 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2481 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
1889 <        Noop r = new Noop();
1890 <        CompletableFuture<Void> g = f.runAfterEither(f2, r);
1891 <        f.completeExceptionally(new CFException());
1892 <        f2.complete(one);
1893 <        checkCompletedWithWrappedCFException(g);
2479 >    public void testRunAfterEither_exceptionalCompletion1() {
2480 >        for (ExecutionMode m : ExecutionMode.values())
2481 >        for (Integer v1 : new Integer[] { 1, null }) {
2482  
2483 <        r = new Noop();
2484 <        f = new CompletableFuture<>();
2485 <        f2 = new CompletableFuture<>();
2486 <        f2.completeExceptionally(new CFException());
2487 <        g = f.runAfterEither(f2, r);
2488 <        checkCompletedWithWrappedCFException(g);
2483 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2484 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2485 >        final Noop r = new Noop();
2486 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2487 >        final CFException ex = new CFException();
2488 >
2489 >        f.completeExceptionally(ex);
2490 >        checkCompletedWithWrappedCFException(h, ex);
2491 >        g.complete(v1);
2492 >
2493 >        assertEquals(0, r.invocationCount);
2494 >        checkCompletedNormally(g, v1);
2495 >        checkCompletedWithWrappedCFException(f, ex);
2496 >        checkCompletedWithWrappedCFException(h, ex);
2497 >        }
2498 >    }
2499 >
2500 >    public void testRunAfterEither_exceptionalCompletion2() {
2501 >        for (ExecutionMode m : ExecutionMode.values())
2502 >        for (Integer v1 : new Integer[] { 1, null }) {
2503 >
2504 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2505 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2506 >        final Noop r = new Noop();
2507 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2508 >        final CFException ex = new CFException();
2509 >
2510 >        g.completeExceptionally(ex);
2511 >        checkCompletedWithWrappedCFException(h, ex);
2512 >        f.complete(v1);
2513 >
2514 >        assertEquals(0, r.invocationCount);
2515 >        checkCompletedNormally(f, v1);
2516 >        checkCompletedWithWrappedCFException(g, ex);
2517 >        checkCompletedWithWrappedCFException(h, ex);
2518 >        }
2519 >    }
2520 >
2521 >    public void testRunAfterEither_exceptionalCompletion3() {
2522 >        for (ExecutionMode m : ExecutionMode.values())
2523 >        for (Integer v1 : new Integer[] { 1, null }) {
2524 >
2525 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2526 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2527 >        final Noop r = new Noop();
2528 >        final CFException ex = new CFException();
2529 >
2530 >        g.completeExceptionally(ex);
2531 >        f.complete(v1);
2532 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2533 >
2534 >        // unspecified behavior
2535 >        Integer v;
2536 >        try {
2537 >            assertEquals(h.join(), null);
2538 >            assertEquals(1, r.invocationCount);
2539 >        } catch (CompletionException ok) {
2540 >            checkCompletedWithWrappedCFException(h, ex);
2541 >            assertEquals(0, r.invocationCount);
2542 >        }
2543 >
2544 >        checkCompletedWithWrappedCFException(g, ex);
2545 >        checkCompletedNormally(f, v1);
2546 >        }
2547 >    }
2548 >
2549 >    public void testRunAfterEither_exceptionalCompletion4() {
2550 >        for (ExecutionMode m : ExecutionMode.values())
2551 >        for (Integer v1 : new Integer[] { 1, null }) {
2552 >
2553 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2554 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2555 >        final Noop r = new Noop();
2556 >        final CFException ex = new CFException();
2557 >
2558 >        f.completeExceptionally(ex);
2559 >        g.complete(v1);
2560 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2561 >
2562 >        // unspecified behavior
2563 >        Integer v;
2564 >        try {
2565 >            assertEquals(h.join(), null);
2566 >            assertEquals(1, r.invocationCount);
2567 >        } catch (CompletionException ok) {
2568 >            checkCompletedWithWrappedCFException(h, ex);
2569 >            assertEquals(0, r.invocationCount);
2570 >        }
2571 >
2572 >        checkCompletedWithWrappedCFException(f, ex);
2573 >        checkCompletedNormally(g, v1);
2574 >        }
2575      }
2576  
2577      /**
2578       * runAfterEither result completes exceptionally if action does
2579       */
2580 <    public void testRunAfterEither3() {
2581 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2582 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2583 <        FailingNoop r = new FailingNoop();
2584 <        CompletableFuture<Void> g = f.runAfterEither(f2, r);
2585 <        f2.complete(two);
2586 <        checkCompletedWithWrappedCFException(g);
2580 >    public void testRunAfterEither_actionFailed1() {
2581 >        for (ExecutionMode m : ExecutionMode.values())
2582 >        for (Integer v1 : new Integer[] { 1, null })
2583 >        for (Integer v2 : new Integer[] { 2, null }) {
2584 >
2585 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2586 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2587 >        final FailingNoop r = new FailingNoop();
2588 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2589 >
2590 >        f.complete(v1);
2591 >        checkCompletedWithWrappedCFException(h);
2592 >        g.complete(v2);
2593 >        checkCompletedNormally(f, v1);
2594 >        checkCompletedNormally(g, v2);
2595 >        }
2596 >    }
2597 >
2598 >    public void testRunAfterEither_actionFailed2() {
2599 >        for (ExecutionMode m : ExecutionMode.values())
2600 >        for (Integer v1 : new Integer[] { 1, null })
2601 >        for (Integer v2 : new Integer[] { 2, null }) {
2602 >
2603 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2604 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2605 >        final FailingNoop r = new FailingNoop();
2606 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2607 >
2608 >        g.complete(v2);
2609 >        checkCompletedWithWrappedCFException(h);
2610 >        f.complete(v1);
2611 >        checkCompletedNormally(f, v1);
2612 >        checkCompletedNormally(g, v2);
2613 >        }
2614      }
2615  
2616      /**
2617       * runAfterEither result completes exceptionally if either source cancelled
2618       */
2619 <    public void testRunAfterEither4() {
2620 <        CompletableFuture<Integer> f = new CompletableFuture<>();
2621 <        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2622 <        Noop r = new Noop();
2623 <        CompletableFuture<Void> g = f.runAfterEither(f2, r);
2624 <        assertTrue(f.cancel(true));
2625 <        checkCompletedWithWrappedCancellationException(g);
2626 <        f = new CompletableFuture<>();
2627 <        f2 = new CompletableFuture<>();
2628 <        assertTrue(f2.cancel(true));
2629 <        checkCompletedWithWrappedCancellationException(g);
2619 >    public void testRunAfterEither_sourceCancelled1() {
2620 >        for (ExecutionMode m : ExecutionMode.values())
2621 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2622 >        for (Integer v1 : new Integer[] { 1, null }) {
2623 >
2624 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2625 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2626 >        final Noop r = new Noop();
2627 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2628 >
2629 >        assertTrue(f.cancel(mayInterruptIfRunning));
2630 >        checkCompletedWithWrappedCancellationException(h);
2631 >        g.complete(v1);
2632 >
2633 >        checkCancelled(f);
2634 >        assertEquals(0, r.invocationCount);
2635 >        checkCompletedNormally(g, v1);
2636 >        checkCompletedWithWrappedCancellationException(h);
2637 >        }
2638 >    }
2639 >
2640 >    public void testRunAfterEither_sourceCancelled2() {
2641 >        for (ExecutionMode m : ExecutionMode.values())
2642 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2643 >        for (Integer v1 : new Integer[] { 1, null }) {
2644 >
2645 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2646 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2647 >        final Noop r = new Noop();
2648 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2649 >
2650 >        assertTrue(g.cancel(mayInterruptIfRunning));
2651 >        checkCompletedWithWrappedCancellationException(h);
2652 >        f.complete(v1);
2653 >
2654 >        checkCancelled(g);
2655 >        assertEquals(0, r.invocationCount);
2656 >        checkCompletedNormally(f, v1);
2657 >        checkCompletedWithWrappedCancellationException(h);
2658 >        }
2659 >    }
2660 >
2661 >    public void testRunAfterEither_sourceCancelled3() {
2662 >        for (ExecutionMode m : ExecutionMode.values())
2663 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2664 >        for (Integer v1 : new Integer[] { 1, null }) {
2665 >
2666 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2667 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2668 >        final Noop r = new Noop();
2669 >
2670 >        assertTrue(g.cancel(mayInterruptIfRunning));
2671 >        f.complete(v1);
2672 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2673 >
2674 >        // unspecified behavior
2675 >        Integer v;
2676 >        try {
2677 >            assertEquals(h.join(), null);
2678 >            assertEquals(1, r.invocationCount);
2679 >        } catch (CompletionException ok) {
2680 >            checkCompletedWithWrappedCancellationException(h);
2681 >            assertEquals(0, r.invocationCount);
2682 >        }
2683 >
2684 >        checkCancelled(g);
2685 >        checkCompletedNormally(f, v1);
2686 >        }
2687 >    }
2688 >
2689 >    public void testRunAfterEither_sourceCancelled4() {
2690 >        for (ExecutionMode m : ExecutionMode.values())
2691 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false })
2692 >        for (Integer v1 : new Integer[] { 1, null }) {
2693 >
2694 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2695 >        final CompletableFuture<Integer> g = new CompletableFuture<>();
2696 >        final Noop r = new Noop();
2697 >
2698 >        assertTrue(f.cancel(mayInterruptIfRunning));
2699 >        g.complete(v1);
2700 >        final CompletableFuture<Void> h = m.runAfterEither(f, g, r);
2701 >
2702 >        // unspecified behavior
2703 >        Integer v;
2704 >        try {
2705 >            assertEquals(h.join(), null);
2706 >            assertEquals(1, r.invocationCount);
2707 >        } catch (CompletionException ok) {
2708 >            checkCompletedWithWrappedCancellationException(h);
2709 >            assertEquals(0, r.invocationCount);
2710 >        }
2711 >
2712 >        checkCancelled(f);
2713 >        checkCompletedNormally(g, v1);
2714 >        }
2715      }
2716  
2717      /**
2718       * thenCompose result completes normally after normal completion of source
2719       */
2720 <    public void testThenCompose() {
2721 <        CompletableFuture<Integer> f, g;
2722 <        CompletableFutureInc r;
2720 >    public void testThenCompose_normalCompletion1() {
2721 >        for (ExecutionMode m : ExecutionMode.values())
2722 >        for (Integer v1 : new Integer[] { 1, null }) {
2723  
2724 <        f = new CompletableFuture<>();
2725 <        g = f.thenCompose(r = new CompletableFutureInc());
2726 <        f.complete(one);
2727 <        checkCompletedNormally(g, two);
2728 <        assertTrue(r.ran);
2724 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2725 >        final CompletableFutureInc r = new CompletableFutureInc();
2726 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2727 >        f.complete(v1);
2728 >        checkCompletedNormally(g, inc(v1));
2729 >        checkCompletedNormally(f, v1);
2730 >        assertEquals(1, r.invocationCount);
2731 >        }
2732 >    }
2733  
2734 <        f = new CompletableFuture<>();
2735 <        f.complete(one);
2736 <        g = f.thenCompose(r = new CompletableFutureInc());
2737 <        checkCompletedNormally(g, two);
2738 <        assertTrue(r.ran);
2734 >    public void testThenCompose_normalCompletion2() {
2735 >        for (ExecutionMode m : ExecutionMode.values())
2736 >        for (Integer v1 : new Integer[] { 1, null }) {
2737 >
2738 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2739 >        final CompletableFutureInc r = new CompletableFutureInc();
2740 >        f.complete(v1);
2741 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2742 >        checkCompletedNormally(g, inc(v1));
2743 >        checkCompletedNormally(f, v1);
2744 >        assertEquals(1, r.invocationCount);
2745 >        }
2746      }
2747  
2748      /**
2749       * thenCompose result completes exceptionally after exceptional
2750       * completion of source
2751       */
2752 <    public void testThenCompose2() {
2753 <        CompletableFuture<Integer> f, g;
1957 <        CompletableFutureInc r;
2752 >    public void testThenCompose_exceptionalCompletion1() {
2753 >        for (ExecutionMode m : ExecutionMode.values()) {
2754  
2755 <        f = new CompletableFuture<>();
2756 <        g = f.thenCompose(r = new CompletableFutureInc());
2757 <        f.completeExceptionally(new CFException());
2758 <        checkCompletedWithWrappedCFException(g);
2755 >        final CFException ex = new CFException();
2756 >        final CompletableFutureInc r = new CompletableFutureInc();
2757 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2758 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2759 >        f.completeExceptionally(ex);
2760 >        checkCompletedWithWrappedCFException(g, ex);
2761 >        checkCompletedWithWrappedCFException(f, ex);
2762 >        }
2763 >    }
2764  
2765 <        f = new CompletableFuture<>();
2766 <        f.completeExceptionally(new CFException());
2767 <        g = f.thenCompose(r = new CompletableFutureInc());
2768 <        checkCompletedWithWrappedCFException(g);
2765 >    public void testThenCompose_exceptionalCompletion2() {
2766 >        for (ExecutionMode m : ExecutionMode.values()) {
2767 >
2768 >        final CFException ex = new CFException();
2769 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2770 >        f.completeExceptionally(ex);
2771 >        final CompletableFutureInc r = new CompletableFutureInc();
2772 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2773 >        checkCompletedWithWrappedCFException(g, ex);
2774 >        checkCompletedWithWrappedCFException(f, ex);
2775 >        }
2776      }
2777  
2778      /**
2779       * thenCompose result completes exceptionally if action does
2780       */
2781 <    public void testThenCompose3() {
2782 <        CompletableFuture<Integer> f, g;
2783 <        FailingCompletableFutureFunction r;
2781 >    public void testThenCompose_actionFailed1() {
2782 >        for (ExecutionMode m : ExecutionMode.values())
2783 >        for (Integer v1 : new Integer[] { 1, null }) {
2784  
2785 <        f = new CompletableFuture<>();
2786 <        g = f.thenCompose(r = new FailingCompletableFutureFunction());
2787 <        f.complete(one);
2785 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2786 >        final FailingCompletableFutureFunction r
2787 >            = new FailingCompletableFutureFunction();
2788 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2789 >        f.complete(v1);
2790          checkCompletedWithWrappedCFException(g);
2791 +        checkCompletedNormally(f, v1);
2792 +        }
2793 +    }
2794  
2795 <        f = new CompletableFuture<>();
2796 <        f.complete(one);
2797 <        g = f.thenCompose(r = new FailingCompletableFutureFunction());
2795 >    public void testThenCompose_actionFailed2() {
2796 >        for (ExecutionMode m : ExecutionMode.values())
2797 >        for (Integer v1 : new Integer[] { 1, null }) {
2798 >
2799 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2800 >        f.complete(v1);
2801 >        final FailingCompletableFutureFunction r
2802 >            = new FailingCompletableFutureFunction();
2803 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2804          checkCompletedWithWrappedCFException(g);
2805 +        checkCompletedNormally(f, v1);
2806 +        }
2807      }
2808  
2809      /**
2810       * thenCompose result completes exceptionally if source cancelled
2811       */
2812 <    public void testThenCompose4() {
2813 <        CompletableFuture<Integer> f, g;
2814 <        CompletableFutureInc r;
2812 >    public void testThenCompose_sourceCancelled1() {
2813 >        for (ExecutionMode m : ExecutionMode.values())
2814 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false }) {
2815  
2816 <        f = new CompletableFuture<>();
2817 <        g = f.thenCompose(r = new CompletableFutureInc());
2818 <        assertTrue(f.cancel(true));
2816 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2817 >        final CompletableFutureInc r = new CompletableFutureInc();
2818 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2819 >        assertTrue(f.cancel(mayInterruptIfRunning));
2820          checkCompletedWithWrappedCancellationException(g);
2821 +        checkCancelled(f);
2822 +        }
2823 +    }
2824  
2825 <        f = new CompletableFuture<>();
2826 <        assertTrue(f.cancel(true));
2827 <        g = f.thenCompose(r = new CompletableFutureInc());
2825 >    public void testThenCompose_sourceCancelled2() {
2826 >        for (ExecutionMode m : ExecutionMode.values())
2827 >        for (boolean mayInterruptIfRunning : new boolean[] { true, false }) {
2828 >
2829 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
2830 >        assertTrue(f.cancel(mayInterruptIfRunning));
2831 >        final CompletableFutureInc r = new CompletableFutureInc();
2832 >        final CompletableFuture<Integer> g = f.thenCompose(r);
2833          checkCompletedWithWrappedCancellationException(g);
2834 +        checkCancelled(f);
2835 +        }
2836      }
2837  
2838      // asyncs
# Line 2113 | Line 2945 | public class CompletableFutureTest exten
2945          CompletableFuture<Void> g = f.thenAcceptAsync(r);
2946          f.complete(one);
2947          checkCompletedNormally(g, null);
2948 <        assertEquals(r.value, 2);
2948 >        assertEquals(r.value, (Integer) 2);
2949      }
2950  
2951      /**
# Line 2150 | Line 2982 | public class CompletableFutureTest exten
2982          checkCompletedWithWrappedCancellationException(g);
2983      }
2984  
2153    /**
2154     * applyToEitherAsync result completes normally after normal
2155     * completion of sources
2156     */
2157    public void testApplyToEitherAsync() {
2158        CompletableFuture<Integer> f = new CompletableFuture<>();
2159        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2160        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc);
2161        f.complete(one);
2162        checkCompletedNormally(g, two);
2163
2164        f = new CompletableFuture<>();
2165        f.complete(one);
2166        f2 = new CompletableFuture<>();
2167        g = f.applyToEitherAsync(f2, inc);
2168        checkCompletedNormally(g, two);
2169    }
2170
2171    /**
2172     * applyToEitherAsync result completes exceptionally after exceptional
2173     * completion of source
2174     */
2175    public void testApplyToEitherAsync2() {
2176        CompletableFuture<Integer> f = new CompletableFuture<>();
2177        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2178        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc);
2179        f.completeExceptionally(new CFException());
2180        checkCompletedWithWrappedCFException(g);
2181
2182        f = new CompletableFuture<>();
2183        f2 = new CompletableFuture<>();
2184        f2.completeExceptionally(new CFException());
2185        g = f.applyToEitherAsync(f2, inc);
2186        f.complete(one);
2187        checkCompletedWithWrappedCFException(g);
2188    }
2189
2190    /**
2191     * applyToEitherAsync result completes exceptionally if action does
2192     */
2193    public void testApplyToEitherAsync3() {
2194        CompletableFuture<Integer> f = new CompletableFuture<>();
2195        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2196        FailingFunction r = new FailingFunction();
2197        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, r);
2198        f.complete(one);
2199        checkCompletedWithWrappedCFException(g);
2200    }
2201
2202    /**
2203     * applyToEitherAsync result completes exceptionally if either source cancelled
2204     */
2205    public void testApplyToEitherAsync4() {
2206        CompletableFuture<Integer> f = new CompletableFuture<>();
2207        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2208        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc);
2209        assertTrue(f.cancel(true));
2210        checkCompletedWithWrappedCancellationException(g);
2211
2212        f = new CompletableFuture<>();
2213        f2 = new CompletableFuture<>();
2214        assertTrue(f2.cancel(true));
2215        g = f.applyToEitherAsync(f2, inc);
2216        checkCompletedWithWrappedCancellationException(g);
2217    }
2218
2219    /**
2220     * acceptEitherAsync result completes normally after normal
2221     * completion of sources
2222     */
2223    public void testAcceptEitherAsync() {
2224        CompletableFuture<Integer> f = new CompletableFuture<>();
2225        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2226        IncAction r = new IncAction();
2227        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r);
2228        f.complete(one);
2229        checkCompletedNormally(g, null);
2230        assertEquals(r.value, 2);
2231
2232        r = new IncAction();
2233        f = new CompletableFuture<>();
2234        f.complete(one);
2235        f2 = new CompletableFuture<>();
2236        g = f.acceptEitherAsync(f2, r);
2237        checkCompletedNormally(g, null);
2238        assertEquals(r.value, 2);
2239    }
2240
2241    /**
2242     * acceptEitherAsync result completes exceptionally after exceptional
2243     * completion of source
2244     */
2245    public void testAcceptEitherAsync2() {
2246        CompletableFuture<Integer> f = new CompletableFuture<>();
2247        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2248        IncAction r = new IncAction();
2249        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r);
2250        f.completeExceptionally(new CFException());
2251        checkCompletedWithWrappedCFException(g);
2252
2253        r = new IncAction();
2254        f = new CompletableFuture<>();
2255        f2 = new CompletableFuture<>();
2256        f2.completeExceptionally(new CFException());
2257        g = f.acceptEitherAsync(f2, r);
2258        f.complete(one);
2259        checkCompletedWithWrappedCFException(g);
2260    }
2261
2262    /**
2263     * acceptEitherAsync result completes exceptionally if action does
2264     */
2265    public void testAcceptEitherAsync3() {
2266        CompletableFuture<Integer> f = new CompletableFuture<>();
2267        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2268        FailingConsumer r = new FailingConsumer();
2269        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r);
2270        f.complete(one);
2271        checkCompletedWithWrappedCFException(g);
2272    }
2273
2274    /**
2275     * acceptEitherAsync result completes exceptionally if either
2276     * source cancelled
2277     */
2278    public void testAcceptEitherAsync4() {
2279        CompletableFuture<Integer> f = new CompletableFuture<>();
2280        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2281        IncAction r = new IncAction();
2282        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r);
2283        assertTrue(f.cancel(true));
2284        checkCompletedWithWrappedCancellationException(g);
2285
2286        r = new IncAction();
2287        f = new CompletableFuture<>();
2288        f2 = new CompletableFuture<>();
2289        assertTrue(f2.cancel(true));
2290        g = f.acceptEitherAsync(f2, r);
2291        checkCompletedWithWrappedCancellationException(g);
2292    }
2293
2294    /**
2295     * runAfterEitherAsync result completes normally after normal
2296     * completion of sources
2297     */
2298    public void testRunAfterEitherAsync() {
2299        CompletableFuture<Integer> f = new CompletableFuture<>();
2300        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2301        Noop r = new Noop();
2302        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r);
2303        f.complete(one);
2304        checkCompletedNormally(g, null);
2305        assertTrue(r.ran);
2306
2307        r = new Noop();
2308        f = new CompletableFuture<>();
2309        f.complete(one);
2310        f2 = new CompletableFuture<>();
2311        g = f.runAfterEitherAsync(f2, r);
2312        checkCompletedNormally(g, null);
2313        assertTrue(r.ran);
2314    }
2315
2316    /**
2317     * runAfterEitherAsync result completes exceptionally after exceptional
2318     * completion of source
2319     */
2320    public void testRunAfterEitherAsync2() {
2321        CompletableFuture<Integer> f = new CompletableFuture<>();
2322        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2323        Noop r = new Noop();
2324        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r);
2325        f.completeExceptionally(new CFException());
2326        checkCompletedWithWrappedCFException(g);
2327
2328        r = new Noop();
2329        f = new CompletableFuture<>();
2330        f2 = new CompletableFuture<>();
2331        f2.completeExceptionally(new CFException());
2332        g = f.runAfterEitherAsync(f2, r);
2333        f.complete(one);
2334        checkCompletedWithWrappedCFException(g);
2335    }
2336
2337    /**
2338     * runAfterEitherAsync result completes exceptionally if action does
2339     */
2340    public void testRunAfterEitherAsync3() {
2341        CompletableFuture<Integer> f = new CompletableFuture<>();
2342        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2343        FailingNoop r = new FailingNoop();
2344        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r);
2345        f.complete(one);
2346        checkCompletedWithWrappedCFException(g);
2347    }
2348
2349    /**
2350     * runAfterEitherAsync result completes exceptionally if either
2351     * source cancelled
2352     */
2353    public void testRunAfterEitherAsync4() {
2354        CompletableFuture<Integer> f = new CompletableFuture<>();
2355        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2356        Noop r = new Noop();
2357        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r);
2358        assertTrue(f.cancel(true));
2359        checkCompletedWithWrappedCancellationException(g);
2360
2361        r = new Noop();
2362        f = new CompletableFuture<>();
2363        f2 = new CompletableFuture<>();
2364        assertTrue(f2.cancel(true));
2365        g = f.runAfterEitherAsync(f2, r);
2366        checkCompletedWithWrappedCancellationException(g);
2367    }
2368
2369    /**
2370     * thenComposeAsync result completes normally after normal
2371     * completion of source
2372     */
2373    public void testThenComposeAsync() {
2374        CompletableFuture<Integer> f, g;
2375        CompletableFutureInc r;
2376
2377        f = new CompletableFuture<>();
2378        g = f.thenComposeAsync(r = new CompletableFutureInc());
2379        f.complete(one);
2380        checkCompletedNormally(g, two);
2381
2382        f = new CompletableFuture<>();
2383        f.complete(one);
2384        g = f.thenComposeAsync(r = new CompletableFutureInc());
2385        checkCompletedNormally(g, two);
2386    }
2387
2388    /**
2389     * thenComposeAsync result completes exceptionally after
2390     * exceptional completion of source
2391     */
2392    public void testThenComposeAsync2() {
2393        CompletableFuture<Integer> f, g;
2394        CompletableFutureInc r;
2395
2396        f = new CompletableFuture<>();
2397        g = f.thenComposeAsync(r = new CompletableFutureInc());
2398        f.completeExceptionally(new CFException());
2399        checkCompletedWithWrappedCFException(g);
2400        assertFalse(r.ran);
2401
2402        f = new CompletableFuture<>();
2403        f.completeExceptionally(new CFException());
2404        g = f.thenComposeAsync(r = new CompletableFutureInc());
2405        checkCompletedWithWrappedCFException(g);
2406        assertFalse(r.ran);
2407    }
2408
2409    /**
2410     * thenComposeAsync result completes exceptionally if action does
2411     */
2412    public void testThenComposeAsync3() {
2413        CompletableFuture<Integer> f, g;
2414        FailingCompletableFutureFunction r;
2415
2416        f = new CompletableFuture<>();
2417        g = f.thenComposeAsync(r = new FailingCompletableFutureFunction());
2418        f.complete(one);
2419        checkCompletedWithWrappedCFException(g);
2420
2421        f = new CompletableFuture<>();
2422        f.complete(one);
2423        g = f.thenComposeAsync(r = new FailingCompletableFutureFunction());
2424        checkCompletedWithWrappedCFException(g);
2425    }
2426
2427    /**
2428     * thenComposeAsync result completes exceptionally if source cancelled
2429     */
2430    public void testThenComposeAsync4() {
2431        CompletableFuture<Integer> f, g;
2432        CompletableFutureInc r;
2433
2434        f = new CompletableFuture<>();
2435        g = f.thenComposeAsync(r = new CompletableFutureInc());
2436        assertTrue(f.cancel(true));
2437        checkCompletedWithWrappedCancellationException(g);
2438
2439        f = new CompletableFuture<>();
2440        assertTrue(f.cancel(true));
2441        g = f.thenComposeAsync(r = new CompletableFutureInc());
2442        checkCompletedWithWrappedCancellationException(g);
2443    }
2444
2985      // async with explicit executors
2986  
2987      /**
# Line 2552 | Line 3092 | public class CompletableFutureTest exten
3092          CompletableFuture<Void> g = f.thenAcceptAsync(r, new ThreadExecutor());
3093          f.complete(one);
3094          checkCompletedNormally(g, null);
3095 <        assertEquals(r.value, 2);
3095 >        assertEquals(r.value, (Integer) 2);
3096      }
3097  
3098      /**
# Line 2589 | Line 3129 | public class CompletableFutureTest exten
3129          checkCompletedWithWrappedCancellationException(g);
3130      }
3131  
2592    /**
2593     * applyToEitherAsync result completes normally after normal
2594     * completion of sources
2595     */
2596    public void testApplyToEitherAsyncE() {
2597        CompletableFuture<Integer> f = new CompletableFuture<>();
2598        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2599        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2600        f.complete(one);
2601        checkCompletedNormally(g, two);
2602
2603        f = new CompletableFuture<>();
2604        f.complete(one);
2605        f2 = new CompletableFuture<>();
2606        g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2607        checkCompletedNormally(g, two);
2608    }
2609
2610    /**
2611     * applyToEitherAsync result completes exceptionally after exceptional
2612     * completion of source
2613     */
2614    public void testApplyToEitherAsync2E() {
2615        CompletableFuture<Integer> f = new CompletableFuture<>();
2616        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2617        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2618        f.completeExceptionally(new CFException());
2619        checkCompletedWithWrappedCFException(g);
2620
2621        f = new CompletableFuture<>();
2622        f2 = new CompletableFuture<>();
2623        f2.completeExceptionally(new CFException());
2624        g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2625        f.complete(one);
2626        checkCompletedWithWrappedCFException(g);
2627    }
2628
2629    /**
2630     * applyToEitherAsync result completes exceptionally if action does
2631     */
2632    public void testApplyToEitherAsync3E() {
2633        CompletableFuture<Integer> f = new CompletableFuture<>();
2634        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2635        FailingFunction r = new FailingFunction();
2636        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, r, new ThreadExecutor());
2637        f.complete(one);
2638        checkCompletedWithWrappedCFException(g);
2639    }
2640
2641    /**
2642     * applyToEitherAsync result completes exceptionally if either source cancelled
2643     */
2644    public void testApplyToEitherAsync4E() {
2645        CompletableFuture<Integer> f = new CompletableFuture<>();
2646        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2647        CompletableFuture<Integer> g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2648        assertTrue(f.cancel(true));
2649        checkCompletedWithWrappedCancellationException(g);
2650
2651        f = new CompletableFuture<>();
2652        f2 = new CompletableFuture<>();
2653        assertTrue(f2.cancel(true));
2654        g = f.applyToEitherAsync(f2, inc, new ThreadExecutor());
2655        checkCompletedWithWrappedCancellationException(g);
2656    }
2657
2658    /**
2659     * acceptEitherAsync result completes normally after normal
2660     * completion of sources
2661     */
2662    public void testAcceptEitherAsyncE() {
2663        CompletableFuture<Integer> f = new CompletableFuture<>();
2664        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2665        IncAction r = new IncAction();
2666        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2667        f.complete(one);
2668        checkCompletedNormally(g, null);
2669        assertEquals(r.value, 2);
2670
2671        r = new IncAction();
2672        f = new CompletableFuture<>();
2673        f.complete(one);
2674        f2 = new CompletableFuture<>();
2675        g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2676        checkCompletedNormally(g, null);
2677        assertEquals(r.value, 2);
2678    }
2679
2680    /**
2681     * acceptEitherAsync result completes exceptionally after exceptional
2682     * completion of source
2683     */
2684    public void testAcceptEitherAsync2E() {
2685        CompletableFuture<Integer> f = new CompletableFuture<>();
2686        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2687        IncAction r = new IncAction();
2688        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2689        f.completeExceptionally(new CFException());
2690        checkCompletedWithWrappedCFException(g);
2691
2692        r = new IncAction();
2693        f = new CompletableFuture<>();
2694        f2 = new CompletableFuture<>();
2695        f2.completeExceptionally(new CFException());
2696        g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2697        f.complete(one);
2698        checkCompletedWithWrappedCFException(g);
2699    }
2700
2701    /**
2702     * acceptEitherAsync result completes exceptionally if action does
2703     */
2704    public void testAcceptEitherAsync3E() {
2705        CompletableFuture<Integer> f = new CompletableFuture<>();
2706        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2707        FailingConsumer r = new FailingConsumer();
2708        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2709        f.complete(one);
2710        checkCompletedWithWrappedCFException(g);
2711    }
2712
2713    /**
2714     * acceptEitherAsync result completes exceptionally if either
2715     * source cancelled
2716     */
2717    public void testAcceptEitherAsync4E() {
2718        CompletableFuture<Integer> f = new CompletableFuture<>();
2719        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2720        IncAction r = new IncAction();
2721        CompletableFuture<Void> g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2722        assertTrue(f.cancel(true));
2723        checkCompletedWithWrappedCancellationException(g);
2724
2725        r = new IncAction();
2726        f = new CompletableFuture<>();
2727        f2 = new CompletableFuture<>();
2728        assertTrue(f2.cancel(true));
2729        g = f.acceptEitherAsync(f2, r, new ThreadExecutor());
2730        checkCompletedWithWrappedCancellationException(g);
2731    }
2732
2733    /**
2734     * runAfterEitherAsync result completes normally after normal
2735     * completion of sources
2736     */
2737    public void testRunAfterEitherAsyncE() {
2738        CompletableFuture<Integer> f = new CompletableFuture<>();
2739        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2740        Noop r = new Noop();
2741        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2742        f.complete(one);
2743        checkCompletedNormally(g, null);
2744        assertTrue(r.ran);
2745
2746        r = new Noop();
2747        f = new CompletableFuture<>();
2748        f.complete(one);
2749        f2 = new CompletableFuture<>();
2750        g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2751        checkCompletedNormally(g, null);
2752        assertTrue(r.ran);
2753    }
2754
2755    /**
2756     * runAfterEitherAsync result completes exceptionally after exceptional
2757     * completion of source
2758     */
2759    public void testRunAfterEitherAsync2E() {
2760        CompletableFuture<Integer> f = new CompletableFuture<>();
2761        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2762        Noop r = new Noop();
2763        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2764        f.completeExceptionally(new CFException());
2765        checkCompletedWithWrappedCFException(g);
2766
2767        r = new Noop();
2768        f = new CompletableFuture<>();
2769        f2 = new CompletableFuture<>();
2770        f2.completeExceptionally(new CFException());
2771        g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2772        f.complete(one);
2773        checkCompletedWithWrappedCFException(g);
2774    }
2775
2776    /**
2777     * runAfterEitherAsync result completes exceptionally if action does
2778     */
2779    public void testRunAfterEitherAsync3E() {
2780        CompletableFuture<Integer> f = new CompletableFuture<>();
2781        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2782        FailingNoop r = new FailingNoop();
2783        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2784        f.complete(one);
2785        checkCompletedWithWrappedCFException(g);
2786    }
2787
2788    /**
2789     * runAfterEitherAsync result completes exceptionally if either
2790     * source cancelled
2791     */
2792    public void testRunAfterEitherAsync4E() {
2793        CompletableFuture<Integer> f = new CompletableFuture<>();
2794        CompletableFuture<Integer> f2 = new CompletableFuture<>();
2795        Noop r = new Noop();
2796        CompletableFuture<Void> g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2797        assertTrue(f.cancel(true));
2798        checkCompletedWithWrappedCancellationException(g);
2799
2800        r = new Noop();
2801        f = new CompletableFuture<>();
2802        f2 = new CompletableFuture<>();
2803        assertTrue(f2.cancel(true));
2804        g = f.runAfterEitherAsync(f2, r, new ThreadExecutor());
2805        checkCompletedWithWrappedCancellationException(g);
2806    }
2807
2808    /**
2809     * thenComposeAsync result completes normally after normal
2810     * completion of source
2811     */
2812    public void testThenComposeAsyncE() {
2813        CompletableFuture<Integer> f = new CompletableFuture<>();
2814        CompletableFutureInc r = new CompletableFutureInc();
2815        CompletableFuture<Integer> g = f.thenComposeAsync(r, new ThreadExecutor());
2816        f.complete(one);
2817        checkCompletedNormally(g, two);
2818    }
2819
2820    /**
2821     * thenComposeAsync result completes exceptionally after
2822     * exceptional completion of source
2823     */
2824    public void testThenComposeAsync2E() {
2825        CompletableFuture<Integer> f = new CompletableFuture<>();
2826        CompletableFutureInc r = new CompletableFutureInc();
2827        CompletableFuture<Integer> g = f.thenComposeAsync(r, new ThreadExecutor());
2828        f.completeExceptionally(new CFException());
2829        checkCompletedWithWrappedCFException(g);
2830    }
2831
2832    /**
2833     * thenComposeAsync result completes exceptionally if action does
2834     */
2835    public void testThenComposeAsync3E() {
2836        CompletableFuture<Integer> f = new CompletableFuture<>();
2837        FailingCompletableFutureFunction r = new FailingCompletableFutureFunction();
2838        CompletableFuture<Integer> g = f.thenComposeAsync(r, new ThreadExecutor());
2839        f.complete(one);
2840        checkCompletedWithWrappedCFException(g);
2841    }
2842
2843    /**
2844     * thenComposeAsync result completes exceptionally if source cancelled
2845     */
2846    public void testThenComposeAsync4E() {
2847        CompletableFuture<Integer> f = new CompletableFuture<>();
2848        CompletableFutureInc r = new CompletableFutureInc();
2849        CompletableFuture<Integer> g = f.thenComposeAsync(r, new ThreadExecutor());
2850        assertTrue(f.cancel(true));
2851        checkCompletedWithWrappedCancellationException(g);
2852    }
2853
3132      // other static methods
3133  
3134      /**
# Line 3049 | Line 3327 | public class CompletableFutureTest exten
3327       * whenComplete action executes on normal completion, propagating
3328       * source result.
3329       */
3330 <    public void testWhenComplete1() {
3330 >    public void testWhenComplete_normalCompletion1() {
3331 >        for (boolean createIncomplete : new boolean[] { true, false })
3332 >        for (ExecutionMode m : ExecutionMode.values())
3333 >        for (Integer v1 : new Integer[] { 1, null }) {
3334 >
3335          final AtomicInteger a = new AtomicInteger();
3336 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3337 <        CompletableFuture<Integer> g =
3338 <            f.whenComplete((Integer x, Throwable t) -> a.getAndIncrement());
3339 <        f.complete(three);
3340 <        checkCompletedNormally(f, three);
3341 <        checkCompletedNormally(g, three);
3336 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
3337 >        if (!createIncomplete) f.complete(v1);
3338 >        final CompletableFuture<Integer> g =
3339 >            m.whenComplete(f,
3340 >                           (Integer x, Throwable t) -> {
3341 >                               threadAssertSame(x, v1);
3342 >                               threadAssertNull(t);
3343 >                               a.getAndIncrement();
3344 >                           });
3345 >        if (createIncomplete) f.complete(v1);
3346 >        checkCompletedNormally(f, v1);
3347 >        checkCompletedNormally(g, v1);
3348          assertEquals(a.get(), 1);
3349 +        }
3350      }
3351  
3352      /**
3353       * whenComplete action executes on exceptional completion, propagating
3354       * source result.
3355       */
3356 <    public void testWhenComplete2() {
3356 >    public void testWhenComplete_exceptionalCompletion() {
3357 >        for (boolean createIncomplete : new boolean[] { true, false })
3358 >        for (ExecutionMode m : ExecutionMode.values())
3359 >        for (Integer v1 : new Integer[] { 1, null }) {
3360 >
3361          final AtomicInteger a = new AtomicInteger();
3362 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3363 <        CompletableFuture<Integer> g =
3364 <            f.whenComplete((Integer x, Throwable t) -> a.getAndIncrement());
3365 <        f.completeExceptionally(new CFException());
3366 <        assertTrue(f.isCompletedExceptionally());
3367 <        assertTrue(g.isCompletedExceptionally());
3362 >        final CFException ex = new CFException();
3363 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
3364 >        if (!createIncomplete) f.completeExceptionally(ex);
3365 >        final CompletableFuture<Integer> g = m.whenComplete
3366 >            (f,
3367 >             (Integer x, Throwable t) -> {
3368 >                threadAssertNull(x);
3369 >                threadAssertSame(t, ex);
3370 >                a.getAndIncrement();
3371 >            });
3372 >        if (createIncomplete) f.completeExceptionally(ex);
3373 >        checkCompletedWithWrappedCFException(f, ex);
3374 >        checkCompletedWithWrappedCFException(g, ex);
3375          assertEquals(a.get(), 1);
3376 +        }
3377      }
3378  
3379      /**
3380       * If a whenComplete action throws an exception when triggered by
3381       * a normal completion, it completes exceptionally
3382       */
3383 <    public void testWhenComplete3() {
3384 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3385 <        CompletableFuture<Integer> g =
3386 <            f.whenComplete((Integer x, Throwable t) ->
3086 <                           { throw new CFException(); } );
3087 <        f.complete(three);
3088 <        checkCompletedNormally(f, three);
3089 <        assertTrue(g.isCompletedExceptionally());
3090 <        checkCompletedWithWrappedCFException(g);
3091 <    }
3092 <
3093 <    /**
3094 <     * whenCompleteAsync action executes on normal completion, propagating
3095 <     * source result.
3096 <     */
3097 <    public void testWhenCompleteAsync1() {
3098 <        final AtomicInteger a = new AtomicInteger();
3099 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3100 <        CompletableFuture<Integer> g =
3101 <            f.whenCompleteAsync((Integer x, Throwable t) -> a.getAndIncrement());
3102 <        f.complete(three);
3103 <        checkCompletedNormally(f, three);
3104 <        checkCompletedNormally(g, three);
3105 <        assertEquals(a.get(), 1);
3106 <    }
3383 >    public void testWhenComplete_actionFailed() {
3384 >        for (boolean createIncomplete : new boolean[] { true, false })
3385 >        for (ExecutionMode m : ExecutionMode.values())
3386 >        for (Integer v1 : new Integer[] { 1, null }) {
3387  
3388 <    /**
3389 <     * whenCompleteAsync action executes on exceptional completion, propagating
3390 <     * source result.
3391 <     */
3392 <    public void testWhenCompleteAsync2() {
3393 <        final AtomicInteger a = new AtomicInteger();
3394 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3395 <        CompletableFuture<Integer> g =
3396 <            f.whenCompleteAsync((Integer x, Throwable t) -> a.getAndIncrement());
3397 <        f.completeExceptionally(new CFException());
3398 <        checkCompletedWithWrappedCFException(f);
3399 <        checkCompletedWithWrappedCFException(g);
3388 >        final CFException ex = new CFException();
3389 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
3390 >        if (!createIncomplete) f.complete(v1);
3391 >        final CompletableFuture<Integer> g = m.whenComplete
3392 >            (f,
3393 >             (Integer x, Throwable t) -> {
3394 >                threadAssertSame(x, v1);
3395 >                threadAssertNull(t);
3396 >                throw ex;
3397 >            });
3398 >        if (createIncomplete) f.complete(v1);
3399 >        checkCompletedNormally(f, v1);
3400 >        checkCompletedWithWrappedCFException(g, ex);
3401 >        }
3402      }
3403  
3404      /**
3405 <     * If a whenCompleteAsync action throws an exception when
3406 <     * triggered by a normal completion, it completes exceptionally
3405 >     * If a whenComplete action throws an exception when triggered by
3406 >     * a source completion that also throws an exception, the source
3407 >     * exception takes precedence.
3408       */
3409 <    public void testWhenCompleteAsync3() {
3410 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3411 <        CompletableFuture<Integer> g =
3412 <            f.whenCompleteAsync((Integer x, Throwable t) ->
3130 <                           { throw new CFException(); } );
3131 <        f.complete(three);
3132 <        checkCompletedNormally(f, three);
3133 <        checkCompletedWithWrappedCFException(g);
3134 <    }
3409 >    public void testWhenComplete_actionFailedSourceFailed() {
3410 >        for (boolean createIncomplete : new boolean[] { true, false })
3411 >        for (ExecutionMode m : ExecutionMode.values())
3412 >        for (Integer v1 : new Integer[] { 1, null }) {
3413  
3414 <    /**
3415 <     * whenCompleteAsync action executes on normal completion, propagating
3416 <     * source result.
3139 <     */
3140 <    public void testWhenCompleteAsync1e() {
3141 <        final AtomicInteger a = new AtomicInteger();
3142 <        ThreadExecutor exec = new ThreadExecutor();
3143 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3144 <        CompletableFuture<Integer> g =
3145 <            f.whenCompleteAsync((Integer x, Throwable t) -> a.getAndIncrement(),
3146 <                                exec);
3147 <        f.complete(three);
3148 <        checkCompletedNormally(f, three);
3149 <        checkCompletedNormally(g, three);
3150 <        assertEquals(a.get(), 1);
3151 <    }
3414 >        final CFException ex1 = new CFException();
3415 >        final CFException ex2 = new CFException();
3416 >        final CompletableFuture<Integer> f = new CompletableFuture<>();
3417  
3418 <    /**
3419 <     * whenCompleteAsync action executes on exceptional completion, propagating
3420 <     * source result.
3421 <     */
3422 <    public void testWhenCompleteAsync2e() {
3423 <        final AtomicInteger a = new AtomicInteger();
3424 <        ThreadExecutor exec = new ThreadExecutor();
3425 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3426 <        CompletableFuture<Integer> g =
3162 <            f.whenCompleteAsync((Integer x, Throwable t) -> a.getAndIncrement(),
3163 <                                exec);
3164 <        f.completeExceptionally(new CFException());
3165 <        checkCompletedWithWrappedCFException(f);
3166 <        checkCompletedWithWrappedCFException(g);
3167 <    }
3418 >        if (!createIncomplete) f.completeExceptionally(ex1);
3419 >        final CompletableFuture<Integer> g = m.whenComplete
3420 >            (f,
3421 >             (Integer x, Throwable t) -> {
3422 >                threadAssertSame(t, ex1);
3423 >                threadAssertNull(x);
3424 >                throw ex2;
3425 >            });
3426 >        if (createIncomplete) f.completeExceptionally(ex1);
3427  
3428 <    /**
3429 <     * If a whenCompleteAsync action throws an exception when triggered
3430 <     * by a normal completion, it completes exceptionally
3172 <     */
3173 <    public void testWhenCompleteAsync3e() {
3174 <        ThreadExecutor exec = new ThreadExecutor();
3175 <        CompletableFuture<Integer> f = new CompletableFuture<>();
3176 <        CompletableFuture<Integer> g =
3177 <            f.whenCompleteAsync((Integer x, Throwable t) ->
3178 <                                { throw new CFException(); },
3179 <                                exec);
3180 <        f.complete(three);
3181 <        checkCompletedNormally(f, three);
3182 <        checkCompletedWithWrappedCFException(g);
3428 >        checkCompletedWithWrappedCFException(f, ex1);
3429 >        checkCompletedWithWrappedCFException(g, ex1);
3430 >        }
3431      }
3432  
3433      /**
# Line 3192 | Line 3440 | public class CompletableFutureTest exten
3440  
3441          f = new CompletableFuture<>();
3442          g = f.handleAsync(r = new IntegerHandler());
3443 <        assertFalse(r.ran);
3443 >        assertEquals(0, r.invocationCount);
3444          f.completeExceptionally(new CFException());
3445          checkCompletedWithWrappedCFException(f);
3446          checkCompletedNormally(g, three);
3447 <        assertTrue(r.ran);
3447 >        assertEquals(1, r.invocationCount);
3448  
3449          f = new CompletableFuture<>();
3450          g = f.handleAsync(r = new IntegerHandler());
3451 <        assertFalse(r.ran);
3451 >        assertEquals(0, r.invocationCount);
3452          f.completeExceptionally(new CFException());
3453          checkCompletedWithWrappedCFException(f);
3454          checkCompletedNormally(g, three);
3455 <        assertTrue(r.ran);
3455 >        assertEquals(1, r.invocationCount);
3456  
3457          f = new CompletableFuture<>();
3458          g = f.handleAsync(r = new IntegerHandler());
3459 <        assertFalse(r.ran);
3459 >        assertEquals(0, r.invocationCount);
3460          f.complete(one);
3461          checkCompletedNormally(f, one);
3462          checkCompletedNormally(g, two);
3463 <        assertTrue(r.ran);
3463 >        assertEquals(1, r.invocationCount);
3464  
3465          f = new CompletableFuture<>();
3466          g = f.handleAsync(r = new IntegerHandler());
3467 <        assertFalse(r.ran);
3467 >        assertEquals(0, r.invocationCount);
3468          f.complete(one);
3469          checkCompletedNormally(f, one);
3470          checkCompletedNormally(g, two);
3471 <        assertTrue(r.ran);
3471 >        assertEquals(1, r.invocationCount);
3472      }
3473  
3474      /**
# Line 3235 | Line 3483 | public class CompletableFutureTest exten
3483  
3484          f = new CompletableFuture<>();
3485          g = f.handleAsync(r = new IntegerHandler(), exec);
3486 <        assertFalse(r.ran);
3486 >        assertEquals(0, r.invocationCount);
3487          f.completeExceptionally(new CFException());
3488          checkCompletedWithWrappedCFException(f);
3489          checkCompletedNormally(g, three);
3490 <        assertTrue(r.ran);
3490 >        assertEquals(1, r.invocationCount);
3491  
3492          f = new CompletableFuture<>();
3493          g = f.handleAsync(r = new IntegerHandler(), exec);
3494 <        assertFalse(r.ran);
3494 >        assertEquals(0, r.invocationCount);
3495          f.completeExceptionally(new CFException());
3496          checkCompletedWithWrappedCFException(f);
3497          checkCompletedNormally(g, three);
3498 <        assertTrue(r.ran);
3498 >        assertEquals(1, r.invocationCount);
3499  
3500          f = new CompletableFuture<>();
3501          g = f.handleAsync(r = new IntegerHandler(), exec);
3502 <        assertFalse(r.ran);
3502 >        assertEquals(0, r.invocationCount);
3503          f.complete(one);
3504          checkCompletedNormally(f, one);
3505          checkCompletedNormally(g, two);
3506 <        assertTrue(r.ran);
3506 >        assertEquals(1, r.invocationCount);
3507  
3508          f = new CompletableFuture<>();
3509          g = f.handleAsync(r = new IntegerHandler(), exec);
3510 <        assertFalse(r.ran);
3510 >        assertEquals(0, r.invocationCount);
3511          f.complete(one);
3512          checkCompletedNormally(f, one);
3513          checkCompletedNormally(g, two);
3514 <        assertTrue(r.ran);
3514 >        assertEquals(1, r.invocationCount);
3515      }
3516  
3517   }

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines