Home | History | Annotate | Download | only in concurrent
      1 /*
      2  * Copyright (C) 2011 The Guava Authors
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  * http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.google.common.util.concurrent;
     18 
     19 import com.google.common.base.Preconditions;
     20 
     21 import junit.framework.TestCase;
     22 
     23 import org.mockito.Mockito;
     24 
     25 import java.util.concurrent.CancellationException;
     26 import java.util.concurrent.Executor;
     27 import java.util.concurrent.Future;
     28 import java.util.concurrent.TimeUnit;
     29 
     30 import javax.annotation.Nullable;
     31 
     32 /**
     33  * Test for {@link FutureCallback}.
     34  *
     35  * @author Anthony Zana
     36  */
     37 public class FutureCallbackTest extends TestCase {
     38   public void testSameThreadSuccess() {
     39     SettableFuture<String> f = SettableFuture.create();
     40     MockCallback callback = new MockCallback("foo");
     41     Futures.addCallback(f, callback);
     42     f.set("foo");
     43   }
     44 
     45   public void testExecutorSuccess() {
     46     CountingSameThreadExecutor ex = new CountingSameThreadExecutor();
     47     SettableFuture<String> f = SettableFuture.create();
     48     MockCallback callback = new MockCallback("foo");
     49     Futures.addCallback(f, callback, ex);
     50     f.set("foo");
     51     assertEquals(1, ex.runCount);
     52   }
     53 
     54   // Error cases
     55   public void testSameThreadExecutionException() {
     56     SettableFuture<String> f = SettableFuture.create();
     57     Exception e = new IllegalArgumentException("foo not found");
     58     MockCallback callback = new MockCallback(e);
     59     Futures.addCallback(f, callback);
     60     f.setException(e);
     61   }
     62 
     63   public void testCancel() {
     64     SettableFuture<String> f = SettableFuture.create();
     65     FutureCallback<String> callback =
     66         new FutureCallback<String>() {
     67           private boolean called = false;
     68           @Override
     69           public void onSuccess(String result) {
     70             fail("Was not expecting onSuccess() to be called.");
     71           }
     72 
     73           @Override
     74           public synchronized void onFailure(Throwable t) {
     75             assertFalse(called);
     76             assertTrue(t instanceof CancellationException);
     77             called = true;
     78           }
     79         };
     80     Futures.addCallback(f, callback);
     81     f.cancel(true);
     82   }
     83 
     84   public void testThrowErrorFromGet() {
     85     Error error = new AssertionError("ASSERT!");
     86     ListenableFuture<String> f = ThrowingFuture.throwingError(error);
     87     MockCallback callback = new MockCallback(error);
     88     Futures.addCallback(f, callback);
     89   }
     90 
     91   public void testRuntimeExeceptionFromGet() {
     92     RuntimeException e = new IllegalArgumentException("foo not found");
     93     ListenableFuture<String> f = ThrowingFuture.throwingRuntimeException(e);
     94     MockCallback callback = new MockCallback(e);
     95     Futures.addCallback(f, callback);
     96   }
     97 
     98   public void testOnSuccessThrowsRuntimeException() throws Exception {
     99     RuntimeException exception = new RuntimeException();
    100     String result = "result";
    101     SettableFuture<String> future = SettableFuture.create();
    102     @SuppressWarnings("unchecked") // Safe for a mock
    103     FutureCallback<String> callback = Mockito.mock(FutureCallback.class);
    104     Futures.addCallback(future, callback);
    105     Mockito.doThrow(exception).when(callback).onSuccess(result);
    106     future.set(result);
    107     assertEquals(result, future.get());
    108     Mockito.verify(callback).onSuccess(result);
    109     Mockito.verifyNoMoreInteractions(callback);
    110   }
    111 
    112   public void testOnSuccessThrowsError() throws Exception {
    113     class TestError extends Error {}
    114     TestError error = new TestError();
    115     String result = "result";
    116     SettableFuture<String> future = SettableFuture.create();
    117     @SuppressWarnings("unchecked") // Safe for a mock
    118     FutureCallback<String> callback = Mockito.mock(FutureCallback.class);
    119     Futures.addCallback(future, callback);
    120     Mockito.doThrow(error).when(callback).onSuccess(result);
    121     try {
    122       future.set(result);
    123       fail("Should have thrown");
    124     } catch (TestError e) {
    125       assertSame(error, e);
    126     }
    127     assertEquals(result, future.get());
    128     Mockito.verify(callback).onSuccess(result);
    129     Mockito.verifyNoMoreInteractions(callback);
    130   }
    131 
    132   public void testWildcardFuture() {
    133     SettableFuture<String> settable = SettableFuture.create();
    134     ListenableFuture<?> f = settable;
    135     FutureCallback<Object> callback = new FutureCallback<Object>() {
    136       @Override
    137       public void onSuccess(Object result) {}
    138 
    139       @Override
    140       public void onFailure(Throwable t) {}
    141     };
    142     Futures.addCallback(f, callback);
    143   }
    144 
    145   private class CountingSameThreadExecutor implements Executor {
    146     int runCount = 0;
    147     @Override
    148     public void execute(Runnable command) {
    149       command.run();
    150       runCount++;
    151     }
    152   }
    153 
    154   // TODO(user): Move to testing, unify with RuntimeExceptionThrowingFuture
    155 
    156   /**
    157    * A {@link Future} implementation which always throws directly from calls to
    158    * get() (i.e. not wrapped in ExecutionException.
    159    * For just a normal Future failure, use {@link SettableFuture}).
    160    *
    161    * <p>Useful for testing the behavior of Future utilities against odd futures.
    162    *
    163    * @author Anthony Zana
    164    */
    165   private static class ThrowingFuture<V> implements ListenableFuture<V> {
    166     private final Error error;
    167     private final RuntimeException runtime;
    168 
    169     public static <V> ListenableFuture<V> throwingError(Error error) {
    170       return new ThrowingFuture<V>(error);
    171     }
    172 
    173     public static <V> ListenableFuture<V>
    174         throwingRuntimeException(RuntimeException e) {
    175       return new ThrowingFuture<V>(e);
    176     }
    177 
    178     private ThrowingFuture(Error error) {
    179       this.error = Preconditions.checkNotNull(error);
    180       this.runtime = null;
    181     }
    182 
    183     public ThrowingFuture(RuntimeException e) {
    184       this.runtime = Preconditions.checkNotNull(e);
    185       this.error = null;
    186     }
    187 
    188     @Override
    189     public boolean cancel(boolean mayInterruptIfRunning) {
    190       return false;
    191     }
    192 
    193     @Override
    194     public boolean isCancelled() {
    195       return false;
    196     }
    197 
    198     @Override
    199     public boolean isDone() {
    200       return true;
    201     }
    202 
    203     @Override
    204     public V get() {
    205       throwOnGet();
    206       throw new AssertionError("Unreachable");
    207     }
    208 
    209     @Override
    210     public V get(long timeout, TimeUnit unit) {
    211       throwOnGet();
    212       throw new AssertionError("Unreachable");
    213     }
    214 
    215     @Override
    216     public void addListener(Runnable listener, Executor executor) {
    217       executor.execute(listener);
    218     }
    219 
    220     private void throwOnGet() {
    221       if (error != null) {
    222         throw error;
    223       } else {
    224         throw runtime;
    225       }
    226     }
    227   }
    228 
    229   private final class MockCallback implements FutureCallback<String> {
    230     @Nullable private String value = null;
    231     @Nullable private Throwable failure = null;
    232     private boolean wasCalled = false;
    233 
    234     MockCallback(String expectedValue) {
    235       this.value = expectedValue;
    236     }
    237 
    238     public MockCallback(Throwable expectedFailure) {
    239       this.failure = expectedFailure;
    240     }
    241 
    242     @Override
    243     public synchronized void onSuccess(String result) {
    244       assertFalse(wasCalled);
    245       wasCalled = true;
    246       assertEquals(value, result);
    247     }
    248 
    249     @Override
    250     public synchronized void onFailure(Throwable t) {
    251       assertFalse(wasCalled);
    252       wasCalled = true;
    253       assertEquals(failure, t);
    254     }
    255   }
    256 }
    257