From 9b190e2d7e1a4fb8a9320dad46afcf1677d8729c Mon Sep 17 00:00:00 2001 From: Scott Mitchell Date: Mon, 7 Dec 2015 18:23:03 -0800 Subject: [PATCH] AbstractFuture should not wrap CancellationException Motivation: AbstractFuture currently wraps CancellationException in a ExecutionException. However the interface of Future says that this exception should be directly thrown. Modifications: - Throw CancellationException from AbstractFuture.get Result: Interface contract for CancellationException is honored in AbstractFuture. --- .../netty/util/concurrent/AbstractFuture.java | 7 +++++++ .../util/concurrent/DefaultPromiseTest.java | 18 ++++++++++++++++++ 2 files changed, 25 insertions(+) diff --git a/common/src/main/java/io/netty/util/concurrent/AbstractFuture.java b/common/src/main/java/io/netty/util/concurrent/AbstractFuture.java index b6073b69cd..c0a95dedfd 100644 --- a/common/src/main/java/io/netty/util/concurrent/AbstractFuture.java +++ b/common/src/main/java/io/netty/util/concurrent/AbstractFuture.java @@ -15,6 +15,7 @@ */ package io.netty.util.concurrent; +import java.util.concurrent.CancellationException; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; @@ -34,6 +35,9 @@ public abstract class AbstractFuture implements Future { if (cause == null) { return getNow(); } + if (cause instanceof CancellationException) { + throw (CancellationException) cause; + } throw new ExecutionException(cause); } @@ -44,6 +48,9 @@ public abstract class AbstractFuture implements Future { if (cause == null) { return getNow(); } + if (cause instanceof CancellationException) { + throw (CancellationException) cause; + } throw new ExecutionException(cause); } throw new TimeoutException(); diff --git a/common/src/test/java/io/netty/util/concurrent/DefaultPromiseTest.java b/common/src/test/java/io/netty/util/concurrent/DefaultPromiseTest.java index 59085d6b89..58e98907b6 100644 --- a/common/src/test/java/io/netty/util/concurrent/DefaultPromiseTest.java +++ b/common/src/test/java/io/netty/util/concurrent/DefaultPromiseTest.java @@ -19,10 +19,13 @@ package io.netty.util.concurrent; import org.junit.Test; import java.util.concurrent.BlockingQueue; +import java.util.concurrent.CancellationException; import java.util.concurrent.CountDownLatch; +import java.util.concurrent.ExecutionException; import java.util.concurrent.Executors; import java.util.concurrent.LinkedBlockingQueue; import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; import java.util.concurrent.atomic.AtomicInteger; import static org.hamcrest.CoreMatchers.is; @@ -34,6 +37,21 @@ import static org.junit.Assert.assertTrue; @SuppressWarnings("unchecked") public class DefaultPromiseTest { + @Test(expected = CancellationException.class) + public void testCancellationExceptionIsThrownWhenBlockingGet() throws InterruptedException, ExecutionException { + final Promise promise = new DefaultPromise(ImmediateEventExecutor.INSTANCE); + promise.cancel(false); + promise.get(); + } + + @Test(expected = CancellationException.class) + public void testCancellationExceptionIsThrownWhenBlockingGetWithTimeout() throws InterruptedException, + ExecutionException, TimeoutException { + final Promise promise = new DefaultPromise(ImmediateEventExecutor.INSTANCE); + promise.cancel(false); + promise.get(1, TimeUnit.SECONDS); + } + @Test public void testNoStackOverflowErrorWithImmediateEventExecutorA() throws Exception { final Promise[] p = new DefaultPromise[128];