All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.dstadler.commons.testing.ThreadTestHelper Maven / Gradle / Ivy

There is a newer version: 1.3.4
Show newest version
package org.dstadler.commons.testing;

import static org.junit.Assert.assertEquals;

import java.lang.Thread.UncaughtExceptionHandler;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.CyclicBarrier;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.apache.commons.lang3.concurrent.BasicThreadFactory;
import org.dstadler.commons.logging.jdk.LoggerFactory;

/**
 * Helper class to test with many threads.
 *
 * Sample usage is as follows:
 *
 * 
    @Test
    public void testMultipleThreads() throws Throwable {
        ThreadTestHelper helper =
            new ThreadTestHelper(NUMBER_OF_THREADS, NUMBER_OF_TESTS);

        helper.executeTest(new ThreadTestHelper.TestRunnable() {
            @Override
            public void doEnd(int threadnum) throws Exception {
                // do stuff at the end ...
            }

            @Override
            public void run(int threadnum, int iter) throws Exception {
                // do the actual threaded work ...
            }
        });
    }
  
 */
public class ThreadTestHelper {

	private static Logger log = LoggerFactory.make();

	private final int threadCount;
	private final int testsPerThread;

	private volatile Throwable exception = null;
	private int executions[] = null;

	/**
	 * Initialize the class with the number of tests that should be executed
	 *
	 * @param threadCount
	 *        The number of threads to start running in parallel.
	 * @param testsPerThread
	 *        The number of single test-executions that are done in each
	 *        thread
	 */
	public ThreadTestHelper(int threadCount, int testsPerThread) {
		this.threadCount = threadCount;
		this.testsPerThread = testsPerThread;

		// Initialize array to allow to summarize afterwards
		executions = new int[threadCount];
	}

	public void executeTest(TestRunnable run) throws Throwable {
		log.fine("Starting thread test");

		List threads = new LinkedList<>();

		// start all threads
		for (int i = 0; i < threadCount; i++) {
			Thread t = startThread(i, run);
			threads.add(t);
		}

		// wait for all threads
		for (int i = 0; i < threadCount; i++) {
			threads.get(i).join();
		}

		// report exceptions if there were any
		if (exception != null) {
			throw exception;
		}

		// make sure the resulting number of executions is correct
		for (int i = 0; i < threadCount; i++) {
			// check if enough items were performed
			assertEquals("Thread " + i
					+ " did not execute all iterations", testsPerThread,
					executions[i]);
		}
	}

	/**
	 * This methods executes the passed {@link Callable}. The number of executions depends
	 * on the given runs number.
	 *
	 * @param testable test {@link Callable} to execute
	 * @param runs defines how many times the passed {@link Callable} is executed
	 * @return the results of the the execution of the passed {@link Callable}
	 * @throws Throwable if an exception happened during the execution of the {@link Callable}
	 */
	public static  List executeTest(final Callable testable, int runs) throws Throwable {
		final CyclicBarrier barrier = new CyclicBarrier(runs);
		ExecutorService executor = Executors.newCachedThreadPool(
				new BasicThreadFactory.Builder()
						.uncaughtExceptionHandler(new UncaughtExceptionHandler() {
							@Override
							public void uncaughtException(Thread t, Throwable e) {
								log.log(Level.SEVERE, "An uncaught exception happened in Thread " + t.getName(), e);
							}
						})
						.namingPattern(ThreadTestHelper.class.getSimpleName() + "-Thread-%d")
						.build());
		try {
			List> tasks = new ArrayList<>(runs);
			for (int i = 0; i < runs; i++) {
				tasks.add(new Callable() {
					@Override
					public T call() throws Exception {
						barrier.await(); // causes more contention
						return testable.call();
					}
				});
			}

			List> futures = executor.invokeAll(tasks);
			List results = new ArrayList<>(futures.size());
			for (Future future : futures) {
				results.add(future.get());
			}
			return results;
		} catch (ExecutionException e) {
			throw e.getCause();
		} finally {
			executor.shutdownNow();
			executor = null;
		}
	}

	/**
	 * This method is executed to start one thread. The thread will execute the
	 * provided runnable a number of times.
	 *
	 * @param threadnum
	 *        The number of this thread
	 * @param run
	 *        The Runnable object that is used to perform the actual test
	 *        operation
	 *
	 * @return The thread that was started.
	 *
	 */
	private Thread startThread(final int threadnum, final TestRunnable run) {
		log.fine("Starting thread number: " + threadnum);

		Thread t1 = null;
		t1 = new Thread(new Runnable() {

			@Override
			public void run() {
				try {
					for (int iter = 0; iter < testsPerThread && exception == null; iter++) {
						// log.fine("Executing iteration " + iter +
						// " in thread" +
						// Thread.currentThread().getName());

						// call the actual testcode
						run.run(threadnum, iter);

						executions[threadnum]++;
					}

					// do end-work here, we don't do this in a finally as we log
					// Exception
					// then anyway
					run.doEnd(threadnum);
				} catch (Throwable e) {
					// log.log(Level.SEVERE, "Caught unexpected Throable", e);
					exception = e;
				}

			}
		}, "ThreadTestHelper-Thread " + threadnum + ": " + run.getClass().getName());

		t1.start();

		return t1;
	}

	public interface TestRunnable {

		/**
		 * When an object implementing interface Runnable is used
		 * to create a thread, starting the thread causes the object's run method to be called in that separately
		 * executing
		 * thread.
		 * 

* The general contract of the method run is that it may take any action whatsoever. * * @param threadnum * The number of the thread executing this run() * @param iter * The count of how many times this thread executed the * method * * @see java.lang.Thread#run() */ public abstract void run(int threadnum, int iter) throws Exception; /** * Perform any action that should be done at the end. * * This method should throw an Exception if any check fails at this * point. */ void doEnd(int threadnum) throws Exception; } public void testDummy() { // small empty test to not fail if this class is executed as test case // by // Hudson/Sonar } /** * Wait for all threads with the specified name to finish, i.e. to not appear in the * list of running threads any more. * * @param name * * @throws InterruptedException * @author dominik.stadler */ public static void waitForThreadToFinish(final String name) throws InterruptedException { int count = Thread.currentThread().getThreadGroup().activeCount(); Thread[] threads = new Thread[count]; Thread.currentThread().getThreadGroup().enumerate(threads); for (Thread t : threads) { if (t != null && name.equals(t.getName())) { t.join(); } } } /** * Wait for thread whose name contain the specified string to finish, i.e. to not appear in the * list of running threads any more. * * @param name * * @throws InterruptedException * @author dominik.stadler */ public static void waitForThreadToFinishSubstring(final String name) throws InterruptedException { int count = Thread.currentThread().getThreadGroup().activeCount(); Thread[] threads = new Thread[count]; Thread.currentThread().getThreadGroup().enumerate(threads); for (Thread t : threads) { if (t != null && t.getName().contains(name)) { t.join(); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy