aioGo is the tiny helper library that doing two simple thinks:
- run your async code
and
- correct terminate your async code
Default python asyncio have troubles with this: when you hit Ctrl+C asyncio just cancel all your coroutines.
This can lead to many race conditions and hanging your application at exit point.
Install from pypi:
pip install aiogo
or, if you prefer poetry:
poetry add aiogo
And just:
from asyncio import Future
from aiogo import go
async def main(termination: Future[None]) -> None:
await termination
if __name__ == "__main__":
go(main)
That's all. You just wait while termination future was resolved (when your application got SIGINT or SIGTERM).
When you will be use this library in real world, just:
from asyncio import Future, wait, FIRST_COMPLETED, get_running_loop
async def main(termination: Future[None]) -> None:
your_awesome_read_task = get_running_loop().create_future()
done, _ = wait((termination, your_awesome_read_task), return_when=FIRST_COMPLETED)
and check done set.
That I have! Again, just:
from asyncio import Future
import uvloop # noqa
from aiogo import go
async def main(termination: Future[None]) -> None:
...
go(main, event_loop_factory=uvloop.new_event_loop)
If you can't write correct exit behaviour, well... Do that:
from asyncio import Future
from aiogo import go
async def main(termination: Future[None]) -> None:
...
go(main, exit_timeout=10)
And all your coroutines will be got CancelledError, as doing old good asyncio.run.
If you needed, you can make your own pool executor to use it as default executor, like this:
from asyncio import Future
from concurrent.futures import ThreadPoolExecutor
from aiogo import go
async def main(termination: Future[None]) -> None:
...
with ThreadPoolExecutor() as pool_executor:
go(main, default_executor=pool_executor)
Warning! Passed pool will be closed after main routine was has been terminated
Just use functools.partial method to bind coroutine to arguments
from asyncio import Future
from functools import partial
from aiogo import go
async def main(terminate: Future[None], value: int) -> None:
await terminate
go(partial(main, value=42))
BTW any value, that you return from main coroutine will be returned from go function.
- Make a repository fork
- Apply your changes (don't forget install pre-commit before making commit)
- Make a pull request