Java程序  |  194行  |  6.98 KB

/*
 * Copyright (C) 2012 The Guava Authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.google.common.util.concurrent;

import junit.framework.TestCase;

import java.util.concurrent.CancellationException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

/**
 * Tests that {@code AsyncSettableFuture} is a valid {@link ListenableFuture}
 * that behaves itself as expected.
 */

public class AsyncSettableFutureTest extends TestCase {

  private static class Foo {}
  private static class FooChild extends Foo {}

  /** Tests the initial state of the future. */
  public void testCreate() throws Exception {
    AsyncSettableFuture<Integer> future = AsyncSettableFuture.create();
    assertFalse(future.isSet());
    assertFalse(future.isDone());
    assertFalse(future.isCancelled());
  }

  public void testSetValue() throws Exception {
    AsyncSettableFuture<Integer> future = AsyncSettableFuture.create();
    assertTrue(future.setValue(42));
    assertTrue(future.isSet());
    // Later attempts to set the future should return false.
    assertFalse(future.setValue(23));
    assertFalse(future.setException(new Exception("bar")));
    assertFalse(future.setFuture(SettableFuture.<Integer>create()));
    // Check that the future has been set properly.
    assertTrue(future.isDone());
    assertFalse(future.isCancelled());
    assertEquals(42, (int) future.get());
  }

  public void testSetException() throws Exception {
    AsyncSettableFuture<Object> future = AsyncSettableFuture.create();
    Exception e = new Exception("foobarbaz");
    assertTrue(future.setException(e));
    assertTrue(future.isSet());
    // Later attempts to set the future should return false.
    assertFalse(future.setValue(23));
    assertFalse(future.setException(new Exception("quux")));
    assertFalse(future.setFuture(SettableFuture.create()));
    // Check that the future has been set properly.
    assertTrue(future.isDone());
    assertFalse(future.isCancelled());
    try {
      future.get();
      fail("Expected ExecutionException");
    } catch (ExecutionException ee) {
      assertSame(e, ee.getCause());
    }
  }

  public void testSetFuture() throws Exception {
    AsyncSettableFuture<String> future = AsyncSettableFuture.create();
    SettableFuture<String> nested = SettableFuture.create();
    assertTrue(future.setFuture(nested));
    assertTrue(future.isSet());
    // Later attempts to set the future should return false.
    assertFalse(future.setValue("x"));
    assertFalse(future.setException(new Exception("bar")));
    assertFalse(future.setFuture(SettableFuture.<String>create()));
    // Check that the future has been set properly.
    assertFalse(future.isDone());
    assertFalse(future.isCancelled());
    try {
      future.get(0, TimeUnit.MILLISECONDS);
      fail("Expected TimeoutException");
    } catch (TimeoutException expected) { /* expected */ }
    nested.set("foo");
    assertTrue(future.isDone());
    assertFalse(future.isCancelled());
    assertEquals("foo", future.get());
  }

  public void testSetFuture_genericsHierarchy() throws Exception {
    AsyncSettableFuture<Foo> future = AsyncSettableFuture.create();
    SettableFuture<FooChild> nested = SettableFuture.create();
    assertTrue(future.setFuture(nested));
    assertTrue(future.isSet());
    // Later attempts to set the future should return false.
    assertFalse(future.setValue(new Foo()));
    assertFalse(future.setException(new Exception("bar")));
    assertFalse(future.setFuture(SettableFuture.<Foo>create()));
    // Check that the future has been set properly.
    assertFalse(future.isDone());
    assertFalse(future.isCancelled());
    try {
      future.get(0, TimeUnit.MILLISECONDS);
      fail("Expected TimeoutException");
    } catch (TimeoutException expected) { /* expected */ }
    FooChild value = new FooChild();
    nested.set(value);
    assertTrue(future.isDone());
    assertFalse(future.isCancelled());
    assertSame(value, future.get());
  }

  public void testCancel_innerCancelsAsync() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    SettableFuture<Object> inner = SettableFuture.create();
    async.setFuture(inner);
    inner.cancel(true);
    assertTrue(async.isCancelled());
    try {
      async.get();
      fail("Expected CancellationException");
    } catch (CancellationException expected) { /* expected */ }
  }

  public void testCancel_resultCancelsInner_interrupted() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    MyFuture<Object> inner = new MyFuture<Object>();
    async.setFuture(inner);
    async.cancel(true);
    assertTrue(inner.isCancelled());
    assertTrue(inner.myWasInterrupted());
    try {
      inner.get();
      fail("Expected CancellationException");
    } catch (CancellationException expected) { /* expected */ }
  }

  public void testCancel_resultCancelsInner() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    MyFuture<Object> inner = new MyFuture<Object>();
    async.setFuture(inner);
    async.cancel(false);
    assertTrue(inner.isCancelled());
    assertFalse(inner.myWasInterrupted());
    try {
      inner.get();
      fail("Expected CancellationException");
    } catch (CancellationException expected) { /* expected */ }
  }

  public void testCancel_beforeSet() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    async.cancel(true);
    assertFalse(async.setValue(42));
  }

  public void testCancel_multipleBeforeSetFuture_noInterruptFirst() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    async.cancel(false);
    async.cancel(true);
    MyFuture<Object> inner = new MyFuture<Object>();
    assertFalse(async.setFuture(inner));
    assertTrue(inner.isCancelled());
    assertFalse(inner.myWasInterrupted());
  }

  public void testCancel_multipleBeforeSetFuture_interruptFirst() throws Exception {
    AsyncSettableFuture<Object> async = AsyncSettableFuture.create();
    async.cancel(true);
    async.cancel(false);
    MyFuture<Object> inner = new MyFuture<Object>();
    assertFalse(async.setFuture(inner));
    assertTrue(inner.isCancelled());
    assertTrue(inner.myWasInterrupted());
  }

  private static class MyFuture<V> extends AbstractFuture<V> {
    boolean myWasInterrupted() {
      // we need a new method since wasInterrupted is final, so we can't increase its visibility.
      return wasInterrupted();
    }
  }
}