tornado.concurrent — Work with threads and futures¶
Utilities for working with threads and Futures.
Futures are a pattern for concurrent programming introduced in
Python 3.2 in the concurrent.futures package. This package defines
a mostly-compatible Future class designed for use from coroutines,
as well as some utility functions for interacting with the
concurrent.futures package.
-
class
tornado.concurrent.Future[source]¶ Placeholder for an asynchronous result.
A
Futureencapsulates the result of an asynchronous operation. In synchronous applicationsFuturesare used to wait for the result from a thread or process pool; in Tornado they are normally used withIOLoop.add_futureor by yielding them in agen.coroutine.tornado.concurrent.Futureis similar toconcurrent.futures.Future, but not thread-safe (and therefore faster for use with single-threaded event loops).In addition to
exceptionandset_exception, methodsexc_infoandset_exc_infoare supported to capture tracebacks in Python 2. The traceback is automatically available in Python 3, but in the Python 2 futures backport this information is discarded. This functionality was previously available in a separate classTracebackFuture, which is now a deprecated alias for this class.Changed in version 4.0:
tornado.concurrent.Futureis always a thread-unsafeFuturewith support for theexc_infomethods. Previously it would be an alias for the thread-safeconcurrent.futures.Futureif that package was available and fall back to the thread-unsafe implementation if it was not.Changed in version 4.1: If a
Futurecontains an error but that error is never observed (by callingresult(),exception(), orexc_info()), a stack trace will be logged when theFutureis garbage collected. This normally indicates an error in the application, but in cases where it results in undesired logging it may be necessary to suppress the logging by ensuring that the exception is observed:f.add_done_callback(lambda f: f.exception()).
Consumer methods¶
-
Future.result(timeout=None)[source]¶ If the operation succeeded, return its result. If it failed, re-raise its exception.
This method takes a
timeoutargument for compatibility withconcurrent.futures.Futurebut it is an error to call it before theFutureis done, so thetimeoutis never used.
-
Future.exception(timeout=None)[source]¶ If the operation raised an exception, return the
Exceptionobject. Otherwise returns None.This method takes a
timeoutargument for compatibility withconcurrent.futures.Futurebut it is an error to call it before theFutureis done, so thetimeoutis never used.
-
Future.exc_info()[source]¶ Returns a tuple in the same format as
sys.exc_infoor None.New in version 4.0.
-
Future.add_done_callback(fn)[source]¶ Attaches the given callback to the
Future.It will be invoked with the
Futureas its argument when the Future has finished running and its result is available. In Tornado consider usingIOLoop.add_futureinstead of callingadd_done_callbackdirectly.
Producer methods¶
-
Future.set_result(result)[source]¶ Sets the result of a
Future.It is undefined to call any of the
setmethods more than once on the same object.
-
Future.set_exc_info(exc_info)[source]¶ Sets the exception information of a
Future.Preserves tracebacks on Python 2.
New in version 4.0.
-
tornado.concurrent.run_on_executor(*args, **kwargs)[source]¶ Decorator to run a synchronous method asynchronously on an executor.
The decorated method may be called with a
callbackkeyword argument and returns a future.The
IOLoopand executor to be used are determined by theio_loopandexecutorattributes ofself. To use different attributes, pass keyword arguments to the decorator:@run_on_executor(executor='_thread_pool') def foo(self): pass
Changed in version 4.2: Added keyword arguments to use alternative attributes.
-
tornado.concurrent.return_future(f)[source]¶ Decorator to make a function that returns via callback return a
Future.The wrapped function should take a
callbackkeyword argument and invoke it with one argument when it has finished. To signal failure, the function can simply raise an exception (which will be captured by theStackContextand passed along to theFuture).From the caller’s perspective, the callback argument is optional. If one is given, it will be invoked when the function is complete with
Future.result()as an argument. If the function fails, the callback will not be run and an exception will be raised into the surroundingStackContext.If no callback is given, the caller should use the
Futureto wait for the function to complete (perhaps by yielding it in agen.enginefunction, or passing it toIOLoop.add_future).Usage:
@return_future def future_func(arg1, arg2, callback): # Do stuff (possibly asynchronous) callback(result) @gen.engine def caller(callback): yield future_func(arg1, arg2) callback()
Note that
@return_futureand@gen.enginecan be applied to the same function, provided@return_futureappears first. However, consider using@gen.coroutineinstead of this combination.