Here's the big idea (how you use it):
import asyncio
from aiorun import run
async def main():
# Put your application code here
await asyncio.sleep(1.0)
if __name__ == '__main__':
run(main())
This package provides a run()
function as the starting point
of your asyncio
-based application.
The run()
function will handle everything that normally needs
to be done during the shutdown sequence of the application. All you
need to do is write your coroutines and run them.
So what the heck does run()
do exactly?? It does these standard,
idiomatic actions for asyncio apps:
- creates a
Task
for the given coroutine (schedules it on the event loop), - calls
loop.run_forever()
, - adds default (and smart) signal handlers for both
SIGINT
andSIGTERM
that will stop the loop; - and when the loop stops, the it will...
- ...gather all outstanding tasks,
- cancel them using
task.cancel()
, - wait for the executor to complete shutdown, and
- finally close the loop.
All of this stuff is boilerplate that you will never have to write
again. So, if you use aiorun
this is what you need to remember:
- Spawn all your work from a single, starting coroutine
- When a shutdown signal is received, all currently-pending tasks
will have
CancelledError
raised internally. It's up to you whether you want to handle this in atry/except
or not. - Try to have executor jobs be shortish, since shutdown will wait for them
to finish. If you need a long-running thread or process tasks, use
a dedicated thread/subprocess and set
daemon=True
instead.
There's not much else to know for general use. aiorun has a few special tools that you might need in unusual circumstances. These are discussed next.
It's unusual, but sometimes you're going to want a coroutine to not get
interrupted by cancellation during the shutdown sequence. You'll look in
the official docs and find asyncio.shield()
.
The problem is that shield()
doesn't work in shutdown scenarios because
the protection offered by shield()
only applies if the specific coroutine
inside which the shield()
is used, gets cancelled directly.
If, however, you go through a conventional shutdown sequence (like aiorun
is doing internally), you would call:
tasks = all_tasks()
, followed bygroup = gather(*tasks)
, and thengroup.cancel()
The problem is that shield() creates a secret, inner
task that will also be captured in the all_tasks()
call above, so it
will also receive a cancellation signal just like everything else.
Therefore, we have a version of shield()
that works better for us:
shutdown_waits_for()
. If you've got a coroutine that must not be
cancelled during the shutdown sequence, just wrap it in
shutdown_waits_for()
!
Here's an example:
import asyncio
from aiorun import run, shutdown_waits_for
async def corofn():
await asyncio.sleep(60)
print('done!')
async def main():
try:
await shutdown_waits_for(corofn())
except asyncio.CancelledError
print('oh noes!')
run(main())
If you run this program and do nothing, it'll run forever ('cause that's
how aiorun.run()
works) and you'll see only done!
printed in the output.
You'll have to send a signal or CTRL-C
to stop it, at which point
you'll see oh noes!
printed. So far no surprises.
If, however, you hit CTRL-C
before 60 seconds has passed, you will see
oh noes!
printed immediately, and then after 60 seconds (since start),
done!
is printed, and thereafter the program exits.
Behind the scenes, all tasks()
would have been cancelled by CTRL-C
,
except ones wrapped in shutdown_waits_for()
calls. In this respect, it
is loosely similar to asyncio.shield()
, but with special applicability
to our shutdown scenario in aiorun()
.
Oh, and you can use shutdown_waits_for()
as if it were asyncio.shield()
too. For that use-case it works the same. If you're using aiorun
, there
is no reason to use shield()
.