添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement . We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Since 3.10 asyncio.get_event_loop() emits a deprecation warning if used outside of the event loop (see #83710 ). It is a time to turn a warning into error and make asyncio.get_event_loop() an alias of asyncio.get_running_loop() .

But maybe we should first deprecate set_event_loop() ? It will be a no-op now.

Linked PRs

  • gh-93453: No longer create an event loop in get_event_loop() #98440
  • [3.11] gh-93453: Only emit deprecation warning in asyncio.get_event_loop when a new event loop is created #99949
  • [3.10] gh-93453: Only emit deprecation warning in asyncio.get_event_loop when a new event loop is created #100059
  • I wonder a bit how much existing code will be broken by this change in Python 3.12?

    If the deprecation warning in Python 3.10-11 remains unnoticed (like for me, until I ran Python with the -Werror flag), any existing code calling aysncio.get_event_loop() outside a running event loop will break.

    A simple workaround that worked in my case is to call asyncio.get_event_loop_policy().get_event_loop() instead. This will return the loop already attached to the current thread, or create a new one if not existing yet. (This seems better than just calling asyncio.new_event_loop() because that creates an extra loop next to already existing loops.)

    Will this workaround of using asyncio.get_event_loop_policy().get_event_loop() remain available in Python 3.12+?

    Ask @asvetlov . I do not know plans about asyncio.get_event_loop_policy().get_event_loop() . It may be deprecated too: you either use the existing running loop, or explicitly create a new one.

    asyncio.get_event_loop() was initially proposed to remove, but since it is use so much in the code that predates asyncio.get_running_loop() , it was decided to make it just an alias of asyncio.get_running_loop() .

    Thanks as well. It's good to have something next to asyncio.get_running_loop() because that can be only be called from a running loop;) For lower-level usage of asyncio 's event loops, accessing a loop even when it's not running seems still very useful to me. (For higher-level usage stricter, disciplined calls using asyncio.run() etc., not handling e.g. any bare asyncio.Future s is indeed advised.)

    For example, I want to call loop.set_exception_handler(handler) to set my own exception handler for the event loop. To do this once and for all, it's convenient to do this before the loop is running. Not sure what the intended way to do this would be, if one can only access the loop after it started running?

    @lschoe I think the intended pattern is

    def loop_factory():
        loop = asyncio.new_event_loop()
        loop.set_exception_handler(...)
    with asyncio.Runner(loop_factory=loop_factory) as runner:
        runner.run(amain())
    with asyncio.Runner() as runner:
        runner.get_loop().set_exception_handler(...)
        runner.run(amain())

    But calling asyncio.get_running_loop().set_exception_handler(...) as the first thing in your async def amain(): seems to be the best choice

    Thanks, these are interesting options, indeed for Python 3.11+ which will have this new Runner class.

    Using a loop_factory() does not seem preferable because asyncio.new_event_loop() is still called, which may cause problems.
    If a loop is already attached to the current thread, this will create a new loop next to the existing one. However, only one of these two loops can run at the same time, so this gives a Runtime error if one tries to run the other one.

    With asyncio.get_event_loop() you can join the existing loop, if any. Whether that loop is running or not doesn't matter either.

    For example, one can run some code in Python's asyncio REPL (via python -m asyncio ) or in Jupyter notebooks, where there is already a loop running providing top-level await, or one can run the same code from a Python script where there is no loop running yet.

    I've also tried the second way you propose, calling runner.get_loop() , but this also gives a Runtime error when called from the asyncio REPL, for example.

    And your third option requires a running loop again (which will not be there yet, if this code is executed upon importing and initializing a module).

    To deal with such varying circumstances, global access to the event loop attached to the current thread via asyncio.get_event_loop() is very useful.

    This is a subtle issue and I would like to go slowly here (but not so slowly to miss the 3.12 feature cut-off!).

    I wasn't aware of this issue and need some time to think about the consequences.

    The growing asymmetry between get_event_loop() and set_event_loop() is definitely bothering me.

    Should we perhaps also deprecate the behavior of Policy.get_event_loop() to sometimes create a new event loop?

    In response to @lschoe 's comments (e.g. #93453 (comment) ), this is indeed a reason to go slowly.

    I wonder what the remaining use cases are for accessing the current thread's loop regardless of whether it's running or not -- is it just loop.set_exception_handler() , or are there other configurations as well that some folks prefer to do before the loop is running?

    If so, maybe we could recommend loop = asyncio.get_event_loop_policy().get_event_loop() , which has the advantage that it works in older versions too. Maybe we can keep this and policy.new_event_loop() but deprecate set_event_loop_policy() right away? That way eventually the policy is just a singleton that holds some global state in a backwards compatible fashion.

    The awkward thing is that the use case requires that we still call set_event_loop() , for the rare cases where someone wants to configure something for the current thread's loop before it is running. How can we convince users not to do that but instead configure the loop once it is running? It does seem a bit clumsy if you want to run the same code in an asyncio-enabled REPL and when using a runner.

    I would very much like to believe that this is a very very small minority use case or that it's just a bad practice, but nothing I've read here convinces me of that. @lschoe can you point to real-world code that uses this pattern?

    the rare cases where someone wants to configure something for the current thread's loop before it is running

    This doesn't work when anything uses asyncio.run, because the first thing asyncio.run does is make a brand new event loop and the last thing it does is set the global event loop to None, which makes subsequent get_event_loop calls crash

    This doesn't work when anything uses asyncio.run, because the first thing asyncio.run does is make a brand new event loop and the last thing it does is set the global event loop to None, which makes subsequent get_event_loop calls crash

    Yeah, we need a better understanding of @lschoe 's use case.

    The backdrop for the use case is a bit extensive, so I'll try to extract the relevant points for your discussion as directly as possible.

    The perspective is that of a (Python package) developer of code that heavily relies on asyncio and its event loop, using lots of Futures, Tasks, coroutines etc. The use of these asyncio primitives, and Futures in particular, is hidden from the end user, who will experience most code at the application level preferably as non-async code. The actual package MPyC is for secure multiparty computation, a kind of distributed computation with privacy guarantees. The parties are connected by point-to-point TCP/IP links which are used continuously for exchanging secret-shares of (intermediate) values in the computation.

    The MPyC code development has a bit of history. It started in Python 2 + twisted, but it's now Python 3 + asyncio since at least five years. The major design decisions pertaining to asyncio's event loop go back to Python 3.6 (currently Python 3.8+ is required).

    I'll simply sketch a few reasons for accessing the loop while it's not running.

    Cache the loop inside application to save on get_event_loop() calls.
    The number of calls can run into the millions because Futures are used to handle individual data values, and many micro-Tasks are used as well. The need for caching the loop may be less important since get_event_loop() has been optimized at some point after Python 3.6.

    Configure the loop, as part of set-up code that typically runs in non-async mode (e.g., when loading modules).
    Concretely, setting the loop's exception_handler with an adapted one.
    By doing this on the cached loop this can be done once and for all.

    Use of loop.run_until_complete() from the (non-running) loop.
    A particular reason is that run_until_complete() accepts a Future as argument and returns the Future's result.
    This cannot be replaced by a call to asyncio.run() ; even if one wraps the given Future in a coroutine, the Future will still be attached to a different loop, and therefore aysncio.run() will fail.
    Note that this only is an issue if one already created a Future before the loop is running! But see the next reason.

    Creating Futures (attached to the not-yet running loop) from non-async code.
    The Futures are hidden from the user inside lower-level constructs, and serve as placeholders which will be assigned the results of Tasks that have been set out. These Tasks correspond to the evaluation of "MPyC coroutines" of which many will be active at the same time. The non-async code we are calling from include:

  • top-level code in a Python script
  • calls inside unit tests
  • top-level code in REPL (less of an issue because of asyncio REPL for Python 3.8+)
  • top-level code in Jupyter notebooks (no issue anymore since top-level await is standard in notebooks)
  • Use Futures across multiple loop.run_until_complete() calls.
    Alternate between non-async code and async code, where Futures carry part of the program's state.

    A typical piece of code using mpc.run as a shorthand for loop.run_until_complete looks as follows:

      x = mpc.input(secint(1)) # creates Task that will set Futures inside x (once all values are exchanged)
      s = mpc.sum(x)           # do some computation on x (which is a list with one entry per party)
      y = mpc.output(s)        # creates a Future that will hold the result of the computation
      m = mpc.run(y)           # m becomes a Python int, equal to the number of parties
    

    Only at the last line we need to wait for the result, and if the above code is part of async code we would use m = await y instead.

    Summarizing, we are using asyncio's event loop freely, without much concern if the loop happens to be running or not. The main goal being to accommodate mixes of non-async code and async code, and viewing the loop attached to a thread as a unique piece of real estate to do so.

    Hm... If you are okay with delegating creation of the loop to something else, may I recommend using the Runner class added to 3.11? You should be able to do something like

    r = asyncio.Runner()
    loop = r.get_loop()
    loop.set_exception_handler(exc_handler)

    Now you can run either futures or coroutines:

    loop.run_until_complete(fut)
    r.run(coro(arg, ...))

    The Runner instance allows reusing the loop as many times as you want. When done, use

    r.close()

    You can hide all this detail inside your own abstraction. There's no need to use with Runner() as r: ....

    Right, thanks. The Runner class was also suggested by @graingert above. Indeed, this should give enough flexibility to set an exception handler and more generally one gets direct access to the underlying loop.

    What I overlooked previously is that--unlike asyncio.run()--the loop is kept alive after r.run() is done. It is not closed until r.close() is called as you point out. This may give just enough flexibility to create and carry Futures between subsequent runs of the event loop.

    I'll try and see how things work out.

    ... trying this out was effortless. Double checked things, and turns out that replacing

    self._loop = asyncio.get_event_loop()  # cache event loop
    self._loop = asyncio.Runner().get_loop()  # cache event loop

    does the job! Everything runs as before, and because the loop was already cached, Futures attached to it persist in between subsequent runs.

    One small issue though: if there's a loop already running, asyncio.Runner() will still create a new loop. Would it also be possible that the existing loop is used, if any (because there's supposed to a unique loop per thread anyway)?

    Nice, thx, that makes it work combined like this:

    self._loop = asyncio.mixins._LoopBoundMixin()._get_loop()
    if self._loop is None:
        self._loop = asyncio.Runner().get_loop()

    Now it also works when a loop is already running (like in the asyncio REPL, and probably in a Jupyter notebook as well but I didn't test that for 3.11rc2).

    The breaking changes begun by the deprecation warnings in 3.10 have
    arrived, but according to python/cpython#93453
    the scope has changed somewhat (for the better, I think). Don't test
    on 3.12 until we've adapted to the new plan.
    The breaking changes begun by the deprecation warnings in 3.10 have
    arrived, but according to python/cpython#93453
    the scope has changed somewhat (for the better, I think). Don't test
    on 3.12 until we've adapted to the new plan.
    [BUG] [Python 3.12] py35_asyncio_async_def raises RuntimeError: There is no current event loop in thread 'MainThread'. cython/cython#5183

    The documentation of asyncio.get_event_loop() says:

    If there is no running event loop set, the function will return the result of calling get_event_loop_policy().get_event_loop().

    Which by default raises a RuntimeError, so it is rather confusing. Should the documentation be updated to explicitly say something like:

    If there is no running event loop set, the function will return the result of calling get_event_loop_policy().get_event_loop() which raises a RuntimError with DefaultEventLoopPolicy.

    Partially revert changes made in pythonGH-93453.
    asyncio.DefaultEventLoopPolicy.get_event_loop() now emits a
    DeprecationWarning and creates and sets a new event loop instead of
    raising a RuntimeError if there is no current event loop set.

    Oooh, the decumentation is indeed a mess. There's no place that I can find where the behavior of DefaultEventLoopPolicy.get_event_loop() is currently documented. AbstractEventLoopPolicy is vague (just says it returns a loop and never None), DefaultEventLoopPolicy doesn't say anything specific, and BaseDefaultEventLoopPolicy is undocumented (and private, apparently, though in the light of the PEP 387 discussions it might be implicitly public). The documentation section linked to should be hyperlinked to get_event_loop_policy() and some policy class's get_event_loop() method, which should be documented.

    Is it fair to say this is now a documentation issue, or is there more to be done? Should this hold up 3.12.0a4?

    Hm, #100410 isn't merged yet, even though I approved it two weeks ago. @serhiy-storchaka do you have any hesitations?

    Partially revert changes made in GH-93453. asyncio.DefaultEventLoopPolicy.get_event_loop() now emits a DeprecationWarning and creates and sets a new event loop instead of raising a RuntimeError if there is no current event loop set. Co-authored-by: Guido van Rossum <[email protected]>