-
-
Notifications
You must be signed in to change notification settings - Fork 31.9k
bpo-29302: Implement contextlib.AsyncExitStack. #4790
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
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
Conversation
thanks @Kentzo ! |
4374de3
to
60000cd
Compare
@1st1 I'd like to take a further look at this one before we merge it, but
won't have time until Jan 6th or so (just before the final alpha). Does
that timeline work for you?
|
The alternative plan would be to merge this version based on the existing
reviews, and if I have any subsequent changes I want to make, I can do that
any time before the first beta. That's probably a better approach, since
it's more resilient against my being otherwise occupied in the first week
of January.
|
Sure, I think this is worth another look so let's wait. Let's just make sure we don't forget about this before the feature freeze. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I like the patch overall, but there's one design concern I have, as well as a backwards compatibility concern:
- for the backwards compatibility issue: calls using keyword arguments mean we can't arbitrarily change parameter names in public APIs
- for the design question, implicitly supporting synchronous context managers in the asynchronous exit stack seems questionable to me.
The design question is one @1st1 and I talked about on the issue tracker (see https://bugs.python.org/issue29302#msg288735 and the next couple of comments), and after seeing the code in PR form, I definitely want to go with the more explicit API:
- add separate
*_sync
variants of all the callback registration APIs in the async exit stack variant rather than trying to guess the user's intent based on the type of the object they passed in (if guessing proves desirable, we can add it later, but if we start with it, we're likely stuck with it even if it proves confusing) - avoid checking for awaitables when unwinding the stack in the async case (either add an awaitable wrapper to the synchronous callbacks as I suggest in this review, or else store a 2-tuple as @1st1 suggested on the tracker)
Lib/contextlib.py
Outdated
self.push(_exit_wrapper) | ||
|
||
def push(self, exit): | ||
def push(self, exit_obj): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Folks may be calling stack.push(exit=obj)
, so we shouldn't gratuitously change parameter names.
Lib/contextlib.py
Outdated
@@ -4,9 +4,10 @@ | |||
import _collections_abc | |||
from collections import deque | |||
from functools import wraps | |||
import inspect |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This introduces a dependency I'd prefer to avoid - inspect
pulls in a lot of other modules, and contextlib
is used frequently enough that we should aim to minimise its dependency tree.
Lib/contextlib.py
Outdated
# We look up the special methods on the type to match the with statement | ||
_cm_type = type(cm) | ||
|
||
# if you have both aenter + enter, aenter will 'win' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This comment is oddly placed, and doesn't seem especially accurate - the check is being made against __aexit__
, not __aenter__
.
More generally, implicitly interleaving synchronous with asynchronous context managers seems dubious to me - while it might be a good idea to allow it, doing it automagically rather than explicitly seems like a recipe for confusion when reading and debugging code.
Lib/contextlib.py
Outdated
result = next(gen) | ||
while 1: | ||
try: | ||
if inspect.isawaitable(result): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If we do decide to allow synchronous context managers in asynchronous exit stacks, I'd suggest wrapping their exit callbacks in an awaitable, so this check can be avoided.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
inspect
is also used by _create_exit_wrapper
and _create_cb_wrapper
.
A Python core developer has requested some changes be made to your pull request before we can consider merging it. If you could please address their requests along with any other requests in other reviews from core developers that would be appreciated. Once you have made the requested changes, please leave a comment on this pull request containing the phrase |
Couple of thoughts:
with Foo:
async with Bar:
with Ham:
...
|
Also, we need to get this merged before Jan 29. @Kentzo if you don't have time to work on this in the next few days I can handle this PR myself. |
@1st1 So the differences between AsyncExitStack and ExitStack would be:
I like that, since it means registering synchronous context managers and callbacks stays the same regardless of which kind of stack you're using. A note regarding the method names: the synchronous method names are just |
Right.
Yeah, the API consistency for synchronous CMs is great. To be honest I haven't thought about it until you highlighted it, but now I see it as a requirement!
+1. |
Please name the close method Regarding the bikeshed, I think my preference FWIW is to make both sync and async explicitly marked, so it's e.g. |
@njsmith following this way we should add |
The semantics around closing things are much more general and subtle than
sending/receiving. Obviously send/recv are inherently semantically blocking
operations, but close() is non-blocking on a regular socket and blocking on
an SSL socket. Plus in general there are just a lot more kinds of things
that get closed so it's useful to have general cross-type conventions here.
You can't use contextlib.closing with an object whose close is async (or
rather, you can if it calls the method 'close', and it will claim to
succeed without doing anything!). You *can* use async_generator.aclosing on
any object with an async close, but it assumes that the method is called
aclose(), because it's not specific to async generators but that is one of
the original use cases. In some cases you might want to provide both a
synchronous close() and an async aclose() on the same object, e.g. if the
object's native close semantics are synchronous, but the way it's used also
needs to match some generic ABC. (Probably it's *better* it you can handle
this via composition – in trio, trio.socket.socket has a synchronous
close() while trio.SocketStream and trio.SocketListener have async aclose()
methods – but sometimes stuff happens.) So personally at least I find it
very useful to have a conventional protocol for async close that doesn't
overlap with the existing convention for sync close.
…On Jan 20, 2018 04:42, "Andrew Svetlov" ***@***.***> wrote:
@njsmith <https://github.com/njsmith> following this way we should add a
prefix to all async functions everywhere: aclose(), asend(), arecv() etc.
The logic is weird, isn't it? I don't see how context manager is related
to generator.
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#4790 (comment)>, or mute
the thread
<https://github.com/notifications/unsubscribe-auth/AAlOaDSFF_kyr6I6me86pr7c6e5W8fUjks5tMd8zgaJpZM4Q85Ru>
.
|
@njsmith Good point regarding close() vs aclose() - I'd forgotten that part of PEP 525. Given that, I agree the method on AsyncExitStack should aclose(), and the synchronous close API should just be missing entirely (as with the other synchronous context management methods). I'm not especially worried about folks having to sprinkle extra "async" mentions through their code to get async context managers to work as expected - that's the default language wide. For example, even though you've written By contrast, I'm definitely not keen on having the correctness of code like This does mean that |
I agree with everything Nick said here. |
👍 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This version looks good to me, but leaving to @1st1 to merge after his review :)
The Appveyor failures look unrelated to me - closing & reopening to check if they're transient errors. |
I'll take a look today/tomorrow, thanks Nick! |
@ncoghlan What do you think about adding |
@Kentzo Worth considering as a separate issue for 3.8, but I wouldn't rush into it for 3.7. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There're multiple formatting issues with the PR. Please fix the code to follow PEP 8. Reformat docstrings to have a short first line ended with a period, optionally followed by an empty line with more paragraphs.
Most importantly, I don't like the design of _shutdown_loop
. I think we shouldn't use generator.send/throw here, I'd prefer to copy/paste the code which will make the code easier to read and maintain (generators are complicated!)
/cc @ncoghlan
Doc/library/contextlib.rst
Outdated
Continuing the example for :func:`asynccontextmanager`:: | ||
|
||
async with AsyncExitStack() as stack: | ||
connections = [await stack.enter_async_context(get_connection()) for i in range(5)] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please trim the lines in your to be max 79 chars wide.
Doc/library/contextlib.rst
Outdated
|
||
An :ref:`asynchronous context manager <async-context-managers>` similar | ||
to :class:`ExitStack` that can additionally combine other asynchronous | ||
context managers and cleanup coroutine functions. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd rewrite to:
An :ref:`asynchronous context manager <async-context-managers>`, similar
to :class:`ExitStack`, that supports combining both synchronous and
asynchronous context managers, as well as having coroutines for
cleanup logic.
Doc/library/contextlib.rst
Outdated
|
||
.. method:: enter_async_context(cm) | ||
|
||
Same as :meth:`enter_context` but expects an asynchronous context manager. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same as -> Similar to
Doc/library/contextlib.rst
Outdated
.. method:: push_async_exit(exit) | ||
|
||
Same as :meth:`push` but expects either an asynchronous context manager or | ||
a coroutine function. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"or a coroutine function" -> "or a coroutine".
Doc/library/contextlib.rst
Outdated
|
||
.. method:: push_async_callback(callback, *args, **kwds) | ||
|
||
Same as :meth:`callback` but expects a coroutine function. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ditto
Lib/contextlib.py
Outdated
class ExitStack(AbstractContextManager): | ||
"""Context manager for dynamic management of a stack of exit callbacks | ||
class _BaseExitStack: | ||
""" |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
First line of the docstring should not be empty. Reformat to
"""A base class for ExitStack and AsyncExitStack."""
Lib/contextlib.py
Outdated
|
||
Can suppress exceptions the same way __exit__ methods can. | ||
|
||
Can suppress exceptions the same way __exit__ method can. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please keep and empty line after the first line of the docstring. I also recommend you to read https://www.python.org/dev/peps/pep-0257/.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@1st1 In that PEP there is no empty line after a doctstring inside def
. Should I add this extra empty line for all functions in my PR?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ilya, I mean this:
def foo():
"""First line
A paragraph
"""
code
should be formatted as this
def foo():
"""First line.
A paragraph.
"""
code
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The first line should be a complete sentence.
I'm not making these rules up -- there're tons of tools out there that expect this specific code style to extract/parse docstrings.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah I see.
Lib/contextlib.py
Outdated
def _push_exit_callback(self, callback, is_sync=True): | ||
self._exit_callbacks.append((is_sync, callback)) | ||
|
||
def _shutdown_loop(self, *exc_details): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't like using generator.send/throw for this. I'd better copy/paste the code. Generators have a pretty complicated semantics (especially around closing) and sometimes have issues with keeping exceptions' tracebacks in tact.
Lib/contextlib.py
Outdated
@@ -494,6 +497,127 @@ def _fix_exception_context(new_exc, old_exc): | |||
return received_exc and suppressed_exc | |||
|
|||
|
|||
# Inspired by discussions on http://bugs.python.org/issue13585 | |||
class ExitStack(_BaseExitStack, AbstractContextManager): | |||
"""Context manager for dynamic management of a stack of exit callbacks |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Put periods .
at the end of all your sentences. Throughout the patch.
Lib/contextlib.py
Outdated
"""Async context manager for dynamic management of a stack of exit callbacks | ||
For example: | ||
async with AsyncExitStack() as stack: | ||
connections = [await stack.enter_async_context(get_connection()) for i in range(5)] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tabulation is off and many lines are longer than 79. Please make sure that your PR follows PEP 8.
A Python core developer has requested some changes be made to your pull request before we can consider merging it. If you could please address their requests along with any other requests in other reviews from core developers that would be appreciated. Once you have made the requested changes, please leave a comment on this pull request containing the phrase |
I have made the requested changes; please review again |
Merged. Thanks so much Ilya! |
This change is entirely based on the work of @thehesiod discussed at https://bugs.python.org/issue29302
I only changed implementation of
__aexit__
:to:
and added tests.
https://bugs.python.org/issue29302