tornado.locks – Synchronization primitives¶
New in version 4.2.
Coordinate coroutines with synchronization primitives analogous to those the standard library provides to threads. These classes are very similar to those provided in the standard library’s asyncio package.
Warning
Note that these primitives are not actually thread-safe and cannot
be used in place of those from the standard library’s threading
module–they are meant to coordinate Tornado coroutines in a
single-threaded app, not to protect shared objects in a
multithreaded app.
Condition¶
- class tornado.locks.Condition[source]¶
A condition allows one or more coroutines to wait until notified.
Like a standard
threading.Condition, but does not need an underlying lock that is acquired and released.With a
Condition, coroutines can wait to be notified by other coroutines:import asyncio from tornado import gen from tornado.locks import Condition condition = Condition() async def waiter(): print("I'll wait right here") await condition.wait() print("I'm done waiting") async def notifier(): print("About to notify") condition.notify() print("Done notifying") async def runner(): # Wait for waiter() and notifier() in parallel await gen.multi([waiter(), notifier()]) asyncio.run(runner())
I'll wait right here About to notify Done notifying I'm done waiting
waittakes an optionaltimeoutargument, which is either an absolute timestamp:io_loop = IOLoop.current() # Wait up to 1 second for a notification. await condition.wait(timeout=io_loop.time() + 1)
…or a
datetime.timedeltafor a timeout relative to the current time:# Wait up to 1 second. await condition.wait(timeout=datetime.timedelta(seconds=1))
The method returns False if there’s no notification before the deadline.
Changed in version 5.0: Previously, waiters could be notified synchronously from within
notify. Now, the notification will always be received on the next iteration of theIOLoop.
Event¶
- class tornado.locks.Event[source]¶
An event blocks coroutines until its internal flag is set to True.
Similar to
threading.Event.A coroutine can wait for an event to be set. Once it is set, calls to
yield event.wait()will not block unless the event has been cleared:import asyncio from tornado import gen from tornado.locks import Event event = Event() async def waiter(): print("Waiting for event") await event.wait() print("Not waiting this time") await event.wait() print("Done") async def setter(): print("About to set the event") event.set() async def runner(): await gen.multi([waiter(), setter()]) asyncio.run(runner())
Waiting for event About to set the event Not waiting this time Done
- set() None[source]¶
Set the internal flag to
True. All waiters are awakened.Calling
waitonce the flag is set will not block.
Semaphore¶
- class tornado.locks.Semaphore(value: int = 1)[source]¶
A lock that can be acquired a fixed number of times before blocking.
A Semaphore manages a counter representing the number of
releasecalls minus the number ofacquirecalls, plus an initial value. Theacquiremethod blocks if necessary until it can return without making the counter negative.Semaphores limit access to a shared resource. To allow access for two workers at a time:
import asyncio from tornado import gen from tornado.locks import Semaphore sem = Semaphore(2) async def worker(worker_id): await sem.acquire() try: print("Worker %d is working" % worker_id) await use_some_resource() finally: print("Worker %d is done" % worker_id) sem.release() async def runner(): # Join all workers. await gen.multi([worker(i) for i in range(3)]) asyncio.run(runner())
Worker 0 is working Worker 1 is working Worker 0 is done Worker 2 is working Worker 1 is done Worker 2 is done
Workers 0 and 1 are allowed to run concurrently, but worker 2 waits until the semaphore has been released once, by worker 0.
The semaphore can be used as an async context manager:
async def worker(worker_id): async with sem: print("Worker %d is working" % worker_id) await use_some_resource() # Now the semaphore has been released. print("Worker %d is done" % worker_id)
For compatibility with older versions of Python,
acquireis a context manager, soworkercould also be written as:@gen.coroutine def worker(worker_id): with (yield sem.acquire()): print("Worker %d is working" % worker_id) yield use_some_resource() # Now the semaphore has been released. print("Worker %d is done" % worker_id)
Changed in version 4.3: Added
async withsupport in Python 3.5.
BoundedSemaphore¶
- class tornado.locks.BoundedSemaphore(value: int = 1)[source]¶
A semaphore that prevents release() being called too many times.
If
releasewould increment the semaphore’s value past the initial value, it raisesValueError. Semaphores are mostly used to guard resources with limited capacity, so a semaphore released too many times is a sign of a bug.
Lock¶
- class tornado.locks.Lock[source]¶
A lock for coroutines.
A Lock begins unlocked, and
acquirelocks it immediately. While it is locked, a coroutine that yieldsacquirewaits until another coroutine callsrelease.Releasing an unlocked lock raises
RuntimeError.A Lock can be used as an async context manager with the
async withstatement:>>> from tornado import locks >>> lock = locks.Lock() >>> >>> async def f(): ... async with lock: ... # Do something holding the lock. ... pass ... ... # Now the lock is released.
For compatibility with older versions of Python, the
acquiremethod asynchronously returns a regular context manager:>>> async def f2(): ... with (yield lock.acquire()): ... # Do something holding the lock. ... pass ... ... # Now the lock is released.
Changed in version 4.3: Added
async withsupport in Python 3.5.- acquire(timeout: Optional[Union[float, timedelta]] = None) Awaitable[_ReleasingContextManager][source]¶
Attempt to lock. Returns an awaitable.
Returns an awaitable, which raises
tornado.util.TimeoutErrorafter a timeout.
- release() None[source]¶
Unlock.
The first coroutine in line waiting for
acquiregets the lock.If not locked, raise a
RuntimeError.