Runners¶
Source code: Lib/asyncio/runners.py
This section outlines high-level asyncio primitives to run asyncio code.
They are built on top of an event loop with the aim to simplify async code usage for common wide-spread scenarios.
Executando um programa asyncio¶
- asyncio.run(coro, *, debug=None, loop_factory=None)¶
- Execute coro in an asyncio event loop and return the result. - The argument can be any awaitable object. - This function runs the awaitable, taking care of managing the asyncio event loop, finalizing asynchronous generators, and closing the executor. - Esta função não pode ser chamada quando outro laço de eventos asyncio está executando na mesma thread. - If debug is - True, the event loop will be run in debug mode.- Falsedisables debug mode explicitly.- Noneis used to respect the global Modo de Depuração settings.- If loop_factory is not - None, it is used to create a new event loop; otherwise- asyncio.new_event_loop()is used. The loop is closed at the end. This function should be used as a main entry point for asyncio programs, and should ideally only be called once. It is recommended to use loop_factory to configure the event loop instead of policies. Passing- asyncio.EventLoopallows running asyncio without the policy system.- The executor is given a timeout duration of 5 minutes to shutdown. If the executor hasn’t finished within that duration, a warning is emitted and the executor is closed. - Exemplo: - async def main(): await asyncio.sleep(1) print('hello') asyncio.run(main()) - Adicionado na versão 3.7. - Alterado na versão 3.9: Atualizado para usar - loop.shutdown_default_executor().- Alterado na versão 3.10: debug is - Noneby default to respect the global debug mode settings.- Alterado na versão 3.12: Added loop_factory parameter. - Alterado na versão 3.14: coro can be any awaitable object. - Nota - The - asynciopolicy system is deprecated and will be removed in Python 3.16; from there on, an explicit loop_factory is needed to configure the event loop.
Runner context manager¶
- class asyncio.Runner(*, debug=None, loop_factory=None)¶
- A context manager that simplifies multiple async function calls in the same context. - Sometimes several top-level async functions should be called in the same event loop and - contextvars.Context.- If debug is - True, the event loop will be run in debug mode.- Falsedisables debug mode explicitly.- Noneis used to respect the global Modo de Depuração settings.- loop_factory could be used for overriding the loop creation. It is the responsibility of the loop_factory to set the created loop as the current one. By default - asyncio.new_event_loop()is used and set as current event loop with- asyncio.set_event_loop()if loop_factory is- None.- Basically, - asyncio.run()example can be rewritten with the runner usage:- async def main(): await asyncio.sleep(1) print('hello') with asyncio.Runner() as runner: runner.run(main()) - Adicionado na versão 3.11. - run(coro, *, context=None)¶
- Execute coro in the embedded event loop. - The argument can be any awaitable object. - If the argument is a coroutine, it is wrapped in a Task. - An optional keyword-only context argument allows specifying a custom - contextvars.Contextfor the code to run in. The runner’s default context is used if context is- None.- Returns the awaitable’s result or raises an exception. - Esta função não pode ser chamada quando outro laço de eventos asyncio está executando na mesma thread. - Alterado na versão 3.14: coro can be any awaitable object. 
 - close()¶
- Close the runner. - Finalize asynchronous generators, shutdown default executor, close the event loop and release embedded - contextvars.Context.
 - get_loop()¶
- Return the event loop associated with the runner instance. 
 - Nota - Runneruses the lazy initialization strategy, its constructor doesn’t initialize underlying low-level structures.- Embedded loop and context are created at the - withbody entering or the first call of- run()or- get_loop().
Handling Keyboard Interruption¶
Adicionado na versão 3.11.
When signal.SIGINT is raised by Ctrl-C, KeyboardInterrupt
exception is raised in the main thread by default. However this doesn’t work with
asyncio because it can interrupt asyncio internals and can hang the program from
exiting.
To mitigate this issue, asyncio handles signal.SIGINT as follows:
- asyncio.Runner.run()installs a custom- signal.SIGINThandler before any user code is executed and removes it when exiting from the function.
- The - Runnercreates the main task for the passed coroutine for its execution.
- When - signal.SIGINTis raised by Ctrl-C, the custom signal handler cancels the main task by calling- asyncio.Task.cancel()which raises- asyncio.CancelledErrorinside the main task. This causes the Python stack to unwind,- try/exceptand- try/finallyblocks can be used for resource cleanup. After the main task is cancelled,- asyncio.Runner.run()raises- KeyboardInterrupt.
- A user could write a tight loop which cannot be interrupted by - asyncio.Task.cancel(), in which case the second following Ctrl-C immediately raises the- KeyboardInterruptwithout cancelling the main task.