The documentation 6.2.9.4. "Asynchronous generator-iterator methods" is phrased really ambiguously. To make matters worse, typing.AsyncGenerator
is not specified fully correctly.
Here, I attempt to more clearly capture the actual interface contract, based on what I've read and observed. See also PEP 492 -- Coroutines with async and await syntax and PEP 525 -- Asynchronous Generators.
This might be useful to implement lower-level behaviors than you can with async for
, like sending values into the generator function.
TSend = TypeVar('TSend', contravariant=True)
TYield = TypeVar('TYield', covariant=True)
class AsyncGenerator(ABC, AsyncIterator[TYield], Generic[TYield, TSend]):
"""
Represents a one-shot "asynchronous generator-iterator" (as it is
referred to in the docs). The concept referred to as an "asynchronous
generator _function_" is the function defined with `async def` that has a
return type of AsyncGenerator.
In other words, `fn` here is an asynchronous generator function:
async def fn() -> AsyncGenerator[...]:
...
And `agen` here is an asynchronous generator-iterator:
agen = fn()
The lifetime of an AsyncGenerator is as follows:
1. The asynchronous generator-iterator is started by awaiting __anext__()
or asend(None). This begins executing the asynchronous generator
function.
2. Once started, you may (but are not required to):
2a. Call asend() with a TSend value and await the result, to continue
executing the asynchronous generator function.
2b. Await athrow() to raise an exception inside the asynchronous
generator function, which may respond by yielding a value.
3. You may repeat step 2 as long as the awaitable returned does not raise
an exception.
4. At any point, you may await aclose() to raise GeneratorExit inside the
asynchronous generator function, requesting that it exit. This has no
effect if an awaitable from step 2 already raised an exception, or if the
asynchronous generator function never began executing, so it is always
safe to invoke.
5. Once the asynchronous generator function has exited (gracefully or
through an exception), or the generator has been closed (even if the
function was never started), the asynchronous generator-iterator instance
may not be restarted. However, a new one can be obtained by calling the
function again:
agen = fn()
"""
def __aiter__(self) -> AsyncIterator[TYield]:
return self
async def __anext__(self) -> TYield: # throws: StopAsyncIteration, ...
"""
Returns an awaitable which, when run, starts to execute the
asynchronous generator, or resumes it from the last executed yield
expression.
If the generator has already exited (gracefully or through an
exception) or been closed previously, nothing happens, and the
awaitable returned by __anext__() will raise a StopAsyncIteration
exception.
If resuming from a yield expression, the expression will evaluate to
None inside the generator, because no value is being provided (use
asend() if you want that).
The generator will run until the next yield expression or it exits
(e.g., through a return statement).
If the generator yields a value, the awaitable returned by
__anext__() will return that value, and the generator's execution
will be re-suspended. (Under the hood, this is implemented as the
generator raising StopIteration, but you don't need to care about
that.)
If the generator raises an exception, the awaitable returned by
__anext__() will raise the same exception. (Note that if a generator
attempts to _explicitly_ raise StopIteration or StopAsyncIteration in
its implementation, it will instead be converted into a RuntimeError,
per PEP 479.)
If the generator exits gracefully, the awaitable returned by
__anext__() will raise a StopAsyncIteration exception.
"""
return await self.asend(None)
async def asend(
self,
input: Optional[TSend]
) -> TYield: # throws: StopAsyncIteration, ...
"""
Returns an awaitable which, when run, starts to execute the
asynchronous generator, or resumes it from the last executed yield
expression.
If asend() is being called to start the generator, it must be called
with None as the argument, because there is no yield expression that
could receive the value. (This is the only reason `input` is typed as
Optional[TSend].)
If the generator has already exited (gracefully or through an
exception) or been closed previously, nothing happens, and the
awaitable returned by asend() will raise a StopAsyncIteration
exception.
If resuming from a yield expression, the expression will evaluate to
`input` inside the generator.
The generator will run until the next yield expression or it exits
(e.g., through a return statement).
If the generator yields a value, the awaitable returned by asend()
will return that value, and the generator's execution will be
re-suspended. (Under the hood, this is implemented as the generator
raising StopIteration, but you don't need to consider that.)
If the generator raises an exception, the awaitable returned by
asend() will raise the same exception. (Note that if a generator
attempts to _explicitly_ raise StopIteration or StopAsyncIteration in
its implementation, it will instead be converted into a RuntimeError,
per PEP 479.)
If the generator exits gracefully, the awaitable returned by asend()
will raise a StopAsyncIteration exception.
"""
...
async def athrow(
self,
exc_type: Type[BaseException],
exc_value: Optional[BaseException] = None,
traceback: Optional[TracebackType] = None,
) -> Optional[TYield]: # throws: exc_type, StopAsyncIteration, ...
"""
Returns an awaitable which, when run, raises an exception _inside_
the generator at the point of execution where it was last suspended.
If the generator has not yet been started, the awaitable returned by
athrow() will immediately raise the passed-in exception, and the
generator will be closed. In other words, the generator is not given
any opportunity to catch the exception, and it will not be able to be
started afterward.
If the generator has already exited (gracefully or through an
exception) or been closed previously, nothing happens, and the
awaitable returned by athrow() will return None.
Otherwise, after raising the exception inside the generator, athrow()
behaves exactly like __anext__().
In other words:
If the generator does not catch the passed-in exception, or raises a
different exception, then the awaitable returned by athrow() will
propagate that exception. (Note that if a generator attempts to
_explicitly_ raise StopIteration or StopAsyncIteration in its
implementation, it will instead be converted into a RuntimeError, per
PEP 479.)
If the generator catches the passed-in exception, then yields a
value, the awaitable returned by athrow() will return that value, and
the generator's execution will be re-suspended. (Under the hood, this
is implemented as the generator raising StopIteration, but you don't
need to consider that.)
If the generator catches the passed-in exception, then exits
gracefully, the awaitable returned by athrow() will raise a
StopAsyncIteration exception.
"""
...
async def aclose(
self
) -> None: # throws RuntimeError, ...
"""
Returns an awaitable which, when run, raises a GeneratorExit
exception _inside_ the generator at the point of execution where it
was last suspended.
If the generator has already exited (gracefully or through an
exception) or been closed previously, or the generator was never
started, nothing happens, and the awaitable returned by aclose() will
return gracefully.
If the generator does not catch the GeneratorExit exception, or
catches GeneratorExit then exits gracefully, the awaitable returned
by aclose() will return gracefully.
If the generator raises a different exception, then the awaitable
returned by aclose() will propagate that exception.
The generator _must not_ yield a value. If the generator catches the
GeneratorExit exception then yields a value, the awaitable returned
by aclose() will raise a RuntimeError.
"""
try:
await self.athrow(GeneratorExit)
except (GeneratorExit, StopAsyncIteration):
pass
else:
raise RuntimeError("...")