added tests for ThreadPoolTaskScheduler
This commit is contained in:
parent
970fcab5de
commit
50956e0d37
|
|
@ -0,0 +1,306 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2009 the original author or 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 org.springframework.scheduling.concurrent;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.ExecutionException;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.springframework.scheduling.Trigger;
|
||||||
|
import org.springframework.scheduling.TriggerContext;
|
||||||
|
import org.springframework.scheduling.support.ErrorHandler;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Mark Fisher
|
||||||
|
* @since 3.0
|
||||||
|
*/
|
||||||
|
public class ThreadPoolTaskSchedulerTests {
|
||||||
|
|
||||||
|
private static final String THREAD_NAME_PREFIX = "test-";
|
||||||
|
|
||||||
|
|
||||||
|
private final ThreadPoolTaskScheduler scheduler = new ThreadPoolTaskScheduler();
|
||||||
|
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void initScheduler() {
|
||||||
|
scheduler.setThreadNamePrefix(THREAD_NAME_PREFIX);
|
||||||
|
scheduler.afterPropertiesSet();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// test methods
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void executeRunnable() {
|
||||||
|
TestTask task = new TestTask(1);
|
||||||
|
scheduler.execute(task);
|
||||||
|
await(task);
|
||||||
|
assertThreadNamePrefix(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void executeFailingRunnableWithoutErrorHandler() {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
scheduler.execute(task);
|
||||||
|
// nothing to assert
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void executeFailingRunnnableWithErrorHandler() {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
TestErrorHandler errorHandler = new TestErrorHandler(1);
|
||||||
|
scheduler.setErrorHandler(errorHandler);
|
||||||
|
scheduler.execute(task);
|
||||||
|
await(errorHandler);
|
||||||
|
assertNotNull(errorHandler.lastError);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void submitRunnable() throws Exception {
|
||||||
|
TestTask task = new TestTask(1);
|
||||||
|
Future<?> future = scheduler.submit(task);
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertNull(result);
|
||||||
|
assertThreadNamePrefix(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = ExecutionException.class)
|
||||||
|
public void submitFailingRunnableWithoutErrorHandler() throws Exception {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
Future<?> future = scheduler.submit(task);
|
||||||
|
try {
|
||||||
|
future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
catch (ExecutionException e) {
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void submitFailingRunnableWithErrorHandler() throws Exception {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
TestErrorHandler errorHandler = new TestErrorHandler(1);
|
||||||
|
scheduler.setErrorHandler(errorHandler);
|
||||||
|
Future<?> future = scheduler.submit(task);
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
assertNull(result);
|
||||||
|
assertNotNull(errorHandler.lastError);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void submitCallable() throws Exception {
|
||||||
|
TestCallable task = new TestCallable(1);
|
||||||
|
Future<String> future = scheduler.submit(task);
|
||||||
|
String result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertEquals(THREAD_NAME_PREFIX, result.substring(0, THREAD_NAME_PREFIX.length()));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = ExecutionException.class)
|
||||||
|
public void submitFailingCallableWithoutErrorHandler() throws Exception {
|
||||||
|
TestCallable task = new TestCallable(0);
|
||||||
|
Future<String> future = scheduler.submit(task);
|
||||||
|
future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void submitFailingCallableWithErrorHandler() throws Exception {
|
||||||
|
TestCallable task = new TestCallable(0);
|
||||||
|
TestErrorHandler errorHandler = new TestErrorHandler(1);
|
||||||
|
scheduler.setErrorHandler(errorHandler);
|
||||||
|
Future<String> future = scheduler.submit(task);
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
assertNull(result);
|
||||||
|
assertNotNull(errorHandler.lastError);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void scheduleOneTimeTask() throws Exception {
|
||||||
|
TestTask task = new TestTask(1);
|
||||||
|
Future<?> future = scheduler.schedule(task, new Date());
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertNull(result);
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
assertThreadNamePrefix(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = ExecutionException.class)
|
||||||
|
public void scheduleOneTimeFailingTaskWithoutErrorHandler() throws Exception {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
Future<?> future = scheduler.schedule(task, new Date());
|
||||||
|
try {
|
||||||
|
future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
catch (ExecutionException e) {
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void scheduleOneTimeFailingTaskWithErrorHandler() throws Exception {
|
||||||
|
TestTask task = new TestTask(0);
|
||||||
|
TestErrorHandler errorHandler = new TestErrorHandler(1);
|
||||||
|
scheduler.setErrorHandler(errorHandler);
|
||||||
|
Future<?> future = scheduler.schedule(task, new Date());
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertTrue(future.isDone());
|
||||||
|
assertNull(result);
|
||||||
|
assertNotNull(errorHandler.lastError);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void scheduleTriggerTask() throws Exception {
|
||||||
|
TestTask task = new TestTask(3);
|
||||||
|
Future<?> future = scheduler.schedule(task, new TestTrigger(3));
|
||||||
|
Object result = future.get(1000, TimeUnit.MILLISECONDS);
|
||||||
|
assertNull(result);
|
||||||
|
await(task);
|
||||||
|
assertThreadNamePrefix(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void scheduleMultipleTriggerTasks() throws Exception {
|
||||||
|
for (int i = 0; i < 1000; i++) {
|
||||||
|
this.scheduleTriggerTask();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// utility methods
|
||||||
|
|
||||||
|
private void assertThreadNamePrefix(TestTask task) {
|
||||||
|
assertEquals(THREAD_NAME_PREFIX, task.lastThread.getName().substring(0, THREAD_NAME_PREFIX.length()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void await(TestTask task) {
|
||||||
|
this.await(task.latch);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void await(TestErrorHandler errorHandler) {
|
||||||
|
this.await(errorHandler.latch);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void await(CountDownLatch latch) {
|
||||||
|
try {
|
||||||
|
latch.await(1000, TimeUnit.MILLISECONDS);
|
||||||
|
}
|
||||||
|
catch (InterruptedException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
assertEquals("latch did not count down,", 0, latch.getCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// helper classes
|
||||||
|
|
||||||
|
private static class TestTask implements Runnable {
|
||||||
|
|
||||||
|
private final int expectedRunCount;
|
||||||
|
|
||||||
|
private final AtomicInteger actualRunCount = new AtomicInteger();
|
||||||
|
|
||||||
|
private final CountDownLatch latch;
|
||||||
|
|
||||||
|
private Thread lastThread;
|
||||||
|
|
||||||
|
TestTask(int expectedRunCount) {
|
||||||
|
this.expectedRunCount = expectedRunCount;
|
||||||
|
this.latch = new CountDownLatch(expectedRunCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void run() {
|
||||||
|
lastThread = Thread.currentThread();
|
||||||
|
if (actualRunCount.incrementAndGet() > expectedRunCount) {
|
||||||
|
throw new RuntimeException("intentional test failure");
|
||||||
|
}
|
||||||
|
latch.countDown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class TestCallable implements Callable<String> {
|
||||||
|
|
||||||
|
private final int expectedRunCount;
|
||||||
|
|
||||||
|
private final AtomicInteger actualRunCount = new AtomicInteger();
|
||||||
|
|
||||||
|
TestCallable(int expectedRunCount) {
|
||||||
|
this.expectedRunCount = expectedRunCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String call() throws Exception {
|
||||||
|
if (actualRunCount.incrementAndGet() > expectedRunCount) {
|
||||||
|
throw new RuntimeException("intentional test failure");
|
||||||
|
}
|
||||||
|
return Thread.currentThread().getName();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class TestErrorHandler implements ErrorHandler {
|
||||||
|
|
||||||
|
private final CountDownLatch latch;
|
||||||
|
|
||||||
|
private volatile Throwable lastError;
|
||||||
|
|
||||||
|
TestErrorHandler(int expectedErrorCount) {
|
||||||
|
this.latch = new CountDownLatch(expectedErrorCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void handleError(Throwable t) {
|
||||||
|
this.lastError = t;
|
||||||
|
this.latch.countDown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static class TestTrigger implements Trigger {
|
||||||
|
|
||||||
|
private final int maxRunCount;
|
||||||
|
|
||||||
|
private final AtomicInteger actualRunCount = new AtomicInteger();
|
||||||
|
|
||||||
|
TestTrigger(int maxRunCount) {
|
||||||
|
this.maxRunCount = maxRunCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date nextExecutionTime(TriggerContext triggerContext) {
|
||||||
|
if (this.actualRunCount.incrementAndGet() > this.maxRunCount) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return new Date();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue