tornado.ioloop — Main event loop¶
An I/O event loop for non-blocking sockets.
In Tornado 6.0, IOLoop is a wrapper around the asyncio event loop, with a
slightly different interface. The IOLoop interface is now provided primarily
for backwards compatibility; new code should generally use the asyncio event
loop interface directly. The IOLoop.current class method provides the
IOLoop instance corresponding to the running asyncio event loop.
IOLoop objects¶
- class tornado.ioloop.IOLoop(*args: Any, **kwargs: Any)[source]¶
An I/O event loop.
As of Tornado 6.0,
IOLoopis a wrapper around theasyncioevent loop.Example usage for a simple TCP server:
import asyncio import errno import functools import socket import tornado from tornado.iostream import IOStream async def handle_connection(connection, address): stream = IOStream(connection) message = await stream.read_until_close() print("message from client:", message.decode().strip()) def connection_ready(sock, fd, events): while True: try: connection, address = sock.accept() except BlockingIOError: return connection.setblocking(0) io_loop = tornado.ioloop.IOLoop.current() io_loop.spawn_callback(handle_connection, connection, address) async def main(): sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM, 0) sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) sock.setblocking(0) sock.bind(("", 8888)) sock.listen(128) io_loop = tornado.ioloop.IOLoop.current() callback = functools.partial(connection_ready, sock) io_loop.add_handler(sock.fileno(), callback, io_loop.READ) await asyncio.Event().wait() if __name__ == "__main__": asyncio.run(main())
Most applications should not attempt to construct an
IOLoopdirectly, and instead initialize theasyncioevent loop and useIOLoop.current(). In some cases, such as in test frameworks when initializing anIOLoopto be run in a secondary thread, it may be appropriate to construct anIOLoopwithIOLoop(make_current=False).In general, an
IOLoopcannot survive a fork or be shared across processes in any way. When multiple processes are being used, each process should create its ownIOLoop, which also implies that any objects which depend on theIOLoop(such asAsyncHTTPClient) must also be created in the child processes. As a guideline, anything that starts processes (including thetornado.processandmultiprocessingmodules) should do so as early as possible, ideally the first thing the application does after loading its configuration, and before any calls toIOLoop.startorasyncio.run.Changed in version 4.2: Added the
make_currentkeyword argument to theIOLoopconstructor.Changed in version 5.0: Uses the
asyncioevent loop by default. TheIOLoop.configuremethod cannot be used on Python 3 except to redundantly specify theasyncioevent loop.Changed in version 6.3:
make_current=Trueis now the default when creating an IOLoop - previously the default was to make the event loop current if there wasn’t already a current one.
Running an IOLoop¶
- static IOLoop.current() IOLoop[source]¶
- static IOLoop.current(instance: bool = True) IOLoop | None
Returns the current thread’s
IOLoop.If an
IOLoopis currently running or has been marked as current bymake_current, returns that instance. If there is no currentIOLoopandinstanceis true, creates one.Changed in version 4.1: Added
instanceargument to control the fallback toIOLoop.instance().Changed in version 5.0: On Python 3, control of the current
IOLoopis delegated toasyncio, with this and other methods as pass-through accessors. Theinstanceargument now controls whether anIOLoopis created automatically when there is none, instead of whether we fall back toIOLoop.instance()(which is now an alias for this method).instance=Falseis deprecated, since even if we do not create anIOLoop, this method may initialize the asyncio loop.Deprecated since version 6.2: It is deprecated to call
IOLoop.current()when noasyncioevent loop is running.
- IOLoop.make_current() None[source]¶
Makes this the
IOLoopfor the current thread.An
IOLoopautomatically becomes current for its thread when it is started, but it is sometimes useful to callmake_currentexplicitly before starting theIOLoop, so that code run at startup time can find the right instance.Changed in version 4.1: An
IOLoopcreated while there is no currentIOLoopwill automatically become current.Changed in version 5.0: This method also sets the current
asyncioevent loop.Deprecated since version 6.2: Setting and clearing the current event loop through Tornado is deprecated. Use
asyncio.set_event_loopinstead if you need this.
- static IOLoop.clear_current() None[source]¶
Clears the
IOLoopfor the current thread.Intended primarily for use by test frameworks in between tests.
Changed in version 5.0: This method also clears the current
asyncioevent loop.Deprecated since version 6.2.
- IOLoop.start() None[source]¶
Starts the I/O loop.
The loop will run until one of the callbacks calls
stop(), which will make the loop stop after the current event iteration completes.
- IOLoop.stop() None[source]¶
Stop the I/O loop.
If the event loop is not currently running, the next call to
start()will return immediately.Note that even after
stophas been called, theIOLoopis not completely stopped untilIOLoop.starthas also returned. Some work that was scheduled before the call tostopmay still be run before theIOLoopshuts down.
- IOLoop.run_sync(func: Callable, timeout: float | None = None) Any[source]¶
Starts the
IOLoop, runs the given function, and stops the loop.The function must return either an awaitable object or
None. If the function returns an awaitable object, theIOLoopwill run until the awaitable is resolved (andrun_sync()will return the awaitable’s result). If it raises an exception, theIOLoopwill stop and the exception will be re-raised to the caller.The keyword-only argument
timeoutmay be used to set a maximum duration for the function. If the timeout expires, aasyncio.TimeoutErroris raised.This method is useful to allow asynchronous calls in a
main()function:async def main(): # do stuff... if __name__ == '__main__': IOLoop.current().run_sync(main)
Changed in version 4.3: Returning a non-
None, non-awaitable value is now an error.Changed in version 5.0: If a timeout occurs, the
funccoroutine will be cancelled.Changed in version 6.2:
tornado.util.TimeoutErroris now an alias toasyncio.TimeoutError.
- IOLoop.close(all_fds: bool = False) None[source]¶
Closes the
IOLoop, freeing any resources used.If
all_fdsis true, all file descriptors registered on the IOLoop will be closed (not just the ones created by theIOLoopitself).Many applications will only use a single
IOLoopthat runs for the entire lifetime of the process. In that case closing theIOLoopis not necessary since everything will be cleaned up when the process exits.IOLoop.closeis provided mainly for scenarios such as unit tests, which create and destroy a large number ofIOLoops.An
IOLoopmust be completely stopped before it can be closed. This means thatIOLoop.stop()must be called andIOLoop.start()must be allowed to return before attempting to callIOLoop.close(). Therefore the call toclosewill usually appear just after the call tostartrather than near the call tostop.Changed in version 3.1: If the
IOLoopimplementation supports non-integer objects for “file descriptors”, those objects will have theirclosemethod whenall_fdsis true.
- static IOLoop.instance() IOLoop[source]¶
Deprecated alias for
IOLoop.current().Changed in version 5.0: Previously, this method returned a global singleton
IOLoop, in contrast with the per-threadIOLoopreturned bycurrent(). In nearly all cases the two were the same (when they differed, it was generally used from non-Tornado threads to communicate back to the main thread’sIOLoop). This distinction is not present inasyncio, so in order to facilitate integration with that packageinstance()was changed to be an alias tocurrent(). Applications using the cross-thread communications aspect ofinstance()should instead set their own global variable to point to theIOLoopthey want to use.Deprecated since version 5.0.
- IOLoop.install() None[source]¶
Deprecated alias for
make_current().Changed in version 5.0: Previously, this method would set this
IOLoopas the global singleton used byIOLoop.instance(). Now thatinstance()is an alias forcurrent(),install()is an alias formake_current().Deprecated since version 5.0.
- static IOLoop.clear_instance() None[source]¶
Deprecated alias for
clear_current().Changed in version 5.0: Previously, this method would clear the
IOLoopused as the global singleton byIOLoop.instance(). Now thatinstance()is an alias forcurrent(),clear_instance()is an alias forclear_current().Deprecated since version 5.0.
I/O events¶
- IOLoop.add_handler(fd: int, handler: Callable[[int, int], None], events: int) None[source]¶
- IOLoop.add_handler(fd: _S, handler: Callable[[_S, int], None], events: int) None
Registers the given handler to receive the given events for
fd.The
fdargument may either be an integer file descriptor or a file-like object with afileno()andclose()method.The
eventsargument is a bitwise or of the constantsIOLoop.READ,IOLoop.WRITE, andIOLoop.ERROR.When an event occurs,
handler(fd, events)will be run.Changed in version 4.0: Added the ability to pass file-like objects in addition to raw file descriptors.
Callbacks and timeouts¶
- IOLoop.add_callback(callback: Callable, *args: Any, **kwargs: Any) None[source]¶
Calls the given callback on the next I/O loop iteration.
It is safe to call this method from any thread at any time, except from a signal handler. Note that this is the only method in
IOLoopthat makes this thread-safety guarantee; all other interaction with theIOLoopmust be done from thatIOLoop’s thread.add_callback()may be used to transfer control from other threads to theIOLoop’s thread.
- IOLoop.add_callback_from_signal(callback: Callable, *args: Any, **kwargs: Any) None[source]¶
Calls the given callback on the next I/O loop iteration.
Intended to be afe for use from a Python signal handler; should not be used otherwise.
Deprecated since version 6.4: Use
asyncio.AbstractEventLoop.add_signal_handlerinstead. This method is suspected to have been broken since Tornado 5.0 and will be removed in version 7.0.
- IOLoop.add_future(future: Future[_T] | Future[_T], callback: Callable[[Future[_T]], None]) None[source]¶
Schedules a callback on the
IOLoopwhen the givenFutureis finished.The callback is invoked with one argument, the
Future.This method only accepts
Futureobjects and not other awaitables (unlike most of Tornado where the two are interchangeable).
- IOLoop.add_timeout(deadline: float | timedelta, callback: Callable, *args: Any, **kwargs: Any) object[source]¶
Runs the
callbackat the timedeadlinefrom the I/O loop.Returns an opaque handle that may be passed to
remove_timeoutto cancel.deadlinemay be a number denoting a time (on the same scale asIOLoop.time, normallytime.time), or adatetime.timedeltaobject for a deadline relative to the current time. Since Tornado 4.0,call_lateris a more convenient alternative for the relative case since it does not require a timedelta object.Note that it is not safe to call
add_timeoutfrom other threads. Instead, you must useadd_callbackto transfer control to theIOLoop’s thread, and then calladd_timeoutfrom there.Subclasses of IOLoop must implement either
add_timeoutorcall_at; the default implementations of each will call the other.call_atis usually easier to implement, but subclasses that wish to maintain compatibility with Tornado versions prior to 4.0 must useadd_timeoutinstead.Changed in version 4.0: Now passes through
*argsand**kwargsto the callback.
- IOLoop.call_at(when: float, callback: Callable, *args: Any, **kwargs: Any) object[source]¶
Runs the
callbackat the absolute time designated bywhen.whenmust be a number using the same reference point asIOLoop.time.Returns an opaque handle that may be passed to
remove_timeoutto cancel. Note that unlike theasynciomethod of the same name, the returned object does not have acancel()method.See
add_timeoutfor comments on thread-safety and subclassing.Added in version 4.0.
- IOLoop.call_later(delay: float, callback: Callable, *args: Any, **kwargs: Any) object[source]¶
Runs the
callbackafterdelayseconds have passed.Returns an opaque handle that may be passed to
remove_timeoutto cancel. Note that unlike theasynciomethod of the same name, the returned object does not have acancel()method.See
add_timeoutfor comments on thread-safety and subclassing.Added in version 4.0.
- IOLoop.remove_timeout(timeout: object) None[source]¶
Cancels a pending timeout.
The argument is a handle as returned by
add_timeout. It is safe to callremove_timeouteven if the callback has already been run.
- IOLoop.spawn_callback(callback: Callable, *args: Any, **kwargs: Any) None[source]¶
Calls the given callback on the next IOLoop iteration.
As of Tornado 6.0, this method is equivalent to
add_callback.Added in version 4.0.
- IOLoop.run_in_executor(executor: Executor | None, func: Callable[[...], _T], *args: Any) Future[_T][source]¶
Runs a function in a
concurrent.futures.Executor. IfexecutorisNone, the IO loop’s default executor will be used.Use
functools.partialto pass keyword arguments tofunc.Added in version 5.0.
- IOLoop.set_default_executor(executor: Executor) None[source]¶
Sets the default executor to use with
run_in_executor().Added in version 5.0.
- IOLoop.time() float[source]¶
Returns the current time according to the
IOLoop’s clock.The return value is a floating-point number relative to an unspecified time in the past.
Historically, the IOLoop could be customized to use e.g.
time.monotonicinstead oftime.time, but this is not currently supported and so this method is equivalent totime.time.
- class tornado.ioloop.PeriodicCallback(callback: Callable[[], Awaitable | None], callback_time: timedelta | float, jitter: float = 0)[source]¶
Schedules the given callback to be called periodically.
The callback is called every
callback_timemilliseconds whencallback_timeis a float. Note that the timeout is given in milliseconds, while most other time-related functions in Tornado use seconds.callback_timemay alternatively be given as adatetime.timedeltaobject.If
jitteris specified, each callback time will be randomly selected within a window ofjitter * callback_timemilliseconds. Jitter can be used to reduce alignment of events with similar periods. A jitter of 0.1 means allowing a 10% variation in callback time. The window is centered oncallback_timeso the total number of calls within a given interval should not be significantly affected by adding jitter.If the callback runs for longer than
callback_timemilliseconds, subsequent invocations will be skipped to get back on schedule.startmust be called after thePeriodicCallbackis created.Changed in version 5.0: The
io_loopargument (deprecated since version 4.1) has been removed.Changed in version 5.1: The
jitterargument is added.Changed in version 6.2: If the
callbackargument is a coroutine, and a callback runs for longer thancallback_time, subsequent invocations will be skipped. Previously this was only true for regular functions, not coroutines, which were “fire-and-forget” forPeriodicCallback.The
callback_timeargument now acceptsdatetime.timedeltaobjects, in addition to the previous numeric milliseconds.- is_running() bool[source]¶
Returns
Trueif thisPeriodicCallbackhas been started.Added in version 4.1.