Refactor WorkQueue to support task inspection
By extending the ScheduledThreadPoolExecutor we are able to apply a
decoration to the tasks before they are added to the queue, giving
us the ability to pull them out at a later point in time and get any
data we want, in particular, the input Runnable.
Signed-off-by: Shawn O. Pearce <sop@google.com>
diff --git a/src/main/java/com/google/gerrit/git/WorkQueue.java b/src/main/java/com/google/gerrit/git/WorkQueue.java
index 95ad304..3c0a5d0 100644
--- a/src/main/java/com/google/gerrit/git/WorkQueue.java
+++ b/src/main/java/com/google/gerrit/git/WorkQueue.java
@@ -14,22 +14,52 @@
package com.google.gerrit.git;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.concurrent.Callable;
+import java.util.concurrent.Delayed;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.RunnableScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
+import java.util.concurrent.TimeoutException;
+/** Delayed execution of tasks using a background thread pool. */
public class WorkQueue {
- private static ScheduledThreadPoolExecutor pool;
+ private static Executor pool;
- public static synchronized void schedule(final Runnable task,
- final long delay, final TimeUnit unit) {
- if (pool == null) {
- pool = new ScheduledThreadPoolExecutor(1);
+ private static synchronized Executor getPool(final boolean autoStart) {
+ if (autoStart && pool == null) {
+ pool = new Executor(1);
pool.setKeepAliveTime(60, TimeUnit.SECONDS);
pool.setMaximumPoolSize(5);
}
- pool.schedule(task, delay, unit);
+ return pool;
}
+ /** Get all of the tasks currently scheduled in the work queue. */
+ public static Task<?>[] getTasks() {
+ final Executor p = getPool(false);
+ final Task<?>[] r;
+ r = p != null ? p.toTaskArray() : new Task[] {};
+ return r;
+ }
+
+ /**
+ * Schedule a task to run at a later point in time.
+ *
+ * @param task the task to invoke the {@code run()} method of later, on a
+ * background thread.
+ * @param delay amount to wait before calling the task. May be 0 to request
+ * "as soon as possible".
+ * @param unit time unit that {@code delay} is measured in.
+ */
+ public static void schedule(final Runnable task, final long delay,
+ final TimeUnit unit) {
+ getPool(true).schedule(task, delay, unit);
+ }
+
+ /** Shutdown the work queue, aborting any pending tasks that haven't started. */
public static void terminate() {
final ScheduledThreadPoolExecutor p = shutdown();
if (p != null) {
@@ -53,4 +83,157 @@
}
return null;
}
+
+ private static class Executor extends ScheduledThreadPoolExecutor {
+ private final List<Task<?>> active = new ArrayList<Task<?>>();
+
+ Executor(final int corePoolSize) {
+ super(corePoolSize);
+ }
+
+ @Override
+ protected <V> RunnableScheduledFuture<V> decorateTask(
+ final Runnable runnable, final RunnableScheduledFuture<V> task) {
+ return new Task<V>(runnable, super.decorateTask(runnable, task));
+ }
+
+ @Override
+ protected <V> RunnableScheduledFuture<V> decorateTask(
+ final Callable<V> callable, final RunnableScheduledFuture<V> task) {
+ throw new UnsupportedOperationException("Callable not implemented");
+ }
+
+ @Override
+ protected void beforeExecute(Thread t, Runnable r) {
+ super.beforeExecute(t, r);
+ synchronized (active) {
+ active.add((Task<?>) r);
+ }
+ }
+
+ @Override
+ protected void afterExecute(Runnable r, Throwable t) {
+ super.afterExecute(r, t);
+ synchronized (active) {
+ active.remove(r);
+ }
+ }
+
+ Task<?>[] toTaskArray() {
+ final List<Task<?>> list = new ArrayList<Task<?>>();
+ synchronized (active) {
+ list.addAll(active);
+ }
+ for (final Runnable task : getQueue()) { // iterator is thread safe
+ list.add((Task<?>) task);
+ }
+ return list.toArray(new Task<?>[list.size()]);
+ }
+ }
+
+ /** A wrapper around a scheduled Runnable, as maintained in the queue. */
+ public static class Task<V> implements RunnableScheduledFuture<V> {
+ /**
+ * Summarized status of a single task.
+ * <p>
+ * Tasks have the following state flow:
+ * <ol>
+ * <li>{@link #SLEEPING}: if scheduled with a non-zero delay.</li>
+ * <li>{@link #READY}: waiting for an available worker thread.</li>
+ * <li>{@link #RUNNING}: actively executing on a worker thread.</li>
+ * <li>{@link #DONE}: finished executing, if not periodic.</li>
+ * </ol>
+ */
+ public static enum State {
+ // Ordered like this so ordinal matches the order we would
+ // prefer to see tasks sorted in: done before running,
+ // running before ready, ready before sleeping.
+ //
+ DONE, CANCELLED, RUNNING, READY, SLEEPING, OTHER;
+ }
+
+ private final Runnable runnable;
+ private final RunnableScheduledFuture<V> task;
+ private volatile boolean running;
+
+ Task(Runnable runnable, RunnableScheduledFuture<V> task) {
+ this.runnable = runnable;
+ this.task = task;
+ }
+
+ /** Get the Runnable this task executes. */
+ public Runnable getRunnable() {
+ return runnable;
+ }
+
+ public State getState() {
+ if (isDone() && !isPeriodic()) {
+ return State.DONE;
+ } else if (isRunning()) {
+ return State.RUNNING;
+ } else if (isCancelled()) {
+ return State.CANCELLED;
+ }
+
+ final long delay = getDelay(TimeUnit.MILLISECONDS);
+ if (delay <= 0) {
+ return State.READY;
+ } else if (0 < delay) {
+ return State.SLEEPING;
+ }
+
+ return State.OTHER;
+ }
+
+ public boolean cancel(boolean mayInterruptIfRunning) {
+ return task.cancel(mayInterruptIfRunning);
+ }
+
+ public int compareTo(Delayed o) {
+ return task.compareTo(o);
+ }
+
+ public V get() throws InterruptedException, ExecutionException {
+ return task.get();
+ }
+
+ public V get(long timeout, TimeUnit unit) throws InterruptedException,
+ ExecutionException, TimeoutException {
+ return task.get(timeout, unit);
+ }
+
+ public long getDelay(TimeUnit unit) {
+ return task.getDelay(unit);
+ }
+
+ public boolean isCancelled() {
+ return task.isCancelled();
+ }
+
+ public boolean isRunning() {
+ return running;
+ }
+
+ public boolean isDone() {
+ return task.isDone();
+ }
+
+ public boolean isPeriodic() {
+ return task.isPeriodic();
+ }
+
+ public void run() {
+ try {
+ running = true;
+ task.run();
+ } finally {
+ running = false;
+ }
+ }
+
+ @Override
+ public String toString() {
+ return runnable.toString();
+ }
+ }
}