-
Notifications
You must be signed in to change notification settings - Fork 158
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
ContextVar not propagated from fixture to test #127
Comments
I've also hit this. I think what's happening is explained by pytest-asyncio/pytest_asyncio/plugin.py Line 97 in 2c8023d
along with understanding of https://www.python.org/dev/peps/pep-0550. The One option is to try to avoid using a coroutine fixture when you need to set context. So for example import asyncio
from contextvars import ContextVar
import pytest
blah = ContextVar("blah")
async def set_context():
blah.set("hello")
assert blah.get() == "hello"
@pytest.fixture
def my_context_var():
asyncio.run(set_context())
assert blah.get() == "hello"
yield blah
@pytest.mark.asyncio
async def test_blah(my_context_var):
assert my_context_var.get() == "hello" |
+1 |
A note on the workaround above: the
A much easier solution is to set the contextvar directly in the sync fixture — note, not async fixture, but sync: from contextvars import ContextVar
import pytest
blah = ContextVar("blah")
@pytest.fixture
def my_context_var(): # << SYNC!!!
blah.set("hello")
assert blah.get() == "hello"
yield blah
@pytest.mark.asyncio
async def test_blah(my_context_var):
assert my_context_var.get() == "hello" I use this approach in Kopf actively, and it works:
|
The cause appears to be that the To augment @nolar 's solution, I'd recommend this:
|
I have async functions that set context vars with objects that should be used in the entire app (I use a file with context vars as an IoC container). I wanted to run that code in a fixture, so I get nice setup and teardown, but sadly, I can't run that in a sync fixture. So the workarounds don't work for me. Just letting anybody interested know :) |
This issue could be fixed easily in Python 3.11, but even for 3.10 this could be fixed like this |
@nikicat Can you show us how to fix this issue in Python 3.11 please? |
@rnarkk I believe that #127 (comment) still works. The essence of Nikolay's suggestion is to make a custom |
#620 introduced the asyncio_event_loop mark, which provides a class-scoped or module-scoped event loop when a class or module is marked, respectively. #646 plans to extend the mark so it can be applied to test coroutines as well. Under the hood, the mark creates an event loop and attaches it to the marked pytest Item. Tests and fixtures run in the same event loop when the asyncio_event_loop mark is used. Unfortunately, tests and fixtures are run using loop.run_until_complete which still prevents context variables from being propagated from fixtures to tests. The next step is to make asyncio_event_loop attach an asyncio.Runner, instead of an event loop to the marked pytest Item. This allows replacing loop.run_until_complete with asyncio.Runner.run, which preserves contextvars. We will also need a backport for Python 3.10 and earlier. There's still work to be done, but I wanted to give a progress update and let you know that this issue hasn't been forgotten. |
This is very troublesome with encode/databases >=0.8.0 (related). They keep state in context vars which results in a lost transaction reference - the DB connection and a unit test rollback transaction is made in a fixture, then in the test the transaction is gone breaking an assert in encode/databases. I was able to get around it with the help of the comment from above and ended up with this - this solves my issue with encode/databases: def task_factory(loop, coro, context=None):
stack = traceback.extract_stack()
for frame in stack[-2::-1]:
package_name = Path(frame.filename).parts[-2]
if package_name != "asyncio":
if package_name == "pytest_asyncio":
# This function was called from pytest_asyncio, use shared context
break
else:
# This function was called from somewhere else, create context copy
context = None
break
return asyncio.Task(coro, loop=loop, context=context)
@pytest.fixture(scope="session")
def event_loop():
loop = asyncio.get_event_loop_policy().new_event_loop()
context = contextvars.copy_context()
loop.set_task_factory(functools.partial(task_factory, context=context))
asyncio.set_event_loop(loop)
return loop It complains that "Replacing the event_loop fixture with a custom implementation is deprecated and will lead to errors in the future." but nothing else I tried seems to work, the context vars are always gone. I thought this would work best: class CustomEventLoop(asyncio.SelectorEventLoop):
_task_factory = functools.partial(task_factory, context=contextvars.copy_context())
# also tried to overload create_task, still no luck
def create_task(self, coro, *, name=None, context=None):
context = contextvars.copy_context()
print("Creating task with context", context)
return super().create_task(coro, name=name, context=context)
class CustomEventLoopPolicy(asyncio.DefaultEventLoopPolicy):
_loop_factory = CustomEventLoop
@pytest.fixture(scope="session")
def event_loop_policy():
policy = CustomEventLoopPolicy()
yield policy
policy.get_event_loop().close() but neither with |
OK, I was able to make it stop complaining about the deprecation with the following: import asyncio
import functools
import traceback
from pathlib import Path
def task_factory(loop, coro, context=None):
stack = traceback.extract_stack()
for frame in stack[-2::-1]:
package_name = Path(frame.filename).parts[-2]
if package_name != "asyncio":
if package_name == "pytest_asyncio":
# This function was called from pytest_asyncio, use shared context
break
else:
# This function was called from somewhere else, create context copy
context = None
break
return asyncio.Task(coro, loop=loop, context=context)
class CustomEventLoopPolicy(asyncio.DefaultEventLoopPolicy):
def __init__(self, context) -> None:
super().__init__()
self.context = context
def new_event_loop(self):
loop = self._loop_factory()
loop.set_task_factory(
functools.partial(task_factory, context=self.context)
)
return loop
@pytest.fixture(scope="session")
def event_loop_policy():
policy = CustomEventLoopPolicy(contextvars.copy_context())
yield policy
policy.get_event_loop().close() |
The approach I've taken here is to maintain a contextvars.Context instance in a contextvars.ContextVar, copying it from the ambient context whenever we create a new event loop. The fixture setup and teardown run within that context, and each test function gets a copy (as if it were created as a new asyncio.Task from within the fixture task). Fixes pytest-dev#127.
The approach I've taken here is to maintain a contextvars.Context instance in a contextvars.ContextVar, copying it from the ambient context whenever we create a new event loop. The fixture setup and teardown run within that context, and each test function gets a copy (as if it were created as a new asyncio.Task from within the fixture task). Fixes pytest-dev#127.
The approach I've taken here is to maintain a contextvars.Context instance in a contextvars.ContextVar, copying it from the ambient context whenever we create a new event loop. The fixture setup and teardown run within that context, and each test function gets a copy (as if it were created as a new asyncio.Task from within the fixture task). Fixes #127.
Big thanks to @bcmills for contributing a fix for Python 3.11 and above! Since pytest-asyncio currently supports Python 3.9 and 3.10 as well, I'm reopening the issue for now. |
I'm trying to figure out why this code sample isn't working. I'm not experienced with asyncio yet, so this is very likely a simple misunderstanding on my part.
The text was updated successfully, but these errors were encountered: