Skip to content

Commit c3ad634

Browse files
provinzkrautseifertm
authored andcommitted
fix: Shutdown generators before closing event loops.
Add a comment
1 parent e8ffb10 commit c3ad634

File tree

3 files changed

+56
-15
lines changed

3 files changed

+56
-15
lines changed

docs/reference/changelog.rst

+5
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,11 @@
22
Changelog
33
=========
44

5+
0.25.2 (UNRELEASED)
6+
===================
7+
8+
- Call ``loop.shutdown_asyncgens()`` before closing the event loop to ensure async generators are closed in the same manner as ``asyncio.run`` does `#1034 <https://github.com/pytest-dev/pytest-asyncio/pull/1034>`_
9+
510
0.25.1 (2025-01-02)
611
===================
712
- Fixes an issue that caused a broken event loop when a function-scoped test was executed in between two tests with wider loop scope `#950 <https://github.com/pytest-dev/pytest-asyncio/issues/950>`_

pytest_asyncio/plugin.py

+24-15
Original file line numberDiff line numberDiff line change
@@ -708,11 +708,12 @@ def scoped_event_loop(
708708
event_loop_policy,
709709
) -> Iterator[asyncio.AbstractEventLoop]:
710710
new_loop_policy = event_loop_policy
711-
with _temporary_event_loop_policy(new_loop_policy):
712-
loop = _make_pytest_asyncio_loop(asyncio.new_event_loop())
711+
with (
712+
_temporary_event_loop_policy(new_loop_policy),
713+
_provide_event_loop() as loop,
714+
):
713715
asyncio.set_event_loop(loop)
714716
yield loop
715-
loop.close()
716717

717718
# @pytest.fixture does not register the fixture anywhere, so pytest doesn't
718719
# know it exists. We work around this by attaching the fixture function to the
@@ -1147,28 +1148,36 @@ def _retrieve_scope_root(item: Collector | Item, scope: str) -> Collector:
11471148
def event_loop(request: FixtureRequest) -> Iterator[asyncio.AbstractEventLoop]:
11481149
"""Create an instance of the default event loop for each test case."""
11491150
new_loop_policy = request.getfixturevalue(event_loop_policy.__name__)
1150-
with _temporary_event_loop_policy(new_loop_policy):
1151-
loop = asyncio.get_event_loop_policy().new_event_loop()
1152-
# Add a magic value to the event loop, so pytest-asyncio can determine if the
1153-
# event_loop fixture was overridden. Other implementations of event_loop don't
1154-
# set this value.
1155-
# The magic value must be set as part of the function definition, because pytest
1156-
# seems to have multiple instances of the same FixtureDef or fixture function
1157-
loop = _make_pytest_asyncio_loop(loop)
1151+
with _temporary_event_loop_policy(new_loop_policy), _provide_event_loop() as loop:
11581152
yield loop
1159-
loop.close()
1153+
1154+
1155+
@contextlib.contextmanager
1156+
def _provide_event_loop() -> Iterator[asyncio.AbstractEventLoop]:
1157+
loop = asyncio.get_event_loop_policy().new_event_loop()
1158+
# Add a magic value to the event loop, so pytest-asyncio can determine if the
1159+
# event_loop fixture was overridden. Other implementations of event_loop don't
1160+
# set this value.
1161+
# The magic value must be set as part of the function definition, because pytest
1162+
# seems to have multiple instances of the same FixtureDef or fixture function
1163+
loop = _make_pytest_asyncio_loop(loop)
1164+
try:
1165+
yield loop
1166+
finally:
1167+
try:
1168+
loop.run_until_complete(loop.shutdown_asyncgens())
1169+
finally:
1170+
loop.close()
11601171

11611172

11621173
@pytest.fixture(scope="session")
11631174
def _session_event_loop(
11641175
request: FixtureRequest, event_loop_policy: AbstractEventLoopPolicy
11651176
) -> Iterator[asyncio.AbstractEventLoop]:
11661177
new_loop_policy = event_loop_policy
1167-
with _temporary_event_loop_policy(new_loop_policy):
1168-
loop = _make_pytest_asyncio_loop(asyncio.new_event_loop())
1178+
with _temporary_event_loop_policy(new_loop_policy), _provide_event_loop() as loop:
11691179
asyncio.set_event_loop(loop)
11701180
yield loop
1171-
loop.close()
11721181

11731182

11741183
@pytest.fixture(scope="session", autouse=True)

tests/test_event_loop_fixture.py

+27
Original file line numberDiff line numberDiff line change
@@ -53,3 +53,30 @@ async def test_custom_policy_is_not_overwritten():
5353
)
5454
result = pytester.runpytest_subprocess("--asyncio-mode=strict")
5555
result.assert_outcomes(passed=2)
56+
57+
58+
def test_event_loop_fixture_handles_unclosed_async_gen(
59+
pytester: Pytester,
60+
):
61+
pytester.makeini("[pytest]\nasyncio_default_fixture_loop_scope = function")
62+
pytester.makepyfile(
63+
dedent(
64+
"""\
65+
import asyncio
66+
import pytest
67+
68+
pytest_plugins = 'pytest_asyncio'
69+
70+
@pytest.mark.asyncio
71+
async def test_something():
72+
async def generator_fn():
73+
yield
74+
yield
75+
76+
gen = generator_fn()
77+
await gen.__anext__()
78+
"""
79+
)
80+
)
81+
result = pytester.runpytest_subprocess("--asyncio-mode=strict", "-W", "default")
82+
result.assert_outcomes(passed=1, warnings=0)

0 commit comments

Comments
 (0)