A lot of the functions in asyncio
have deprecated loop
parameters, scheduled to be removed in Python 3.10. Examples include as_completed()
, sleep()
, and wait()
.
I'm looking for some historical context on these parameters and their removal.
loop
solve? Why would one have used it in the first place?loop
? Why is it being removed en masse?loop
, now that it's gone?The event loop is the core of every asyncio application. Event loops run asynchronous tasks and callbacks, perform network IO operations, and run subprocesses. Application developers should typically use the high-level asyncio functions, such as asyncio.
Run an asyncio Event Loop run_until_complete(<some Future object>) – this function runs a given Future object, usually a coroutine defined by the async / await pattern, until it's complete. run_forever() – this function runs the loop forever. stop() – the stop function stops a running loop.
run_in_executor is used to manage threads from within an event loop. To this end, it needs to wrap the thread into a Future, which needs to be assigned to an event loop (in one way or another). The reason the method is stored directly in a loop object is probably historical.
gather() method - It runs awaitable objects (objects which have await keyword) concurrently.
A lot of the functions in asyncio have deprecated loop parameters, scheduled to be removed in Python 3.10. Examples include as_completed (), sleep (), and wait (). I'm looking for some historical context on these parameters and their removal. What problems did loop solve? Why would one have used it in the first place? What was wrong with loop?
Prior to Python 3.6, asyncio.get_event_loop () was not guaranteed to return the event loop currently running when called from an asyncio coroutine or callback. It would return whatever event loop was previously set using set_event_loop (some_loop), or the one automatically created by asyncio.
Application developers should typically use the high-level asyncio functions, such as asyncio.run (), and should rarely need to reference the loop object or call its methods. This section is intended mostly for authors of lower-level code, libraries, and frameworks, who need finer control over the event loop behavior.
Using partial objects is usually more convenient than using lambdas, as asyncio can render partial objects better in debug and error messages. Event loop provides mechanisms to schedule callback functions to be called at some point in the future. Event loop uses monotonic clocks to track time.
What problems did
loop
solve? Why would one have used it in the first place?
Prior to Python 3.6, asyncio.get_event_loop()
was not guaranteed to return the event loop currently running when called from an asyncio coroutine or callback. It would return whatever event loop was previously set using set_event_loop(some_loop)
, or the one automatically created by asyncio. But sync code could easily create a different loop with another_loop = asyncio.new_event_loop()
and spin it up using another_loop.run_until_complete(some_coroutine())
. In this scenario, get_event_loop()
called inside some_coroutine
and the coroutines it awaits would return some_loop
rather than another_loop
. This kind of thing wouldn't occur when using asyncio casually, but it had to be accounted for by async libraries which couldn't assume that they were running under the default event loop. (For example, in tests or in some usages involving threads, one might want to spin up an event loop without disturbing the global setting with set_event_loop
.) The libraries would offer the explicit loop
argument where you'd pass another_loop
in the above case, and which you'd use whenever the running loop differed from the loop set up with asyncio.set_event_loop()
.
This issue would be fixed in Python 3.6 and 3.5.3, where get_event_loop()
was modified to reliably return the running loop if called from inside one, returning another_loop
in the above scenario. Python 3.7 would additionally introduced get_running_loop()
which completely ignores the global setting and always returns the currently running loop, raising an exception if not inside one. See this thread for the original discussion.
Once get_event_loop()
became reliable, another problem was that of performance. Since the event loop was needed for some very frequently used calls, most notably call_soon
, it was simply more efficient to pass around and cache the loop object. Asyncio itself did that, and many libraries followed suit. Eventually get_event_loop()
was accelerated in C and was no longer a bottleneck.
These two changes made the loop
arguments redundant.
What was wrong with
loop
? Why is it being removed en masse?
As any other redundancy, it complicates the API and opens up possibilities for errors. Async code should almost never just randomly communicate with a different loop, and now that get_event_loop()
is both correct and fast, there is no reason not to use it.
Also, passing the loop through all the layers of abstraction of a typical application is simply tedious. With async/await becoming mainstream in other languages, it has become clear that manually propagating a global object is not ergonomic and should not be required from programmers.
What replaces
loop
, now that it's gone?
Just use get_event_loop()
to get the loop when you need it. Alternatively, you can use get_running_loop()
to assert that a loop is running.
The need for accessing the event loop is somewhat reduced in Python 3.7, as some functions that were previously only available as methods on the loop, such as create_task
, are now available as stand-alone functions.
The loop
parameter was the way to pass the global event loop around. New implementations of the same functions no longer require you to pass the global event loop, they instead just request it where it's needed.
As the documentation suggests https://docs.python.org/3/library/asyncio-eventloop.html: "Application developers should typically use the high-level asyncio functions, such as asyncio.run(), and should rarely need to reference the loop object or call its methods."
Removing the need for you to pass it around to library functions aligns with that principle. The loop is not replaced, but its disappearance simply means you no longer have to deal with it 'manually'.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With