blob: f14788aab3aae6494aa40a004f3a052aa60edcc5 [file] [log] [blame]
/*
* Written by Doug Lea with assistance from members of JCP JSR-166
* Expert Group and released to the public domain, as explained at
* http://creativecommons.org/licenses/publicdomain
* Other contributors include Andrew Wright, Jeffrey Hayes,
* Pat Fisher, Mike Judd.
*/
import junit.framework.*;
import java.util.concurrent.*;
import java.util.*;
public class FutureTaskTest extends JSR166TestCase {
public static void main(String[] args) {
junit.textui.TestRunner.run (suite());
}
public static Test suite() {
return new TestSuite(FutureTaskTest.class);
}
/**
* Subclass to expose protected methods
*/
static class PublicFutureTask extends FutureTask {
public PublicFutureTask(Callable r) { super(r); }
public boolean runAndReset() { return super.runAndReset(); }
public void set(Object x) { super.set(x); }
public void setException(Throwable t) { super.setException(t); }
}
/**
* Creating a future with a null callable throws NPE
*/
public void testConstructor() {
try {
FutureTask task = new FutureTask(null);
shouldThrow();
}
catch(NullPointerException success) {
}
}
/**
* creating a future with null runnable fails
*/
public void testConstructor2() {
try {
FutureTask task = new FutureTask(null, Boolean.TRUE);
shouldThrow();
}
catch(NullPointerException success) {
}
}
/**
* isDone is true when a task completes
*/
public void testIsDone() {
FutureTask task = new FutureTask( new NoOpCallable());
task.run();
assertTrue(task.isDone());
assertFalse(task.isCancelled());
}
/**
* runAndReset of a non-cancelled task succeeds
*/
public void testRunAndReset() {
PublicFutureTask task = new PublicFutureTask(new NoOpCallable());
assertTrue(task.runAndReset());
assertFalse(task.isDone());
}
/**
* runAndReset after cancellation fails
*/
public void testResetAfterCancel() {
PublicFutureTask task = new PublicFutureTask(new NoOpCallable());
assertTrue(task.cancel(false));
assertFalse(task.runAndReset());
assertTrue(task.isDone());
assertTrue(task.isCancelled());
}
/**
* setting value causes get to return it
*/
public void testSet() {
PublicFutureTask task = new PublicFutureTask(new NoOpCallable());
task.set(one);
try {
assertEquals(task.get(), one);
}
catch(Exception e) {
unexpectedException();
}
}
/**
* setException causes get to throw ExecutionException
*/
public void testSetException() {
Exception nse = new NoSuchElementException();
PublicFutureTask task = new PublicFutureTask(new NoOpCallable());
task.setException(nse);
try {
Object x = task.get();
shouldThrow();
}
catch(ExecutionException ee) {
Throwable cause = ee.getCause();
assertEquals(cause, nse);
}
catch(Exception e) {
unexpectedException();
}
}
/**
* Cancelling before running succeeds
*/
public void testCancelBeforeRun() {
FutureTask task = new FutureTask( new NoOpCallable());
assertTrue(task.cancel(false));
task.run();
assertTrue(task.isDone());
assertTrue(task.isCancelled());
}
/**
* Cancel(true) before run succeeds
*/
public void testCancelBeforeRun2() {
FutureTask task = new FutureTask( new NoOpCallable());
assertTrue(task.cancel(true));
task.run();
assertTrue(task.isDone());
assertTrue(task.isCancelled());
}
/**
* cancel of a completed task fails
*/
public void testCancelAfterRun() {
FutureTask task = new FutureTask( new NoOpCallable());
task.run();
assertFalse(task.cancel(false));
assertTrue(task.isDone());
assertFalse(task.isCancelled());
}
/**
* cancel(true) interrupts a running task
*/
public void testCancelInterrupt() {
FutureTask task = new FutureTask( new Callable() {
public Object call() {
try {
Thread.sleep(MEDIUM_DELAY_MS);
threadShouldThrow();
}
catch (InterruptedException success) {}
return Boolean.TRUE;
} });
Thread t = new Thread(task);
t.start();
try {
Thread.sleep(SHORT_DELAY_MS);
assertTrue(task.cancel(true));
t.join();
assertTrue(task.isDone());
assertTrue(task.isCancelled());
} catch(InterruptedException e){
unexpectedException();
}
}
/**
* cancel(false) does not interrupt a running task
*/
public void testCancelNoInterrupt() {
FutureTask task = new FutureTask( new Callable() {
public Object call() {
try {
Thread.sleep(MEDIUM_DELAY_MS);
}
catch (InterruptedException success) {
threadFail("should not interrupt");
}
return Boolean.TRUE;
} });
Thread t = new Thread(task);
t.start();
try {
Thread.sleep(SHORT_DELAY_MS);
assertTrue(task.cancel(false));
t.join();
assertTrue(task.isDone());
assertTrue(task.isCancelled());
} catch(InterruptedException e){
unexpectedException();
}
}
/**
* set in one thread causes get in another thread to retrieve value
*/
public void testGet1() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
try {
Thread.sleep(MEDIUM_DELAY_MS);
} catch(InterruptedException e){
threadUnexpectedException();
}
return Boolean.TRUE;
}
});
Thread t = new Thread(new Runnable() {
public void run() {
try {
ft.get();
} catch(Exception e){
threadUnexpectedException();
}
}
});
try {
assertFalse(ft.isDone());
assertFalse(ft.isCancelled());
t.start();
Thread.sleep(SHORT_DELAY_MS);
ft.run();
t.join();
assertTrue(ft.isDone());
assertFalse(ft.isCancelled());
} catch(InterruptedException e){
unexpectedException();
}
}
/**
* set in one thread causes timed get in another thread to retrieve value
*/
public void testTimedGet1() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
try {
Thread.sleep(MEDIUM_DELAY_MS);
} catch(InterruptedException e){
threadUnexpectedException();
}
return Boolean.TRUE;
}
});
Thread t = new Thread(new Runnable() {
public void run() {
try {
ft.get(SHORT_DELAY_MS, TimeUnit.MILLISECONDS);
} catch(TimeoutException success) {
} catch(Exception e){
threadUnexpectedException();
}
}
});
try {
assertFalse(ft.isDone());
assertFalse(ft.isCancelled());
t.start();
ft.run();
t.join();
assertTrue(ft.isDone());
assertFalse(ft.isCancelled());
} catch(InterruptedException e){
unexpectedException();
}
}
/**
* Cancelling a task causes timed get in another thread to throw CancellationException
*/
public void testTimedGet_Cancellation() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
try {
Thread.sleep(SMALL_DELAY_MS);
threadShouldThrow();
} catch(InterruptedException e) {
}
return Boolean.TRUE;
}
});
try {
Thread t1 = new Thread(new Runnable() {
public void run() {
try {
ft.get(MEDIUM_DELAY_MS, TimeUnit.MILLISECONDS);
threadShouldThrow();
} catch(CancellationException success) {}
catch(Exception e){
threadUnexpectedException();
}
}
});
Thread t2 = new Thread(ft);
t1.start();
t2.start();
Thread.sleep(SHORT_DELAY_MS);
ft.cancel(true);
t1.join();
t2.join();
} catch(InterruptedException ie){
unexpectedException();
}
}
/**
* Cancelling a task causes get in another thread to throw CancellationException
*/
public void testGet_Cancellation() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
try {
Thread.sleep(MEDIUM_DELAY_MS);
threadShouldThrow();
} catch(InterruptedException e){
}
return Boolean.TRUE;
}
});
try {
Thread t1 = new Thread(new Runnable() {
public void run() {
try {
ft.get();
threadShouldThrow();
} catch(CancellationException success){
}
catch(Exception e){
threadUnexpectedException();
}
}
});
Thread t2 = new Thread(ft);
t1.start();
t2.start();
Thread.sleep(SHORT_DELAY_MS);
ft.cancel(true);
t1.join();
t2.join();
} catch(InterruptedException success){
unexpectedException();
}
}
/**
* A runtime exception in task causes get to throw ExecutionException
*/
public void testGet_ExecutionException() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
int i = 5/0;
return Boolean.TRUE;
}
});
try {
ft.run();
ft.get();
shouldThrow();
} catch(ExecutionException success){
}
catch(Exception e){
unexpectedException();
}
}
/**
* A runtime exception in task causes timed get to throw ExecutionException
*/
public void testTimedGet_ExecutionException2() {
final FutureTask ft = new FutureTask(new Callable() {
public Object call() {
int i = 5/0;
return Boolean.TRUE;
}
});
try {
ft.run();
ft.get(SHORT_DELAY_MS, TimeUnit.MILLISECONDS);
shouldThrow();
} catch(ExecutionException success) {
} catch(TimeoutException success) { } // unlikely but OK
catch(Exception e){
unexpectedException();
}
}
/**
* Interrupting a waiting get causes it to throw InterruptedException
*/
public void testGet_InterruptedException() {
final FutureTask ft = new FutureTask(new NoOpCallable());
Thread t = new Thread(new Runnable() {
public void run() {
try {
ft.get();
threadShouldThrow();
} catch(InterruptedException success){
} catch(Exception e){
threadUnexpectedException();
}
}
});
try {
t.start();
Thread.sleep(SHORT_DELAY_MS);
t.interrupt();
t.join();
} catch(Exception e){
unexpectedException();
}
}
/**
* Interrupting a waiting timed get causes it to throw InterruptedException
*/
public void testTimedGet_InterruptedException2() {
final FutureTask ft = new FutureTask(new NoOpCallable());
Thread t = new Thread(new Runnable() {
public void run() {
try {
ft.get(LONG_DELAY_MS,TimeUnit.MILLISECONDS);
threadShouldThrow();
} catch(InterruptedException success){}
catch(Exception e){
threadUnexpectedException();
}
}
});
try {
t.start();
Thread.sleep(SHORT_DELAY_MS);
t.interrupt();
t.join();
} catch(Exception e){
unexpectedException();
}
}
/**
* A timed out timed get throws TimeoutException
*/
public void testGet_TimeoutException() {
try {
FutureTask ft = new FutureTask(new NoOpCallable());
ft.get(1,TimeUnit.MILLISECONDS);
shouldThrow();
} catch(TimeoutException success){}
catch(Exception success){
unexpectedException();
}
}
}