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

org.restlet.service.TaskService Maven / Gradle / Ivy

Go to download

This OSGi bundle wraps org.restlet, and com.noelios.restlet ${pkgVersion} jar files.

The newest version!
/**
 * Copyright 2005-2008 Noelios Technologies.
 * 
 * The contents of this file are subject to the terms of the following open
 * source licenses: LGPL 3.0 or LGPL 2.1 or CDDL 1.0 (the "Licenses"). You can
 * select the license that you prefer but you may not use this file except in
 * compliance with one of these Licenses.
 * 
 * You can obtain a copy of the LGPL 3.0 license at
 * http://www.gnu.org/licenses/lgpl-3.0.html
 * 
 * You can obtain a copy of the LGPL 2.1 license at
 * http://www.gnu.org/licenses/lgpl-2.1.html
 * 
 * You can obtain a copy of the CDDL 1.0 license at
 * http://www.sun.com/cddl/cddl.html
 * 
 * See the Licenses for the specific language governing permissions and
 * limitations under the Licenses.
 * 
 * Alternatively, you can obtain a royaltee free commercial license with less
 * limitations, transferable or non-transferable, directly at
 * http://www.noelios.com/products/restlet-engine
 * 
 * Restlet is a registered trademark of Noelios Technologies.
 */

package org.restlet.service;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.AbstractExecutorService;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import org.restlet.Application;
import org.restlet.Context;
import org.restlet.VirtualHost;
import org.restlet.data.Response;

/**
 * Service capable of running tasks asynchronously. The service instance
 * returned will not invoke the runnable task in the current thread.
*
* In addition to allowing pooling, this method will ensure that the threads * executing the tasks will have the thread local variables copied from the * calling thread. This will ensure that call to static methods like * {@link Application#getCurrent()} still work.
*
* Also, note that this executor service will be shared among all Restlets and * Resources that are part of your context. In general this context corresponds * to a parent Application's context. If you want to have your own service * instance, you can use the {@link #wrap(ExecutorService)} method to ensure * that thread local variables are correctly set. * * @author Jerome Louvel * @author Doug Lea (docs of ExecutorService in public domain) * @author Tim Peierls */ public class TaskService extends Service implements ExecutorService { /** * The default thread factory * * @author Jerome Louvel * @author Tim Peierls */ private static class RestletThreadFactory implements ThreadFactory { final ThreadFactory factory = Executors.defaultThreadFactory(); public Thread newThread(Runnable runnable) { Thread t = factory.newThread(runnable); // Default factory is documented as producing names of the // form "pool-N-thread-M". t.setName(t.getName().replaceFirst("pool", "restlet")); return t; } } /** * Wraps a JDK executor service to ensure that the threads executing the * tasks will have the thread local variables copied from the calling * thread. This will ensure that call to static methods like * {@link Application#getCurrent()} still work. * * @param executorService * The JDK service to wrap. * @return The wrapper service to use. */ public static ExecutorService wrap(final ExecutorService executorService) { return new AbstractExecutorService() { public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { return executorService.awaitTermination(timeout, unit); } public void execute(final Runnable runnable) { // Save the thread local variables final Application currentApplication = Application.getCurrent(); final Context currentContext = Context.getCurrent(); final Integer currentVirtualHost = VirtualHost.getCurrent(); final Response currentResponse = Response.getCurrent(); executorService.execute(new Runnable() { public void run() { // Copy the thread local variables Response.setCurrent(currentResponse); Context.setCurrent(currentContext); VirtualHost.setCurrent(currentVirtualHost); Application.setCurrent(currentApplication); try { // Run the user task runnable.run(); } finally { // Reset the thread local variables Response.setCurrent(null); Context.setCurrent(null); VirtualHost.setCurrent(-1); Application.setCurrent(null); } } }); } public boolean isShutdown() { return executorService.isShutdown(); } public boolean isTerminated() { return executorService.isTerminated(); } public void shutdown() { executorService.shutdown(); } public List shutdownNow() { return executorService.shutdownNow(); } }; } /** * Allow {@link #shutdown()} and {@link #shutdownNow()} methods to * effectively shutdown the wrapped executor service. */ private volatile boolean shutdownAllowed; /** The wrapped JDK executor service. */ private volatile ExecutorService wrapped; /** * Constructor. */ public TaskService() { this.shutdownAllowed = false; setWrapped(wrap(createExecutorService())); } /** * Blocks until all tasks have completed execution after a shutdown request, * or the timeout occurs, or the current thread is interrupted, whichever * happens first. * * @param timeout * The maximum time to wait. * @param unit * The time unit. * @return True if this executor terminated and false if the timeout elapsed * before termination. */ public boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException { return getWrapped().awaitTermination(timeout, unit); } /** * Creates a new JDK executor service that will be wrapped. By default it * calls {@link Executors#newCachedThreadPool(ThreadFactory)}, passing the * result of {@link #createThreadFactory()} as a parameter. * * * @return A new JDK executor service. */ protected ExecutorService createExecutorService() { return Executors.newCachedThreadPool(createThreadFactory()); } /** * Creates a new thread factory that will properly name the Restlet created * threads with a "restlet-" prefix. * * @return A new thread factory. */ protected ThreadFactory createThreadFactory() { return new RestletThreadFactory(); } /** * Executes the given command asynchronously. * * @param command * The command to execute. */ public void execute(Runnable command) { getWrapped().execute(command); } /** * Returns the wrapped JDK executor service. * * @return The wrapped JDK executor service. */ private ExecutorService getWrapped() { return wrapped; } /** * Executes the given tasks, returning a list of Futures holding their * status and results when all complete. * * Due to a breaking change between Java SE versions 5 and 6, and in order * to maintain compatibility both at the source and binary level, we have * removed the generic information from this method. You can check the * {@link ExecutorService} interface for typing details. * * @param tasks * The task to execute. * @return The list of futures. */ @SuppressWarnings("unchecked") public List invokeAll(Collection tasks) throws InterruptedException { return getWrapped().invokeAll(tasks); } /** * Executes the given tasks, returning a list of Futures holding their * status and results when all complete or the timeout expires, whichever * happens first. Future.isDone() is true for each element of the returned * list. Upon return, tasks that have not completed are cancelled. Note that * a completed task could have terminated either normally or by throwing an * exception. The results of this method are undefined if the given * collection is modified while this operation is in progress. * * Due to a breaking change between Java SE versions 5 and 6, and in order * to maintain compatibility both at the source and binary level, we have * removed the generic information from this method. You can check the * {@link ExecutorService} interface for typing details. * * @param tasks * The task to execute. * @param timeout * The maximum time to wait. * @param unit * The time unit. * @return The list of futures. */ @SuppressWarnings("unchecked") public List invokeAll(Collection tasks, long timeout, TimeUnit unit) throws InterruptedException { return getWrapped().invokeAll(tasks, timeout, unit); } /** * Executes the given tasks, returning the result of one that has completed * successfully (i.e., without throwing an exception), if any do. Upon * normal or exceptional return, tasks that have not completed are * cancelled. The results of this method are undefined if the given * collection is modified while this operation is in progress. * * Due to a breaking change between Java SE versions 5 and 6, and in order * to maintain compatibility both at the source and binary level, we have * removed the generic information from this method. You can check the * {@link ExecutorService} interface for typing details. * * @param tasks * The task to execute. * @return The result returned by one of the tasks. */ @SuppressWarnings("unchecked") public Object invokeAny(Collection tasks) throws InterruptedException, ExecutionException { return getWrapped().invokeAny(tasks); } /** * Executes the given tasks, returning the result of one that has completed * successfully (i.e., without throwing an exception), if any do before the * given timeout elapses. Upon normal or exceptional return, tasks that have * not completed are cancelled. The results of this method are undefined if * the given collection is modified while this operation is in progress. * * Due to a breaking change between Java SE versions 5 and 6, and in order * to maintain compatibility both at the source and binary level, we have * removed the generic information from this method. You can check the * {@link ExecutorService} interface for typing details. * * @param tasks * The task to execute. * @param timeout * The maximum time to wait. * @param unit * The time unit. * @return The result returned by one of the tasks. */ @SuppressWarnings("unchecked") public Object invokeAny(Collection tasks, long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return getWrapped().invokeAny(tasks, timeout, unit); } /** * Returns true if this executor has been shut down. * * @return True if this executor has been shut down. */ public boolean isShutdown() { return getWrapped().isShutdown(); } /** * Indicates if the {@link #shutdown()} and {@link #shutdownNow()} methods * are allowed to effectively shutdown the wrapped executor service. Return * false by default. * * @return True if shutdown is allowed. */ public boolean isShutdownAllowed() { return shutdownAllowed; } /** * Returns true if all tasks have completed following shut down. Note that * isTerminated is never true unless either shutdown or shutdownNow was * called first. * * @return True if all tasks have completed following shut down. */ public boolean isTerminated() { return getWrapped().isTerminated(); } /** * Indicates if the {@link #shutdown()} and {@link #shutdownNow()} methods * are allowed to effectively shutdown the wrapped executor service. * * @param allowShutdown * True if shutdown is allowed. */ public void setShutdownAllowed(boolean allowShutdown) { this.shutdownAllowed = allowShutdown; } /** * Sets the wrapped JDK executor service. * * @param wrapped * The wrapped JDK executor service. */ private void setWrapped(ExecutorService wrapped) { this.wrapped = wrapped; } /** * Initiates an orderly shutdown in which previously submitted tasks are * executed, but no new tasks will be accepted. */ public void shutdown() { if (isShutdownAllowed()) { getWrapped().shutdown(); } } /** * Attempts to stop all actively executing tasks, halts the processing of * waiting tasks, and returns a list of the tasks that were awaiting * execution. * * @return The list of tasks that never commenced execution; */ public List shutdownNow() { return isShutdownAllowed() ? getWrapped().shutdownNow() : Collections . emptyList(); } @Override public synchronized void start() throws Exception { if (getWrapped().isShutdown()) { setWrapped(wrap(createExecutorService())); } super.start(); } @Override public synchronized void stop() throws Exception { super.stop(); if (!getWrapped().isShutdown()) { getWrapped().shutdown(); } } /** * Submits a value-returning task for execution and returns a Future * representing the pending results of the task. * * @param task * The task to submit. * @return A Future representing pending completion of the task, and whose * get() method will return the given result upon completion. */ public Future submit(Callable task) { return getWrapped().submit(task); } /** * * @param task * The task to submit. * @return A Future representing pending completion of the task, and whose * get() method will return the given result upon completion. */ public Future submit(Runnable task) { return getWrapped().submit(task); } /** * * @param task * The task to submit. * @param result * The result to return. * @return A Future representing pending completion of the task, and whose * get() method will return the given result upon completion. */ public Future submit(Runnable task, T result) { return getWrapped().submit(task, result); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy