/*
* Copyright (C) 2006 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.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
/**
* Unit test for {@link SimpleTimeLimiter}.
*
* @author kevinb
*/
public class SimpleTimeLimiterTest extends TestCase {
private static final int DELAY_MS = 50;
private static final int ENOUGH_MS = 500;
private static final int NOT_ENOUGH_MS = 5;
private TimeLimiter service;
private static final ExecutorService executor
= Executors.newFixedThreadPool(1);
private static String someGoodStaticMethod() throws InterruptedException {
TimeUnit.MILLISECONDS.sleep(DELAY_MS);
return "yes";
}
private static String someBadStaticMethod() throws InterruptedException,
SampleException {
TimeUnit.MILLISECONDS.sleep(DELAY_MS);
throw new SampleException();
}
@Override protected void setUp() throws Exception {
super.setUp();
service = new SimpleTimeLimiter(executor);
}
public void testGoodCallableWithEnoughTime() throws Exception {
long start = System.nanoTime();
String result = service.callWithTimeout(
new Callable<String>() {
@Override
public String call() throws InterruptedException {
return someGoodStaticMethod();
}
}, ENOUGH_MS, TimeUnit.MILLISECONDS, true);
assertEquals("yes", result);
assertTheCallTookBetween(start, DELAY_MS, ENOUGH_MS);
}
public void testGoodCallableWithNotEnoughTime() throws Exception {
long start = System.nanoTime();
try {
service.callWithTimeout(
new Callable<String>() {
@Override
public String call() throws InterruptedException {
return someGoodStaticMethod();
}
}, NOT_ENOUGH_MS, TimeUnit.MILLISECONDS, true);
fail("no exception thrown");
} catch (UncheckedTimeoutException expected) {
}
assertTheCallTookBetween(start, NOT_ENOUGH_MS, DELAY_MS);
}
public void testBadCallableWithEnoughTime() throws Exception {
long start = System.nanoTime();
try {
service.callWithTimeout(
new Callable<String>() {
@Override
public String call() throws SampleException, InterruptedException {
return someBadStaticMethod();
}
}, ENOUGH_MS, TimeUnit.MILLISECONDS, true);
fail("no exception thrown");
} catch (SampleException expected) {
}
assertTheCallTookBetween(start, DELAY_MS, ENOUGH_MS);
}
public void testBadCallableWithNotEnoughTime() throws Exception {
long start = System.nanoTime();
try {
service.callWithTimeout(
new Callable<String>() {
@Override
public String call() throws SampleException, InterruptedException {
return someBadStaticMethod();
}
}, NOT_ENOUGH_MS, TimeUnit.MILLISECONDS, true);
fail("no exception thrown");
} catch (UncheckedTimeoutException expected) {
}
assertTheCallTookBetween(start, NOT_ENOUGH_MS, DELAY_MS);
}
public void testGoodMethodWithEnoughTime() throws Exception {
SampleImpl target = new SampleImpl();
Sample proxy = service.newProxy(
target, Sample.class, ENOUGH_MS, TimeUnit.MILLISECONDS);
long start = System.nanoTime();
assertEquals("x", proxy.sleepThenReturnInput("x"));
assertTheCallTookBetween(start, DELAY_MS, ENOUGH_MS);
assertTrue(target.finished);
}
public void testGoodMethodWithNotEnoughTime() throws Exception {
SampleImpl target = new SampleImpl();
Sample proxy = service.newProxy(
target, Sample.class, NOT_ENOUGH_MS, TimeUnit.MILLISECONDS);
long start = System.nanoTime();
try {
proxy.sleepThenReturnInput("x");
fail("no exception thrown");
} catch (UncheckedTimeoutException expected) {
}
assertTheCallTookBetween(start, NOT_ENOUGH_MS, DELAY_MS);
// Is it still computing away anyway?
assertFalse(target.finished);
TimeUnit.MILLISECONDS.sleep(ENOUGH_MS);
assertFalse(target.finished);
}
public void testBadMethodWithEnoughTime() throws Exception {
SampleImpl target = new SampleImpl();
Sample proxy = service.newProxy(
target, Sample.class, ENOUGH_MS, TimeUnit.MILLISECONDS);
long start = System.nanoTime();
try {
proxy.sleepThenThrowException();
fail("no exception thrown");
} catch (SampleException expected) {
}
assertTheCallTookBetween(start, DELAY_MS, ENOUGH_MS);
}
public void testBadMethodWithNotEnoughTime() throws Exception {
SampleImpl target = new SampleImpl();
Sample proxy = service.newProxy(
target, Sample.class, NOT_ENOUGH_MS, TimeUnit.MILLISECONDS);
long start = System.nanoTime();
try {
proxy.sleepThenThrowException();
fail("no exception thrown");
} catch (UncheckedTimeoutException expected) {
}
assertTheCallTookBetween(start, NOT_ENOUGH_MS, DELAY_MS);
}
private static void assertTheCallTookBetween(
long startNanos, int atLeastMillis, int atMostMillis) {
long nanos = System.nanoTime() - startNanos;
assertTrue(nanos >= atLeastMillis * 1000000);
assertTrue(nanos <= atMostMillis * 1000000);
}
public interface Sample {
String sleepThenReturnInput(String input);
void sleepThenThrowException() throws SampleException;
}
@SuppressWarnings("serial")
public static class SampleException extends Exception { }
public static class SampleImpl implements Sample {
boolean finished;
@Override
public String sleepThenReturnInput(String input) {
try {
TimeUnit.MILLISECONDS.sleep(DELAY_MS);
finished = true;
return input;
} catch (InterruptedException e) {
return null;
}
}
@Override
public void sleepThenThrowException() throws SampleException {
try {
TimeUnit.MILLISECONDS.sleep(DELAY_MS);
} catch (InterruptedException e) {
}
throw new SampleException();
}
}
}