# mypy: allow-untyped-defs import os from pathlib import Path import sys import textwrap from _pytest.compat import getfuncargnames from _pytest.config import ExitCode from _pytest.fixtures import deduplicate_names from _pytest.fixtures import TopRequest from _pytest.monkeypatch import MonkeyPatch from _pytest.pytester import get_public_names from _pytest.pytester import Pytester from _pytest.python import Function import pytest def test_getfuncargnames_functions(): """Test getfuncargnames for normal functions""" def f(): raise NotImplementedError() assert not getfuncargnames(f) def g(arg): raise NotImplementedError() assert getfuncargnames(g) == ("arg",) def h(arg1, arg2="hello"): raise NotImplementedError() assert getfuncargnames(h) == ("arg1",) def j(arg1, arg2, arg3="hello"): raise NotImplementedError() assert getfuncargnames(j) == ("arg1", "arg2") def test_getfuncargnames_methods(): """Test getfuncargnames for normal methods""" class A: def f(self, arg1, arg2="hello"): raise NotImplementedError() assert getfuncargnames(A().f) == ("arg1",) def test_getfuncargnames_staticmethod(): """Test getfuncargnames for staticmethods""" class A: @staticmethod def static(arg1, arg2, x=1): raise NotImplementedError() assert getfuncargnames(A.static, cls=A) == ("arg1", "arg2") def test_getfuncargnames_staticmethod_inherited() -> None: """Test getfuncargnames for inherited staticmethods (#8061)""" class A: @staticmethod def static(arg1, arg2, x=1): raise NotImplementedError() class B(A): pass assert getfuncargnames(B.static, cls=B) == ("arg1", "arg2") def test_getfuncargnames_partial(): """Check getfuncargnames for methods defined with functools.partial (#5701)""" import functools def check(arg1, arg2, i): raise NotImplementedError() class T: test_ok = functools.partial(check, i=2) values = getfuncargnames(T().test_ok, name="test_ok") assert values == ("arg1", "arg2") def test_getfuncargnames_staticmethod_partial(): """Check getfuncargnames for staticmethods defined with functools.partial (#5701)""" import functools def check(arg1, arg2, i): raise NotImplementedError() class T: test_ok = staticmethod(functools.partial(check, i=2)) values = getfuncargnames(T().test_ok, name="test_ok") assert values == ("arg1", "arg2") @pytest.mark.pytester_example_path("fixtures/fill_fixtures") class TestFillFixtures: def test_funcarg_lookupfails(self, pytester: Pytester) -> None: pytester.copy_example() result = pytester.runpytest() # "--collect-only") assert result.ret != 0 result.stdout.fnmatch_lines( """ *def test_func(some)* *fixture*some*not found* *xyzsomething* """ ) def test_detect_recursive_dependency_error(self, pytester: Pytester) -> None: pytester.copy_example() result = pytester.runpytest() result.stdout.fnmatch_lines( ["*recursive dependency involving fixture 'fix1' detected*"] ) def test_funcarg_basic(self, pytester: Pytester) -> None: pytester.copy_example() item = pytester.getitem(Path("test_funcarg_basic.py")) assert isinstance(item, Function) # Execute's item's setup, which fills fixtures. item.session._setupstate.setup(item) del item.funcargs["request"] assert len(get_public_names(item.funcargs)) == 2 assert item.funcargs["some"] == "test_func" assert item.funcargs["other"] == 42 def test_funcarg_lookup_modulelevel(self, pytester: Pytester) -> None: pytester.copy_example() reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_funcarg_lookup_classlevel(self, pytester: Pytester) -> None: p = pytester.copy_example() result = pytester.runpytest(p) result.stdout.fnmatch_lines(["*1 passed*"]) def test_conftest_funcargs_only_available_in_subdir( self, pytester: Pytester ) -> None: pytester.copy_example() result = pytester.runpytest("-v") result.assert_outcomes(passed=2) def test_extend_fixture_module_class(self, pytester: Pytester) -> None: testfile = pytester.copy_example() result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*1 passed*"]) def test_extend_fixture_conftest_module(self, pytester: Pytester) -> None: p = pytester.copy_example() result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) result = pytester.runpytest(str(next(Path(str(p)).rglob("test_*.py")))) result.stdout.fnmatch_lines(["*1 passed*"]) def test_extend_fixture_conftest_conftest(self, pytester: Pytester) -> None: p = pytester.copy_example() result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) result = pytester.runpytest(str(next(Path(str(p)).rglob("test_*.py")))) result.stdout.fnmatch_lines(["*1 passed*"]) def test_extend_fixture_conftest_plugin(self, pytester: Pytester) -> None: pytester.makepyfile( testplugin=""" import pytest @pytest.fixture def foo(): return 7 """ ) pytester.syspathinsert() pytester.makeconftest( """ import pytest pytest_plugins = 'testplugin' @pytest.fixture def foo(foo): return foo + 7 """ ) pytester.makepyfile( """ def test_foo(foo): assert foo == 14 """ ) result = pytester.runpytest("-s") assert result.ret == 0 def test_extend_fixture_plugin_plugin(self, pytester: Pytester) -> None: # Two plugins should extend each order in loading order pytester.makepyfile( testplugin0=""" import pytest @pytest.fixture def foo(): return 7 """ ) pytester.makepyfile( testplugin1=""" import pytest @pytest.fixture def foo(foo): return foo + 7 """ ) pytester.syspathinsert() pytester.makepyfile( """ pytest_plugins = ['testplugin0', 'testplugin1'] def test_foo(foo): assert foo == 14 """ ) result = pytester.runpytest() assert result.ret == 0 def test_override_parametrized_fixture_conftest_module( self, pytester: Pytester ) -> None: """Test override of the parametrized fixture with non-parametrized one on the test module level.""" pytester.makeconftest( """ import pytest @pytest.fixture(params=[1, 2, 3]) def spam(request): return request.param """ ) testfile = pytester.makepyfile( """ import pytest @pytest.fixture def spam(): return 'spam' def test_spam(spam): assert spam == 'spam' """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*1 passed*"]) def test_override_parametrized_fixture_conftest_conftest( self, pytester: Pytester ) -> None: """Test override of the parametrized fixture with non-parametrized one on the conftest level.""" pytester.makeconftest( """ import pytest @pytest.fixture(params=[1, 2, 3]) def spam(request): return request.param """ ) subdir = pytester.mkpydir("subdir") subdir.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture def spam(): return 'spam' """ ), encoding="utf-8", ) testfile = subdir.joinpath("test_spam.py") testfile.write_text( textwrap.dedent( """\ def test_spam(spam): assert spam == "spam" """ ), encoding="utf-8", ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*1 passed*"]) def test_override_non_parametrized_fixture_conftest_module( self, pytester: Pytester ) -> None: """Test override of the non-parametrized fixture with parametrized one on the test module level.""" pytester.makeconftest( """ import pytest @pytest.fixture def spam(): return 'spam' """ ) testfile = pytester.makepyfile( """ import pytest @pytest.fixture(params=[1, 2, 3]) def spam(request): return request.param params = {'spam': 1} def test_spam(spam): assert spam == params['spam'] params['spam'] += 1 """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*3 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*3 passed*"]) def test_override_non_parametrized_fixture_conftest_conftest( self, pytester: Pytester ) -> None: """Test override of the non-parametrized fixture with parametrized one on the conftest level.""" pytester.makeconftest( """ import pytest @pytest.fixture def spam(): return 'spam' """ ) subdir = pytester.mkpydir("subdir") subdir.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture(params=[1, 2, 3]) def spam(request): return request.param """ ), encoding="utf-8", ) testfile = subdir.joinpath("test_spam.py") testfile.write_text( textwrap.dedent( """\ params = {'spam': 1} def test_spam(spam): assert spam == params['spam'] params['spam'] += 1 """ ), encoding="utf-8", ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*3 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*3 passed*"]) def test_override_autouse_fixture_with_parametrized_fixture_conftest_conftest( self, pytester: Pytester ) -> None: """Test override of the autouse fixture with parametrized one on the conftest level. This test covers the issue explained in issue 1601 """ pytester.makeconftest( """ import pytest @pytest.fixture(autouse=True) def spam(): return 'spam' """ ) subdir = pytester.mkpydir("subdir") subdir.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture(params=[1, 2, 3]) def spam(request): return request.param """ ), encoding="utf-8", ) testfile = subdir.joinpath("test_spam.py") testfile.write_text( textwrap.dedent( """\ params = {'spam': 1} def test_spam(spam): assert spam == params['spam'] params['spam'] += 1 """ ), encoding="utf-8", ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*3 passed*"]) result = pytester.runpytest(testfile) result.stdout.fnmatch_lines(["*3 passed*"]) def test_override_fixture_reusing_super_fixture_parametrization( self, pytester: Pytester ) -> None: """Override a fixture at a lower level, reusing the higher-level fixture that is parametrized (#1953). """ pytester.makeconftest( """ import pytest @pytest.fixture(params=[1, 2]) def foo(request): return request.param """ ) pytester.makepyfile( """ import pytest @pytest.fixture def foo(foo): return foo * 2 def test_spam(foo): assert foo in (2, 4) """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*2 passed*"]) def test_override_parametrize_fixture_and_indirect( self, pytester: Pytester ) -> None: """Override a fixture at a lower level, reusing the higher-level fixture that is parametrized, while also using indirect parametrization. """ pytester.makeconftest( """ import pytest @pytest.fixture(params=[1, 2]) def foo(request): return request.param """ ) pytester.makepyfile( """ import pytest @pytest.fixture def foo(foo): return foo * 2 @pytest.fixture def bar(request): return request.param * 100 @pytest.mark.parametrize("bar", [42], indirect=True) def test_spam(bar, foo): assert bar == 4200 assert foo in (2, 4) """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*2 passed*"]) def test_override_top_level_fixture_reusing_super_fixture_parametrization( self, pytester: Pytester ) -> None: """Same as the above test, but with another level of overwriting.""" pytester.makeconftest( """ import pytest @pytest.fixture(params=['unused', 'unused']) def foo(request): return request.param """ ) pytester.makepyfile( """ import pytest @pytest.fixture(params=[1, 2]) def foo(request): return request.param class Test: @pytest.fixture def foo(self, foo): return foo * 2 def test_spam(self, foo): assert foo in (2, 4) """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*2 passed*"]) def test_override_parametrized_fixture_with_new_parametrized_fixture( self, pytester: Pytester ) -> None: """Overriding a parametrized fixture, while also parametrizing the new fixture and simultaneously requesting the overwritten fixture as parameter, yields the same value as ``request.param``. """ pytester.makeconftest( """ import pytest @pytest.fixture(params=['ignored', 'ignored']) def foo(request): return request.param """ ) pytester.makepyfile( """ import pytest @pytest.fixture(params=[10, 20]) def foo(foo, request): assert request.param == foo return foo * 2 def test_spam(foo): assert foo in (20, 40) """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*2 passed*"]) def test_autouse_fixture_plugin(self, pytester: Pytester) -> None: # A fixture from a plugin has no baseid set, which screwed up # the autouse fixture handling. pytester.makepyfile( testplugin=""" import pytest @pytest.fixture(autouse=True) def foo(request): request.function.foo = 7 """ ) pytester.syspathinsert() pytester.makepyfile( """ pytest_plugins = 'testplugin' def test_foo(request): assert request.function.foo == 7 """ ) result = pytester.runpytest() assert result.ret == 0 def test_funcarg_lookup_error(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture def a_fixture(): pass @pytest.fixture def b_fixture(): pass @pytest.fixture def c_fixture(): pass @pytest.fixture def d_fixture(): pass """ ) pytester.makepyfile( """ def test_lookup_error(unknown): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "*ERROR at setup of test_lookup_error*", " def test_lookup_error(unknown):*", "E fixture 'unknown' not found", "> available fixtures:*a_fixture,*b_fixture,*c_fixture,*d_fixture*monkeypatch,*", # sorted "> use 'py*test --fixtures *' for help on them.", "*1 error*", ] ) result.stdout.no_fnmatch_line("*INTERNAL*") def test_fixture_excinfo_leak(self, pytester: Pytester) -> None: # on python2 sys.excinfo would leak into fixture executions pytester.makepyfile( """ import sys import traceback import pytest @pytest.fixture def leak(): if sys.exc_info()[0]: # python3 bug :) traceback.print_exc() #fails assert sys.exc_info() == (None, None, None) def test_leak(leak): if sys.exc_info()[0]: # python3 bug :) traceback.print_exc() assert sys.exc_info() == (None, None, None) """ ) result = pytester.runpytest() assert result.ret == 0 class TestRequestBasic: def test_request_attributes(self, pytester: Pytester) -> None: item = pytester.getitem( """ import pytest @pytest.fixture def something(request): pass def test_func(something): pass """ ) assert isinstance(item, Function) req = TopRequest(item, _ispytest=True) assert req.function == item.obj assert req.keywords == item.keywords assert hasattr(req.module, "test_func") assert req.cls is None assert req.function.__name__ == "test_func" assert req.config == item.config assert repr(req).find(req.function.__name__) != -1 def test_request_attributes_method(self, pytester: Pytester) -> None: (item,) = pytester.getitems( """ import pytest class TestB(object): @pytest.fixture def something(self, request): return 1 def test_func(self, something): pass """ ) assert isinstance(item, Function) req = item._request assert req.cls.__name__ == "TestB" assert req.instance.__class__ == req.cls def test_request_contains_funcarg_arg2fixturedefs(self, pytester: Pytester) -> None: modcol = pytester.getmodulecol( """ import pytest @pytest.fixture def something(request): pass class TestClass(object): def test_method(self, something): pass """ ) (item1,) = pytester.genitems([modcol]) assert isinstance(item1, Function) assert item1.name == "test_method" arg2fixturedefs = TopRequest(item1, _ispytest=True)._arg2fixturedefs assert len(arg2fixturedefs) == 1 assert arg2fixturedefs["something"][0].argname == "something" @pytest.mark.skipif( hasattr(sys, "pypy_version_info"), reason="this method of test doesn't work on pypy", ) def test_request_garbage(self, pytester: Pytester) -> None: try: import xdist # noqa: F401 except ImportError: pass else: pytest.xfail("this test is flaky when executed with xdist") pytester.makepyfile( """ import sys import pytest from _pytest.fixtures import PseudoFixtureDef import gc @pytest.fixture(autouse=True) def something(request): original = gc.get_debug() gc.set_debug(gc.DEBUG_SAVEALL) gc.collect() yield try: gc.collect() leaked = [x for _ in gc.garbage if isinstance(_, PseudoFixtureDef)] assert leaked == [] finally: gc.set_debug(original) def test_func(): pass """ ) result = pytester.runpytest_subprocess() result.stdout.fnmatch_lines(["* 1 passed in *"]) def test_getfixturevalue_recursive(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture def something(request): return 1 """ ) pytester.makepyfile( """ import pytest @pytest.fixture def something(request): return request.getfixturevalue("something") + 1 def test_func(something): assert something == 2 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_getfixturevalue_teardown(self, pytester: Pytester) -> None: """ Issue #1895 `test_inner` requests `inner` fixture, which in turn requests `resource` using `getfixturevalue`. `test_func` then requests `resource`. `resource` is teardown before `inner` because the fixture mechanism won't consider `inner` dependent on `resource` when it is used via `getfixturevalue`: `test_func` will then cause the `resource`'s finalizer to be called first because of this. """ pytester.makepyfile( """ import pytest @pytest.fixture(scope='session') def resource(): r = ['value'] yield r r.pop() @pytest.fixture(scope='session') def inner(request): resource = request.getfixturevalue('resource') assert resource == ['value'] yield assert resource == ['value'] def test_inner(inner): pass def test_func(resource): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["* 2 passed in *"]) def test_getfixturevalue(self, pytester: Pytester) -> None: item = pytester.getitem( """ import pytest @pytest.fixture def something(request): return 1 values = [2] @pytest.fixture def other(request): return values.pop() def test_func(something): pass """ ) assert isinstance(item, Function) req = item._request # Execute item's setup. item.session._setupstate.setup(item) with pytest.raises(pytest.FixtureLookupError): req.getfixturevalue("notexists") val = req.getfixturevalue("something") assert val == 1 val = req.getfixturevalue("something") assert val == 1 val2 = req.getfixturevalue("other") assert val2 == 2 val2 = req.getfixturevalue("other") # see about caching assert val2 == 2 assert item.funcargs["something"] == 1 assert len(get_public_names(item.funcargs)) == 2 assert "request" in item.funcargs def test_request_addfinalizer(self, pytester: Pytester) -> None: item = pytester.getitem( """ import pytest teardownlist = [] @pytest.fixture def something(request): request.addfinalizer(lambda: teardownlist.append(1)) def test_func(something): pass """ ) assert isinstance(item, Function) item.session._setupstate.setup(item) item._request._fillfixtures() # successively check finalization calls parent = item.getparent(pytest.Module) assert parent is not None teardownlist = parent.obj.teardownlist ss = item.session._setupstate assert not teardownlist ss.teardown_exact(None) print(ss.stack) assert teardownlist == [1] def test_request_addfinalizer_failing_setup(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [1] @pytest.fixture def myfix(request): request.addfinalizer(values.pop) assert 0 def test_fix(myfix): pass def test_finalizer_ran(): assert not values """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(failed=1, passed=1) def test_request_addfinalizer_failing_setup_module( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest values = [1, 2] @pytest.fixture(scope="module") def myfix(request): request.addfinalizer(values.pop) request.addfinalizer(values.pop) assert 0 def test_fix(myfix): pass """ ) reprec = pytester.inline_run("-s") mod = reprec.getcalls("pytest_runtest_setup")[0].item.module assert not mod.values def test_request_addfinalizer_partial_setup_failure( self, pytester: Pytester ) -> None: p = pytester.makepyfile( """ import pytest values = [] @pytest.fixture def something(request): request.addfinalizer(lambda: values.append(None)) def test_func(something, missingarg): pass def test_second(): assert len(values) == 1 """ ) result = pytester.runpytest(p) result.stdout.fnmatch_lines( ["*1 error*"] # XXX the whole module collection fails ) def test_request_subrequest_addfinalizer_exceptions( self, pytester: Pytester ) -> None: """ Ensure exceptions raised during teardown by finalizers are suppressed until all finalizers are called, then re-reaised together in an exception group (#2440) """ pytester.makepyfile( """ import pytest values = [] def _excepts(where): raise Exception('Error in %s fixture' % where) @pytest.fixture def subrequest(request): return request @pytest.fixture def something(subrequest): subrequest.addfinalizer(lambda: values.append(1)) subrequest.addfinalizer(lambda: values.append(2)) subrequest.addfinalizer(lambda: _excepts('something')) @pytest.fixture def excepts(subrequest): subrequest.addfinalizer(lambda: _excepts('excepts')) subrequest.addfinalizer(lambda: values.append(3)) def test_first(something, excepts): pass def test_second(): assert values == [3, 2, 1] """ ) result = pytester.runpytest() result.assert_outcomes(passed=2, errors=1) result.stdout.fnmatch_lines( [ ' | *ExceptionGroup: errors while tearing down fixture "subrequest" of (2 sub-exceptions)', # noqa: E501 " +-+---------------- 1 ----------------", " | Exception: Error in something fixture", " +---------------- 2 ----------------", " | Exception: Error in excepts fixture", " +------------------------------------", ], ) def test_request_getmodulepath(self, pytester: Pytester) -> None: modcol = pytester.getmodulecol("def test_somefunc(): pass") (item,) = pytester.genitems([modcol]) assert isinstance(item, Function) req = TopRequest(item, _ispytest=True) assert req.path == modcol.path def test_request_fixturenames(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest from _pytest.pytester import get_public_names @pytest.fixture() def arg1(): pass @pytest.fixture() def farg(arg1): pass @pytest.fixture(autouse=True) def sarg(tmp_path): pass def test_function(request, farg): assert set(get_public_names(request.fixturenames)) == \ set(["sarg", "arg1", "request", "farg", "tmp_path", "tmp_path_factory"]) """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_request_fixturenames_dynamic_fixture(self, pytester: Pytester) -> None: """Regression test for #3057""" pytester.copy_example("fixtures/test_getfixturevalue_dynamic.py") result = pytester.runpytest() result.stdout.fnmatch_lines(["*1 passed*"]) def test_setupdecorator_and_xunit(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope='module', autouse=True) def setup_module(): values.append("module") @pytest.fixture(autouse=True) def setup_function(): values.append("function") def test_func(): pass class TestClass(object): @pytest.fixture(scope="class", autouse=True) def setup_class(self): values.append("class") @pytest.fixture(autouse=True) def setup_method(self): values.append("method") def test_method(self): pass def test_all(): assert values == ["module", "function", "class", "function", "method", "function"] """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=3) def test_fixtures_sub_subdir_normalize_sep(self, pytester: Pytester) -> None: # this tests that normalization of nodeids takes place b = pytester.path.joinpath("tests", "unit") b.mkdir(parents=True) b.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture def arg1(): pass """ ), encoding="utf-8", ) p = b.joinpath("test_module.py") p.write_text("def test_func(arg1): pass", encoding="utf-8") result = pytester.runpytest(p, "--fixtures") assert result.ret == 0 result.stdout.fnmatch_lines( """ *fixtures defined*conftest* *arg1* """ ) def test_show_fixtures_color_yes(self, pytester: Pytester) -> None: pytester.makepyfile("def test_this(): assert 1") result = pytester.runpytest("--color=yes", "--fixtures") assert "\x1b[32mtmp_path" in result.stdout.str() def test_newstyle_with_request(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture() def arg(request): pass def test_1(arg): pass """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_setupcontext_no_param(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=[1,2]) def arg(request): return request.param @pytest.fixture(autouse=True) def mysetup(request, arg): assert not hasattr(request, "param") def test_1(arg): assert arg in (1,2) """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) class TestRequestSessionScoped: @pytest.fixture(scope="session") def session_request(self, request): return request @pytest.mark.parametrize("name", ["path", "module"]) def test_session_scoped_unavailable_attributes(self, session_request, name): with pytest.raises( AttributeError, match=f"{name} not available in session-scoped context", ): getattr(session_request, name) class TestRequestMarking: def test_applymarker(self, pytester: Pytester) -> None: item1, item2 = pytester.getitems( """ import pytest @pytest.fixture def something(request): pass class TestClass(object): def test_func1(self, something): pass def test_func2(self, something): pass """ ) assert isinstance(item1, Function) req1 = TopRequest(item1, _ispytest=True) assert "xfail" not in item1.keywords req1.applymarker(pytest.mark.xfail) assert "xfail" in item1.keywords assert "skipif" not in item1.keywords req1.applymarker(pytest.mark.skipif) assert "skipif" in item1.keywords with pytest.raises(ValueError): req1.applymarker(42) # type: ignore[arg-type] def test_accesskeywords(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture() def keywords(request): return request.keywords @pytest.mark.XYZ def test_function(keywords): assert keywords["XYZ"] assert "abc" not in keywords """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_accessmarker_dynamic(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture() def keywords(request): return request.keywords @pytest.fixture(scope="class", autouse=True) def marking(request): request.applymarker(pytest.mark.XYZ("hello")) """ ) pytester.makepyfile( """ import pytest def test_fun1(keywords): assert keywords["XYZ"] is not None assert "abc" not in keywords def test_fun2(keywords): assert keywords["XYZ"] is not None assert "abc" not in keywords """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) class TestFixtureUsages: def test_noargfixturedec(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture def arg1(): return 1 def test_func(arg1): assert arg1 == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_receives_funcargs(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture() def arg1(): return 1 @pytest.fixture() def arg2(arg1): return arg1 + 1 def test_add(arg2): assert arg2 == 2 def test_all(arg1, arg2): assert arg1 == 1 assert arg2 == 2 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_receives_funcargs_scope_mismatch(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="function") def arg1(): return 1 @pytest.fixture(scope="module") def arg2(arg1): return arg1 + 1 def test_add(arg2): assert arg2 == 2 """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "*ScopeMismatch*Requesting fixture stack*", "test_receives_funcargs_scope_mismatch.py:6: def arg2(arg1)", "Requested fixture:", "test_receives_funcargs_scope_mismatch.py:2: def arg1()", "*1 error*", ] ) def test_receives_funcargs_scope_mismatch_issue660( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="function") def arg1(): return 1 @pytest.fixture(scope="module") def arg2(arg1): return arg1 + 1 def test_add(arg1, arg2): assert arg2 == 2 """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "*ScopeMismatch*Requesting fixture stack*", "* def arg2(arg1)", "Requested fixture:", "* def arg1()", "*1 error*", ], ) def test_invalid_scope(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="functions") def badscope(): pass def test_nothing(badscope): pass """ ) result = pytester.runpytest_inprocess() result.stdout.fnmatch_lines( "*Fixture 'badscope' from test_invalid_scope.py got an unexpected scope value 'functions'" ) @pytest.mark.parametrize("scope", ["function", "session"]) def test_parameters_without_eq_semantics(self, scope, pytester: Pytester) -> None: pytester.makepyfile( f""" class NoEq1: # fails on `a == b` statement def __eq__(self, _): raise RuntimeError class NoEq2: # fails on `if a == b:` statement def __eq__(self, _): class NoBool: def __bool__(self): raise RuntimeError return NoBool() import pytest @pytest.fixture(params=[NoEq1(), NoEq2()], scope={scope!r}) def no_eq(request): return request.param def test1(no_eq): pass def test2(no_eq): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*4 passed*"]) def test_funcarg_parametrized_and_used_twice(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(params=[1,2]) def arg1(request): values.append(1) return request.param @pytest.fixture() def arg2(arg1): return arg1 + 1 def test_add(arg1, arg2): assert arg2 == arg1 + 1 assert len(values) == arg1 """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*2 passed*"]) def test_factory_uses_unknown_funcarg_as_dependency_error( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest @pytest.fixture() def fail(missing): return @pytest.fixture() def call_fail(fail): return def test_missing(call_fail): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( """ *pytest.fixture()* *def call_fail(fail)* *pytest.fixture()* *def fail* *fixture*'missing'*not found* """ ) def test_factory_setup_as_classes_fails(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest class arg1(object): def __init__(self, request): self.x = 1 arg1 = pytest.fixture()(arg1) """ ) reprec = pytester.inline_run() values = reprec.getfailedcollections() assert len(values) == 1 def test_usefixtures_marker(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="class") def myfix(request): request.cls.hello = "world" values.append(1) class TestClass(object): def test_one(self): assert self.hello == "world" assert len(values) == 1 def test_two(self): assert self.hello == "world" assert len(values) == 1 pytest.mark.usefixtures("myfix")(TestClass) """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_usefixtures_ini(self, pytester: Pytester) -> None: pytester.makeini( """ [pytest] usefixtures = myfix """ ) pytester.makeconftest( """ import pytest @pytest.fixture(scope="class") def myfix(request): request.cls.hello = "world" """ ) pytester.makepyfile( """ class TestClass(object): def test_one(self): assert self.hello == "world" def test_two(self): assert self.hello == "world" """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_usefixtures_seen_in_showmarkers(self, pytester: Pytester) -> None: result = pytester.runpytest("--markers") result.stdout.fnmatch_lines( """ *usefixtures(fixturename1*mark tests*fixtures* """ ) def test_request_instance_issue203(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest class TestClass(object): @pytest.fixture def setup1(self, request): assert self == request.instance self.arg1 = 1 def test_hello(self, setup1): assert self.arg1 == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_fixture_parametrized_with_iterator(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] def f(): yield 1 yield 2 dec = pytest.fixture(scope="module", params=f()) @dec def arg(request): return request.param @dec def arg2(request): return request.param def test_1(arg): values.append(arg) def test_2(arg2): values.append(arg2*10) """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=4) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values assert values == [1, 2, 10, 20] def test_setup_functions_as_fixtures(self, pytester: Pytester) -> None: """Ensure setup_* methods obey fixture scope rules (#517, #3094).""" pytester.makepyfile( """ import pytest DB_INITIALIZED = None @pytest.fixture(scope="session", autouse=True) def db(): global DB_INITIALIZED DB_INITIALIZED = True yield DB_INITIALIZED = False def setup_module(): assert DB_INITIALIZED def teardown_module(): assert DB_INITIALIZED class TestClass(object): def setup_method(self, method): assert DB_INITIALIZED def teardown_method(self, method): assert DB_INITIALIZED def test_printer_1(self): pass def test_printer_2(self): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["* 2 passed in *"]) class TestFixtureManagerParseFactories: @pytest.fixture def pytester(self, pytester: Pytester) -> Pytester: pytester.makeconftest( """ import pytest @pytest.fixture def hello(request): return "conftest" @pytest.fixture def fm(request): return request._fixturemanager @pytest.fixture def item(request): return request._pyfuncitem """ ) return pytester def test_parsefactories_evil_objects_issue214(self, pytester: Pytester) -> None: pytester.makepyfile( """ class A(object): def __call__(self): pass def __getattr__(self, name): raise RuntimeError() a = A() def test_hello(): pass """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1, failed=0) def test_parsefactories_conftest(self, pytester: Pytester) -> None: pytester.makepyfile( """ def test_hello(item, fm): for name in ("fm", "hello", "item"): faclist = fm.getfixturedefs(name, item) assert len(faclist) == 1 fac = faclist[0] assert fac.func.__name__ == name """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=1) def test_parsefactories_conftest_and_module_and_class( self, pytester: Pytester ) -> None: pytester.makepyfile( """\ import pytest @pytest.fixture def hello(request): return "module" class TestClass(object): @pytest.fixture def hello(self, request): return "class" def test_hello(self, item, fm): faclist = fm.getfixturedefs("hello", item) print(faclist) assert len(faclist) == 3 assert faclist[0].func(item._request) == "conftest" assert faclist[1].func(item._request) == "module" assert faclist[2].func(item._request) == "class" """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=1) def test_parsefactories_relative_node_ids( self, pytester: Pytester, monkeypatch: MonkeyPatch ) -> None: # example mostly taken from: # https://mail.python.org/pipermail/pytest-dev/2014-September/002617.html runner = pytester.mkdir("runner") package = pytester.mkdir("package") package.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture def one(): return 1 """ ), encoding="utf-8", ) package.joinpath("test_x.py").write_text( textwrap.dedent( """\ def test_x(one): assert one == 1 """ ), encoding="utf-8", ) sub = package.joinpath("sub") sub.mkdir() sub.joinpath("__init__.py").touch() sub.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture def one(): return 2 """ ), encoding="utf-8", ) sub.joinpath("test_y.py").write_text( textwrap.dedent( """\ def test_x(one): assert one == 2 """ ), encoding="utf-8", ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) with monkeypatch.context() as mp: mp.chdir(runner) reprec = pytester.inline_run("..") reprec.assertoutcome(passed=2) def test_package_xunit_fixture(self, pytester: Pytester) -> None: pytester.makepyfile( __init__="""\ values = [] """ ) package = pytester.mkdir("package") package.joinpath("__init__.py").write_text( textwrap.dedent( """\ from .. import values def setup_module(): values.append("package") def teardown_module(): values[:] = [] """ ), encoding="utf-8", ) package.joinpath("test_x.py").write_text( textwrap.dedent( """\ from .. import values def test_x(): assert values == ["package"] """ ), encoding="utf-8", ) package = pytester.mkdir("package2") package.joinpath("__init__.py").write_text( textwrap.dedent( """\ from .. import values def setup_module(): values.append("package2") def teardown_module(): values[:] = [] """ ), encoding="utf-8", ) package.joinpath("test_x.py").write_text( textwrap.dedent( """\ from .. import values def test_x(): assert values == ["package2"] """ ), encoding="utf-8", ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_package_fixture_complex(self, pytester: Pytester) -> None: pytester.makepyfile( __init__="""\ values = [] """ ) pytester.syspathinsert(pytester.path.name) package = pytester.mkdir("package") package.joinpath("__init__.py").write_text("", encoding="utf-8") package.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest from .. import values @pytest.fixture(scope="package") def one(): values.append("package") yield values values.pop() @pytest.fixture(scope="package", autouse=True) def two(): values.append("package-auto") yield values values.pop() """ ), encoding="utf-8", ) package.joinpath("test_x.py").write_text( textwrap.dedent( """\ from .. import values def test_package_autouse(): assert values == ["package-auto"] def test_package(one): assert values == ["package-auto", "package"] """ ), encoding="utf-8", ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_collect_custom_items(self, pytester: Pytester) -> None: pytester.copy_example("fixtures/custom_item") result = pytester.runpytest("foo") result.stdout.fnmatch_lines(["*passed*"]) class TestAutouseDiscovery: @pytest.fixture def pytester(self, pytester: Pytester) -> Pytester: pytester.makeconftest( """ import pytest @pytest.fixture(autouse=True) def perfunction(request, tmp_path): pass @pytest.fixture() def arg1(tmp_path): pass @pytest.fixture(autouse=True) def perfunction2(arg1): pass @pytest.fixture def fm(request): return request._fixturemanager @pytest.fixture def item(request): return request._pyfuncitem """ ) return pytester def test_parsefactories_conftest(self, pytester: Pytester) -> None: pytester.makepyfile( """ from _pytest.pytester import get_public_names def test_check_setup(item, fm): autousenames = list(fm._getautousenames(item)) assert len(get_public_names(autousenames)) == 2 assert "perfunction2" in autousenames assert "perfunction" in autousenames """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=1) def test_two_classes_separated_autouse(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest class TestA(object): values = [] @pytest.fixture(autouse=True) def setup1(self): self.values.append(1) def test_setup1(self): assert self.values == [1] class TestB(object): values = [] @pytest.fixture(autouse=True) def setup2(self): self.values.append(1) def test_setup2(self): assert self.values == [1] """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_setup_at_classlevel(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest class TestClass(object): @pytest.fixture(autouse=True) def permethod(self, request): request.instance.funcname = request.function.__name__ def test_method1(self): assert self.funcname == "test_method1" def test_method2(self): assert self.funcname == "test_method2" """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=2) @pytest.mark.xfail(reason="'enabled' feature not implemented") def test_setup_enabled_functionnode(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest def enabled(parentnode, markers): return "needsdb" in markers @pytest.fixture(params=[1,2]) def db(request): return request.param @pytest.fixture(enabled=enabled, autouse=True) def createdb(db): pass def test_func1(request): assert "db" not in request.fixturenames @pytest.mark.needsdb def test_func2(request): assert "db" in request.fixturenames """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=2) def test_callables_nocode(self, pytester: Pytester) -> None: """An imported mock.call would break setup/factory discovery due to it being callable and __code__ not being a code object.""" pytester.makepyfile( """ class _call(tuple): def __call__(self, *k, **kw): pass def __getattr__(self, k): return self call = _call() """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(failed=0, passed=0) def test_autouse_in_conftests(self, pytester: Pytester) -> None: a = pytester.mkdir("a") b = pytester.mkdir("a1") conftest = pytester.makeconftest( """ import pytest @pytest.fixture(autouse=True) def hello(): xxx """ ) conftest.rename(a.joinpath(conftest.name)) a.joinpath("test_something.py").write_text( "def test_func(): pass", encoding="utf-8" ) b.joinpath("test_otherthing.py").write_text( "def test_func(): pass", encoding="utf-8" ) result = pytester.runpytest() result.stdout.fnmatch_lines( """ *1 passed*1 error* """ ) def test_autouse_in_module_and_two_classes(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(autouse=True) def append1(): values.append("module") def test_x(): assert values == ["module"] class TestA(object): @pytest.fixture(autouse=True) def append2(self): values.append("A") def test_hello(self): assert values == ["module", "module", "A"], values class TestA2(object): def test_world(self): assert values == ["module", "module", "A", "module"], values """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=3) class TestAutouseManagement: def test_autouse_conftest_mid_directory(self, pytester: Pytester) -> None: pkgdir = pytester.mkpydir("xyz123") pkgdir.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture(autouse=True) def app(): import sys sys._myapp = "hello" """ ), encoding="utf-8", ) sub = pkgdir.joinpath("tests") sub.mkdir() t = sub.joinpath("test_app.py") t.touch() t.write_text( textwrap.dedent( """\ import sys def test_app(): assert sys._myapp == "hello" """ ), encoding="utf-8", ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=1) def test_funcarg_and_setup(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="module") def arg(): values.append(1) return 0 @pytest.fixture(scope="module", autouse=True) def something(arg): values.append(2) def test_hello(arg): assert len(values) == 2 assert values == [1,2] assert arg == 0 def test_hello2(arg): assert len(values) == 2 assert values == [1,2] assert arg == 0 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_uses_parametrized_resource(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(params=[1,2]) def arg(request): return request.param @pytest.fixture(autouse=True) def something(arg): values.append(arg) def test_hello(): if len(values) == 1: assert values == [1] elif len(values) == 2: assert values == [1, 2] else: 0/0 """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=2) def test_session_parametrized_function(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="session", params=[1,2]) def arg(request): return request.param @pytest.fixture(scope="function", autouse=True) def append(request, arg): if request.function.__name__ == "test_some": values.append(arg) def test_some(): pass def test_result(arg): assert len(values) == arg assert values[:arg] == [1,2][:arg] """ ) reprec = pytester.inline_run("-v", "-s") reprec.assertoutcome(passed=4) def test_class_function_parametrization_finalization( self, pytester: Pytester ) -> None: p = pytester.makeconftest( """ import pytest import pprint values = [] @pytest.fixture(scope="function", params=[1,2]) def farg(request): return request.param @pytest.fixture(scope="class", params=list("ab")) def carg(request): return request.param @pytest.fixture(scope="function", autouse=True) def append(request, farg, carg): def fin(): values.append("fin_%s%s" % (carg, farg)) request.addfinalizer(fin) """ ) pytester.makepyfile( """ import pytest class TestClass(object): def test_1(self): pass class TestClass2(object): def test_2(self): pass """ ) reprec = pytester.inline_run("-v", "-s", "--confcutdir", pytester.path) reprec.assertoutcome(passed=8) config = reprec.getcalls("pytest_unconfigure")[0].config values = config.pluginmanager._getconftestmodules(p)[0].values assert values == ["fin_a1", "fin_a2", "fin_b1", "fin_b2"] * 2 def test_scope_ordering(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="function", autouse=True) def fappend2(): values.append(2) @pytest.fixture(scope="class", autouse=True) def classappend3(): values.append(3) @pytest.fixture(scope="module", autouse=True) def mappend(): values.append(1) class TestHallo(object): def test_method(self): assert values == [1,3,2] """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_parametrization_setup_teardown_ordering(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] def pytest_generate_tests(metafunc): if metafunc.cls is None: assert metafunc.function is test_finish if metafunc.cls is not None: metafunc.parametrize("item", [1,2], scope="class") class TestClass(object): @pytest.fixture(scope="class", autouse=True) def addteardown(self, item, request): values.append("setup-%d" % item) request.addfinalizer(lambda: values.append("teardown-%d" % item)) def test_step1(self, item): values.append("step1-%d" % item) def test_step2(self, item): values.append("step2-%d" % item) def test_finish(): print(values) assert values == ["setup-1", "step1-1", "step2-1", "teardown-1", "setup-2", "step1-2", "step2-2", "teardown-2",] """ ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=5) def test_ordering_autouse_before_explicit(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(autouse=True) def fix1(): values.append(1) @pytest.fixture() def arg1(): values.append(2) def test_hello(arg1): assert values == [1,2] """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) @pytest.mark.parametrize("param1", ["", "params=[1]"], ids=["p00", "p01"]) @pytest.mark.parametrize("param2", ["", "params=[1]"], ids=["p10", "p11"]) def test_ordering_dependencies_torndown_first( self, pytester: Pytester, param1, param2 ) -> None: """#226""" pytester.makepyfile( """ import pytest values = [] @pytest.fixture(%(param1)s) def arg1(request): request.addfinalizer(lambda: values.append("fin1")) values.append("new1") @pytest.fixture(%(param2)s) def arg2(request, arg1): request.addfinalizer(lambda: values.append("fin2")) values.append("new2") def test_arg(arg2): pass def test_check(): assert values == ["new1", "new2", "fin2", "fin1"] """ # noqa: UP031 (python syntax issues) % locals() ) reprec = pytester.inline_run("-s") reprec.assertoutcome(passed=2) class TestFixtureMarker: def test_parametrize(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=["a", "b", "c"]) def arg(request): return request.param values = [] def test_param(arg): values.append(arg) def test_result(): assert values == list("abc") """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=4) def test_multiple_parametrization_issue_736(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=[1,2,3]) def foo(request): return request.param @pytest.mark.parametrize('foobar', [4,5,6]) def test_issue(foo, foobar): assert foo in [1,2,3] assert foobar in [4,5,6] """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=9) @pytest.mark.parametrize( "param_args", ["'fixt, val'", "'fixt,val'", "['fixt', 'val']", "('fixt', 'val')"], ) def test_override_parametrized_fixture_issue_979( self, pytester: Pytester, param_args ) -> None: """Make sure a parametrized argument can override a parametrized fixture. This was a regression introduced in the fix for #736. """ pytester.makepyfile( f""" import pytest @pytest.fixture(params=[1, 2]) def fixt(request): return request.param @pytest.mark.parametrize({param_args}, [(3, 'x'), (4, 'x')]) def test_foo(fixt, val): pass """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_scope_session(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="module") def arg(): values.append(1) return 1 def test_1(arg): assert arg == 1 def test_2(arg): assert arg == 1 assert len(values) == 1 class TestClass(object): def test3(self, arg): assert arg == 1 assert len(values) == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=3) def test_scope_session_exc(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="session") def fix(): values.append(1) pytest.skip('skipping') def test_1(fix): pass def test_2(fix): pass def test_last(): assert values == [1] """ ) reprec = pytester.inline_run() reprec.assertoutcome(skipped=2, passed=1) def test_scope_session_exc_two_fix(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] m = [] @pytest.fixture(scope="session") def a(): values.append(1) pytest.skip('skipping') @pytest.fixture(scope="session") def b(a): m.append(1) def test_1(b): pass def test_2(b): pass def test_last(): assert values == [1] assert m == [] """ ) reprec = pytester.inline_run() reprec.assertoutcome(skipped=2, passed=1) def test_scope_exc(self, pytester: Pytester) -> None: pytester.makepyfile( test_foo=""" def test_foo(fix): pass """, test_bar=""" def test_bar(fix): pass """, conftest=""" import pytest reqs = [] @pytest.fixture(scope="session") def fix(request): reqs.append(1) pytest.skip() @pytest.fixture def req_list(): return reqs """, test_real=""" def test_last(req_list): assert req_list == [1] """, ) reprec = pytester.inline_run() reprec.assertoutcome(skipped=2, passed=1) def test_scope_module_uses_session(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="module") def arg(): values.append(1) return 1 def test_1(arg): assert arg == 1 def test_2(arg): assert arg == 1 assert len(values) == 1 class TestClass(object): def test3(self, arg): assert arg == 1 assert len(values) == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=3) def test_scope_module_and_finalizer(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest finalized_list = [] created_list = [] @pytest.fixture(scope="module") def arg(request): created_list.append(1) assert request.scope == "module" request.addfinalizer(lambda: finalized_list.append(1)) @pytest.fixture def created(request): return len(created_list) @pytest.fixture def finalized(request): return len(finalized_list) """ ) pytester.makepyfile( test_mod1=""" def test_1(arg, created, finalized): assert created == 1 assert finalized == 0 def test_2(arg, created, finalized): assert created == 1 assert finalized == 0""", test_mod2=""" def test_3(arg, created, finalized): assert created == 2 assert finalized == 1""", test_mode3=""" def test_4(arg, created, finalized): assert created == 3 assert finalized == 2 """, ) reprec = pytester.inline_run() reprec.assertoutcome(passed=4) def test_scope_mismatch_various(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest finalized = [] created = [] @pytest.fixture(scope="function") def arg(request): pass """ ) pytester.makepyfile( test_mod1=""" import pytest @pytest.fixture(scope="session") def arg(request): request.getfixturevalue("arg") def test_1(arg): pass """ ) result = pytester.runpytest() assert result.ret != 0 result.stdout.fnmatch_lines( ["*ScopeMismatch*You tried*function*session*request*"] ) def test_scope_mismatch_already_computed_dynamic(self, pytester: Pytester) -> None: pytester.makepyfile( test_it=""" import pytest @pytest.fixture(scope="function") def fixfunc(): pass @pytest.fixture(scope="module") def fixmod(fixfunc): pass def test_it(request, fixfunc): request.getfixturevalue("fixmod") """, ) result = pytester.runpytest() assert result.ret == ExitCode.TESTS_FAILED result.stdout.fnmatch_lines( [ "*ScopeMismatch*Requesting fixture stack*", "test_it.py:6: def fixmod(fixfunc)", "Requested fixture:", "test_it.py:3: def fixfunc()", ] ) def test_dynamic_scope(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest def pytest_addoption(parser): parser.addoption("--extend-scope", action="store_true", default=False) def dynamic_scope(fixture_name, config): if config.getoption("--extend-scope"): return "session" return "function" @pytest.fixture(scope=dynamic_scope) def dynamic_fixture(calls=[]): calls.append("call") return len(calls) """ ) pytester.makepyfile( """ def test_first(dynamic_fixture): assert dynamic_fixture == 1 def test_second(dynamic_fixture): assert dynamic_fixture == 2 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) reprec = pytester.inline_run("--extend-scope") reprec.assertoutcome(passed=1, failed=1) def test_dynamic_scope_bad_return(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest def dynamic_scope(**_): return "wrong-scope" @pytest.fixture(scope=dynamic_scope) def fixture(): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( "Fixture 'fixture' from test_dynamic_scope_bad_return.py " "got an unexpected scope value 'wrong-scope'" ) def test_register_only_with_mark(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture() def arg(): return 1 """ ) pytester.makepyfile( test_mod1=""" import pytest @pytest.fixture() def arg(arg): return arg + 1 def test_1(arg): assert arg == 2 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_parametrize_and_scope(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module", params=["a", "b", "c"]) def arg(request): return request.param values = [] def test_param(arg): values.append(arg) """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=3) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values assert len(values) == 3 assert "a" in values assert "b" in values assert "c" in values def test_scope_mismatch(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture(scope="function") def arg(request): pass """ ) pytester.makepyfile( """ import pytest @pytest.fixture(scope="session") def arg(arg): pass def test_mismatch(arg): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines(["*ScopeMismatch*", "*1 error*"]) def test_parametrize_separated_order(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module", params=[1, 2]) def arg(request): return request.param values = [] def test_1(arg): values.append(arg) def test_2(arg): values.append(arg) """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=4) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values assert values == [1, 1, 2, 2] def test_module_parametrized_ordering(self, pytester: Pytester) -> None: pytester.makeini( """ [pytest] console_output_style=classic """ ) pytester.makeconftest( """ import pytest @pytest.fixture(scope="session", params="s1 s2".split()) def sarg(): pass @pytest.fixture(scope="module", params="m1 m2".split()) def marg(): pass """ ) pytester.makepyfile( test_mod1=""" def test_func(sarg): pass def test_func1(marg): pass """, test_mod2=""" def test_func2(sarg): pass def test_func3(sarg, marg): pass def test_func3b(sarg, marg): pass def test_func4(marg): pass """, ) result = pytester.runpytest("-v") result.stdout.fnmatch_lines( """ test_mod1.py::test_func[s1] PASSED test_mod2.py::test_func2[s1] PASSED test_mod2.py::test_func3[s1-m1] PASSED test_mod2.py::test_func3b[s1-m1] PASSED test_mod2.py::test_func3[s1-m2] PASSED test_mod2.py::test_func3b[s1-m2] PASSED test_mod1.py::test_func[s2] PASSED test_mod2.py::test_func2[s2] PASSED test_mod2.py::test_func3[s2-m1] PASSED test_mod2.py::test_func3b[s2-m1] PASSED test_mod2.py::test_func4[m1] PASSED test_mod2.py::test_func3[s2-m2] PASSED test_mod2.py::test_func3b[s2-m2] PASSED test_mod2.py::test_func4[m2] PASSED test_mod1.py::test_func1[m1] PASSED test_mod1.py::test_func1[m2] PASSED """ ) def test_dynamic_parametrized_ordering(self, pytester: Pytester) -> None: pytester.makeini( """ [pytest] console_output_style=classic """ ) pytester.makeconftest( """ import pytest def pytest_configure(config): class DynamicFixturePlugin(object): @pytest.fixture(scope='session', params=['flavor1', 'flavor2']) def flavor(self, request): return request.param config.pluginmanager.register(DynamicFixturePlugin(), 'flavor-fixture') @pytest.fixture(scope='session', params=['vxlan', 'vlan']) def encap(request): return request.param @pytest.fixture(scope='session', autouse='True') def reprovision(request, flavor, encap): pass """ ) pytester.makepyfile( """ def test(reprovision): pass def test2(reprovision): pass """ ) result = pytester.runpytest("-v") result.stdout.fnmatch_lines( """ test_dynamic_parametrized_ordering.py::test[flavor1-vxlan] PASSED test_dynamic_parametrized_ordering.py::test2[flavor1-vxlan] PASSED test_dynamic_parametrized_ordering.py::test[flavor1-vlan] PASSED test_dynamic_parametrized_ordering.py::test2[flavor1-vlan] PASSED test_dynamic_parametrized_ordering.py::test[flavor2-vlan] PASSED test_dynamic_parametrized_ordering.py::test2[flavor2-vlan] PASSED test_dynamic_parametrized_ordering.py::test[flavor2-vxlan] PASSED test_dynamic_parametrized_ordering.py::test2[flavor2-vxlan] PASSED """ ) def test_class_ordering(self, pytester: Pytester) -> None: pytester.makeini( """ [pytest] console_output_style=classic """ ) pytester.makeconftest( """ import pytest values = [] @pytest.fixture(scope="function", params=[1,2]) def farg(request): return request.param @pytest.fixture(scope="class", params=list("ab")) def carg(request): return request.param @pytest.fixture(scope="function", autouse=True) def append(request, farg, carg): def fin(): values.append("fin_%s%s" % (carg, farg)) request.addfinalizer(fin) """ ) pytester.makepyfile( """ import pytest class TestClass2(object): def test_1(self): pass def test_2(self): pass class TestClass(object): def test_3(self): pass """ ) result = pytester.runpytest("-vs") result.stdout.re_match_lines( r""" test_class_ordering.py::TestClass2::test_1\[a-1\] PASSED test_class_ordering.py::TestClass2::test_1\[a-2\] PASSED test_class_ordering.py::TestClass2::test_2\[a-1\] PASSED test_class_ordering.py::TestClass2::test_2\[a-2\] PASSED test_class_ordering.py::TestClass2::test_1\[b-1\] PASSED test_class_ordering.py::TestClass2::test_1\[b-2\] PASSED test_class_ordering.py::TestClass2::test_2\[b-1\] PASSED test_class_ordering.py::TestClass2::test_2\[b-2\] PASSED test_class_ordering.py::TestClass::test_3\[a-1\] PASSED test_class_ordering.py::TestClass::test_3\[a-2\] PASSED test_class_ordering.py::TestClass::test_3\[b-1\] PASSED test_class_ordering.py::TestClass::test_3\[b-2\] PASSED """ ) def test_parametrize_separated_order_higher_scope_first( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="function", params=[1, 2]) def arg(request): param = request.param request.addfinalizer(lambda: values.append("fin:%s" % param)) values.append("create:%s" % param) return request.param @pytest.fixture(scope="module", params=["mod1", "mod2"]) def modarg(request): param = request.param request.addfinalizer(lambda: values.append("fin:%s" % param)) values.append("create:%s" % param) return request.param values = [] def test_1(arg): values.append("test1") def test_2(modarg): values.append("test2") def test_3(arg, modarg): values.append("test3") def test_4(modarg, arg): values.append("test4") """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=12) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values expected = [ "create:1", "test1", "fin:1", "create:2", "test1", "fin:2", "create:mod1", "test2", "create:1", "test3", "fin:1", "create:2", "test3", "fin:2", "create:1", "test4", "fin:1", "create:2", "test4", "fin:2", "fin:mod1", "create:mod2", "test2", "create:1", "test3", "fin:1", "create:2", "test3", "fin:2", "create:1", "test4", "fin:1", "create:2", "test4", "fin:2", "fin:mod2", ] import pprint pprint.pprint(list(zip(values, expected))) assert values == expected def test_parametrized_fixture_teardown_order(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=[1,2], scope="class") def param1(request): return request.param values = [] class TestClass(object): @classmethod @pytest.fixture(scope="class", autouse=True) def setup1(self, request, param1): values.append(1) request.addfinalizer(self.teardown1) @classmethod def teardown1(self): assert values.pop() == 1 @pytest.fixture(scope="class", autouse=True) def setup2(self, request, param1): values.append(2) request.addfinalizer(self.teardown2) @classmethod def teardown2(self): assert values.pop() == 2 def test(self): pass def test_finish(): assert not values """ ) result = pytester.runpytest("-v") result.stdout.fnmatch_lines( """ *3 passed* """ ) result.stdout.no_fnmatch_line("*error*") def test_fixture_finalizer(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest import sys @pytest.fixture def browser(request): def finalize(): sys.stdout.write_text('Finalized', encoding='utf-8') request.addfinalizer(finalize) return {} """ ) b = pytester.mkdir("subdir") b.joinpath("test_overridden_fixture_finalizer.py").write_text( textwrap.dedent( """\ import pytest @pytest.fixture def browser(browser): browser['visited'] = True return browser def test_browser(browser): assert browser['visited'] is True """ ), encoding="utf-8", ) reprec = pytester.runpytest("-s") for test in ["test_browser"]: reprec.stdout.fnmatch_lines(["*Finalized*"]) def test_class_scope_with_normal_tests(self, pytester: Pytester) -> None: testpath = pytester.makepyfile( """ import pytest class Box(object): value = 0 @pytest.fixture(scope='class') def a(request): Box.value += 1 return Box.value def test_a(a): assert a == 1 class Test1(object): def test_b(self, a): assert a == 2 class Test2(object): def test_c(self, a): assert a == 3""" ) reprec = pytester.inline_run(testpath) for test in ["test_a", "test_b", "test_c"]: assert reprec.matchreport(test).passed def test_request_is_clean(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(params=[1, 2]) def fix(request): request.addfinalizer(lambda: values.append(request.param)) def test_fix(fix): pass """ ) reprec = pytester.inline_run("-s") values = reprec.getcalls("pytest_runtest_call")[0].item.module.values assert values == [1, 2] def test_parametrize_separated_lifecycle(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope="module", params=[1, 2]) def arg(request): x = request.param request.addfinalizer(lambda: values.append("fin%s" % x)) return request.param def test_1(arg): values.append(arg) def test_2(arg): values.append(arg) """ ) reprec = pytester.inline_run("-vs") reprec.assertoutcome(passed=4) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values import pprint pprint.pprint(values) # assert len(values) == 6 assert values[0] == values[1] == 1 assert values[2] == "fin1" assert values[3] == values[4] == 2 assert values[5] == "fin2" def test_parametrize_function_scoped_finalizers_called( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="function", params=[1, 2]) def arg(request): x = request.param request.addfinalizer(lambda: values.append("fin%s" % x)) return request.param values = [] def test_1(arg): values.append(arg) def test_2(arg): values.append(arg) def test_3(): assert len(values) == 8 assert values == [1, "fin1", 2, "fin2", 1, "fin1", 2, "fin2"] """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=5) @pytest.mark.parametrize("scope", ["session", "function", "module"]) def test_finalizer_order_on_parametrization( self, scope, pytester: Pytester ) -> None: """#246""" pytester.makepyfile( """ import pytest values = [] @pytest.fixture(scope=%(scope)r, params=["1"]) def fix1(request): return request.param @pytest.fixture(scope=%(scope)r) def fix2(request, base): def cleanup_fix2(): assert not values, "base should not have been finalized" request.addfinalizer(cleanup_fix2) @pytest.fixture(scope=%(scope)r) def base(request, fix1): def cleanup_base(): values.append("fin_base") print("finalizing base") request.addfinalizer(cleanup_base) def test_begin(): pass def test_baz(base, fix2): pass def test_other(): pass """ # noqa: UP031 (python syntax issues) % {"scope": scope} ) reprec = pytester.inline_run("-lvs") reprec.assertoutcome(passed=3) def test_class_scope_parametrization_ordering(self, pytester: Pytester) -> None: """#396""" pytester.makepyfile( """ import pytest values = [] @pytest.fixture(params=["John", "Doe"], scope="class") def human(request): request.addfinalizer(lambda: values.append("fin %s" % request.param)) return request.param class TestGreetings(object): def test_hello(self, human): values.append("test_hello") class TestMetrics(object): def test_name(self, human): values.append("test_name") def test_population(self, human): values.append("test_population") """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=6) values = reprec.getcalls("pytest_runtest_call")[0].item.module.values assert values == [ "test_hello", "fin John", "test_hello", "fin Doe", "test_name", "test_population", "fin John", "test_name", "test_population", "fin Doe", ] def test_parametrize_setup_function(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module", params=[1, 2]) def arg(request): return request.param @pytest.fixture(scope="module", autouse=True) def mysetup(request, arg): request.addfinalizer(lambda: values.append("fin%s" % arg)) values.append("setup%s" % arg) values = [] def test_1(arg): values.append(arg) def test_2(arg): values.append(arg) def test_3(): import pprint pprint.pprint(values) if arg == 1: assert values == ["setup1", 1, 1, ] elif arg == 2: assert values == ["setup1", 1, 1, "fin1", "setup2", 2, 2, ] """ ) reprec = pytester.inline_run("-v") reprec.assertoutcome(passed=6) def test_fixture_marked_function_not_collected_as_test( self, pytester: Pytester ) -> None: pytester.makepyfile( """ import pytest @pytest.fixture def test_app(): return 1 def test_something(test_app): assert test_app == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_params_and_ids(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=[object(), object()], ids=['alpha', 'beta']) def fix(request): return request.param def test_foo(fix): assert 1 """ ) res = pytester.runpytest("-v") res.stdout.fnmatch_lines(["*test_foo*alpha*", "*test_foo*beta*"]) def test_params_and_ids_yieldfixture(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(params=[object(), object()], ids=['alpha', 'beta']) def fix(request): yield request.param def test_foo(fix): assert 1 """ ) res = pytester.runpytest("-v") res.stdout.fnmatch_lines(["*test_foo*alpha*", "*test_foo*beta*"]) def test_deterministic_fixture_collection( self, pytester: Pytester, monkeypatch ) -> None: """#920""" pytester.makepyfile( """ import pytest @pytest.fixture(scope="module", params=["A", "B", "C"]) def A(request): return request.param @pytest.fixture(scope="module", params=["DDDDDDDDD", "EEEEEEEEEEEE", "FFFFFFFFFFF", "banansda"]) def B(request, A): return request.param def test_foo(B): # Something funky is going on here. # Despite specified seeds, on what is collected, # sometimes we get unexpected passes. hashing B seems # to help? assert hash(B) or True """ ) monkeypatch.setenv("PYTHONHASHSEED", "1") out1 = pytester.runpytest_subprocess("-v") monkeypatch.setenv("PYTHONHASHSEED", "2") out2 = pytester.runpytest_subprocess("-v") output1 = [ line for line in out1.outlines if line.startswith("test_deterministic_fixture_collection.py::test_foo") ] output2 = [ line for line in out2.outlines if line.startswith("test_deterministic_fixture_collection.py::test_foo") ] assert len(output1) == 12 assert output1 == output2 class TestRequestScopeAccess: pytestmark = pytest.mark.parametrize( ("scope", "ok", "error"), [ ["session", "", "path class function module"], ["module", "module path", "cls function"], ["class", "module path cls", "function"], ["function", "module path cls function", ""], ], ) def test_setup(self, pytester: Pytester, scope, ok, error) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope=%r, autouse=True) def myscoped(request): for x in %r: assert hasattr(request, x) for x in %r: pytest.raises(AttributeError, lambda: getattr(request, x)) assert request.session assert request.config def test_func(): pass """ # noqa: UP031 (python syntax issues) % (scope, ok.split(), error.split()) ) reprec = pytester.inline_run("-l") reprec.assertoutcome(passed=1) def test_funcarg(self, pytester: Pytester, scope, ok, error) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope=%r) def arg(request): for x in %r: assert hasattr(request, x) for x in %r: pytest.raises(AttributeError, lambda: getattr(request, x)) assert request.session assert request.config def test_func(arg): pass """ # noqa: UP031 (python syntax issues) % (scope, ok.split(), error.split()) ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) class TestErrors: def test_subfactory_missing_funcarg(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture() def gen(qwe123): return 1 def test_something(gen): pass """ ) result = pytester.runpytest() assert result.ret != 0 result.stdout.fnmatch_lines( ["*def gen(qwe123):*", "*fixture*qwe123*not found*", "*1 error*"] ) def test_issue498_fixture_finalizer_failing(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture def fix1(request): def f(): raise KeyError request.addfinalizer(f) return object() values = [] def test_1(fix1): values.append(fix1) def test_2(fix1): values.append(fix1) def test_3(): assert values[0] != values[1] """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( """ *ERROR*teardown*test_1* *KeyError* *ERROR*teardown*test_2* *KeyError* *3 pass*2 errors* """ ) def test_setupfunc_missing_funcarg(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(autouse=True) def gen(qwe123): return 1 def test_something(): pass """ ) result = pytester.runpytest() assert result.ret != 0 result.stdout.fnmatch_lines( ["*def gen(qwe123):*", "*fixture*qwe123*not found*", "*1 error*"] ) def test_cached_exception_doesnt_get_longer(self, pytester: Pytester) -> None: """Regression test for #12204.""" pytester.makepyfile( """ import pytest @pytest.fixture(scope="session") def bad(): 1 / 0 def test_1(bad): pass def test_2(bad): pass def test_3(bad): pass """ ) result = pytester.runpytest_inprocess("--tb=native") assert result.ret == ExitCode.TESTS_FAILED failures = result.reprec.getfailures() # type: ignore[attr-defined] assert len(failures) == 3 lines1 = failures[1].longrepr.reprtraceback.reprentries[0].lines lines2 = failures[2].longrepr.reprtraceback.reprentries[0].lines assert len(lines1) == len(lines2) class TestShowFixtures: def test_funcarg_compat(self, pytester: Pytester) -> None: config = pytester.parseconfigure("--funcargs") assert config.option.showfixtures def test_show_help(self, pytester: Pytester) -> None: result = pytester.runpytest("--fixtures", "--help") assert not result.ret def test_show_fixtures(self, pytester: Pytester) -> None: result = pytester.runpytest("--fixtures") result.stdout.fnmatch_lines( [ "tmp_path_factory [[]session scope[]] -- .../_pytest/tmpdir.py:*", "*for the test session*", "tmp_path -- .../_pytest/tmpdir.py:*", "*temporary directory*", ] ) def test_show_fixtures_verbose(self, pytester: Pytester) -> None: result = pytester.runpytest("--fixtures", "-v") result.stdout.fnmatch_lines( [ "tmp_path_factory [[]session scope[]] -- .../_pytest/tmpdir.py:*", "*for the test session*", "tmp_path -- .../_pytest/tmpdir.py:*", "*temporary directory*", ] ) def test_show_fixtures_testmodule(self, pytester: Pytester) -> None: p = pytester.makepyfile( ''' import pytest @pytest.fixture def _arg0(): """ hidden """ @pytest.fixture def arg1(): """ hello world """ ''' ) result = pytester.runpytest("--fixtures", p) result.stdout.fnmatch_lines( """ *tmp_path -- * *fixtures defined from* *arg1 -- test_show_fixtures_testmodule.py:6* *hello world* """ ) result.stdout.no_fnmatch_line("*arg0*") @pytest.mark.parametrize("testmod", [True, False]) def test_show_fixtures_conftest(self, pytester: Pytester, testmod) -> None: pytester.makeconftest( ''' import pytest @pytest.fixture def arg1(): """ hello world """ ''' ) if testmod: pytester.makepyfile( """ def test_hello(): pass """ ) result = pytester.runpytest("--fixtures") result.stdout.fnmatch_lines( """ *tmp_path* *fixtures defined from*conftest* *arg1* *hello world* """ ) def test_show_fixtures_trimmed_doc(self, pytester: Pytester) -> None: p = pytester.makepyfile( textwrap.dedent( '''\ import pytest @pytest.fixture def arg1(): """ line1 line2 """ @pytest.fixture def arg2(): """ line1 line2 """ ''' ) ) result = pytester.runpytest("--fixtures", p) result.stdout.fnmatch_lines( textwrap.dedent( """\ * fixtures defined from test_show_fixtures_trimmed_doc * arg2 -- test_show_fixtures_trimmed_doc.py:10 line1 line2 arg1 -- test_show_fixtures_trimmed_doc.py:3 line1 line2 """ ) ) def test_show_fixtures_indented_doc(self, pytester: Pytester) -> None: p = pytester.makepyfile( textwrap.dedent( '''\ import pytest @pytest.fixture def fixture1(): """ line1 indented line """ ''' ) ) result = pytester.runpytest("--fixtures", p) result.stdout.fnmatch_lines( textwrap.dedent( """\ * fixtures defined from test_show_fixtures_indented_doc * fixture1 -- test_show_fixtures_indented_doc.py:3 line1 indented line """ ) ) def test_show_fixtures_indented_doc_first_line_unindented( self, pytester: Pytester ) -> None: p = pytester.makepyfile( textwrap.dedent( '''\ import pytest @pytest.fixture def fixture1(): """line1 line2 indented line """ ''' ) ) result = pytester.runpytest("--fixtures", p) result.stdout.fnmatch_lines( textwrap.dedent( """\ * fixtures defined from test_show_fixtures_indented_doc_first_line_unindented * fixture1 -- test_show_fixtures_indented_doc_first_line_unindented.py:3 line1 line2 indented line """ ) ) def test_show_fixtures_indented_in_class(self, pytester: Pytester) -> None: p = pytester.makepyfile( textwrap.dedent( '''\ import pytest class TestClass(object): @pytest.fixture def fixture1(self): """line1 line2 indented line """ ''' ) ) result = pytester.runpytest("--fixtures", p) result.stdout.fnmatch_lines( textwrap.dedent( """\ * fixtures defined from test_show_fixtures_indented_in_class * fixture1 -- test_show_fixtures_indented_in_class.py:4 line1 line2 indented line """ ) ) def test_show_fixtures_different_files(self, pytester: Pytester) -> None: """`--fixtures` only shows fixtures from first file (#833).""" pytester.makepyfile( test_a=''' import pytest @pytest.fixture def fix_a(): """Fixture A""" pass def test_a(fix_a): pass ''' ) pytester.makepyfile( test_b=''' import pytest @pytest.fixture def fix_b(): """Fixture B""" pass def test_b(fix_b): pass ''' ) result = pytester.runpytest("--fixtures") result.stdout.fnmatch_lines( """ * fixtures defined from test_a * fix_a -- test_a.py:4 Fixture A * fixtures defined from test_b * fix_b -- test_b.py:4 Fixture B """ ) def test_show_fixtures_with_same_name(self, pytester: Pytester) -> None: pytester.makeconftest( ''' import pytest @pytest.fixture def arg1(): """Hello World in conftest.py""" return "Hello World" ''' ) pytester.makepyfile( """ def test_foo(arg1): assert arg1 == "Hello World" """ ) pytester.makepyfile( ''' import pytest @pytest.fixture def arg1(): """Hi from test module""" return "Hi" def test_bar(arg1): assert arg1 == "Hi" ''' ) result = pytester.runpytest("--fixtures") result.stdout.fnmatch_lines( """ * fixtures defined from conftest * arg1 -- conftest.py:3 Hello World in conftest.py * fixtures defined from test_show_fixtures_with_same_name * arg1 -- test_show_fixtures_with_same_name.py:3 Hi from test module """ ) def test_fixture_disallow_twice(self): """Test that applying @pytest.fixture twice generates an error (#2334).""" with pytest.raises(ValueError): @pytest.fixture @pytest.fixture def foo(): raise NotImplementedError() class TestContextManagerFixtureFuncs: def test_simple(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture def arg1(): print("setup") yield 1 print("teardown") def test_1(arg1): print("test1", arg1) def test_2(arg1): print("test2", arg1) assert 0 """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines( """ *setup* *test1 1* *teardown* *setup* *test2 1* *teardown* """ ) def test_scoped(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module") def arg1(): print("setup") yield 1 print("teardown") def test_1(arg1): print("test1", arg1) def test_2(arg1): print("test2", arg1) """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines( """ *setup* *test1 1* *test2 1* *teardown* """ ) def test_setup_exception(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module") def arg1(): pytest.fail("setup") yield 1 def test_1(arg1): pass """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines( """ *pytest.fail*setup* *1 error* """ ) def test_teardown_exception(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module") def arg1(): yield 1 pytest.fail("teardown") def test_1(arg1): pass """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines( """ *pytest.fail*teardown* *1 passed*1 error* """ ) def test_yields_more_than_one(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope="module") def arg1(): yield 1 yield 2 def test_1(arg1): pass """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines( """ *fixture function* *test_yields*:2* """ ) def test_custom_name(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(name='meow') def arg1(): return 'mew' def test_1(meow): print(meow) """ ) result = pytester.runpytest("-s") result.stdout.fnmatch_lines(["*mew*"]) class TestParameterizedSubRequest: def test_call_from_fixture(self, pytester: Pytester) -> None: pytester.makepyfile( test_call_from_fixture=""" import pytest @pytest.fixture(params=[0, 1, 2]) def fix_with_param(request): return request.param @pytest.fixture def get_named_fixture(request): return request.getfixturevalue('fix_with_param') def test_foo(request, get_named_fixture): pass """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "The requested fixture has no parameter defined for test:", " test_call_from_fixture.py::test_foo", "Requested fixture 'fix_with_param' defined in:", "test_call_from_fixture.py:4", "Requested here:", "test_call_from_fixture.py:9", "*1 error in*", ] ) def test_call_from_test(self, pytester: Pytester) -> None: pytester.makepyfile( test_call_from_test=""" import pytest @pytest.fixture(params=[0, 1, 2]) def fix_with_param(request): return request.param def test_foo(request): request.getfixturevalue('fix_with_param') """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "The requested fixture has no parameter defined for test:", " test_call_from_test.py::test_foo", "Requested fixture 'fix_with_param' defined in:", "test_call_from_test.py:4", "Requested here:", "test_call_from_test.py:8", "*1 failed*", ] ) def test_external_fixture(self, pytester: Pytester) -> None: pytester.makeconftest( """ import pytest @pytest.fixture(params=[0, 1, 2]) def fix_with_param(request): return request.param """ ) pytester.makepyfile( test_external_fixture=""" def test_foo(request): request.getfixturevalue('fix_with_param') """ ) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "The requested fixture has no parameter defined for test:", " test_external_fixture.py::test_foo", "", "Requested fixture 'fix_with_param' defined in:", "conftest.py:4", "Requested here:", "test_external_fixture.py:2", "*1 failed*", ] ) def test_non_relative_path(self, pytester: Pytester) -> None: tests_dir = pytester.mkdir("tests") fixdir = pytester.mkdir("fixtures") fixfile = fixdir.joinpath("fix.py") fixfile.write_text( textwrap.dedent( """\ import pytest @pytest.fixture(params=[0, 1, 2]) def fix_with_param(request): return request.param """ ), encoding="utf-8", ) testfile = tests_dir.joinpath("test_foos.py") testfile.write_text( textwrap.dedent( """\ from fix import fix_with_param def test_foo(request): request.getfixturevalue('fix_with_param') """ ), encoding="utf-8", ) os.chdir(tests_dir) pytester.syspathinsert(fixdir) result = pytester.runpytest() result.stdout.fnmatch_lines( [ "The requested fixture has no parameter defined for test:", " test_foos.py::test_foo", "", "Requested fixture 'fix_with_param' defined in:", f"{fixfile}:4", "Requested here:", "test_foos.py:4", "*1 failed*", ] ) # With non-overlapping rootdir, passing tests_dir. rootdir = pytester.mkdir("rootdir") os.chdir(rootdir) result = pytester.runpytest("--rootdir", rootdir, tests_dir) result.stdout.fnmatch_lines( [ "The requested fixture has no parameter defined for test:", " test_foos.py::test_foo", "", "Requested fixture 'fix_with_param' defined in:", f"{fixfile}:4", "Requested here:", f"{testfile}:4", "*1 failed*", ] ) def test_pytest_fixture_setup_and_post_finalizer_hook(pytester: Pytester) -> None: pytester.makeconftest( """ def pytest_fixture_setup(fixturedef, request): print('ROOT setup hook called for {0} from {1}'.format(fixturedef.argname, request.node.name)) def pytest_fixture_post_finalizer(fixturedef, request): print('ROOT finalizer hook called for {0} from {1}'.format(fixturedef.argname, request.node.name)) """ ) pytester.makepyfile( **{ "tests/conftest.py": """ def pytest_fixture_setup(fixturedef, request): print('TESTS setup hook called for {0} from {1}'.format(fixturedef.argname, request.node.name)) def pytest_fixture_post_finalizer(fixturedef, request): print('TESTS finalizer hook called for {0} from {1}'.format(fixturedef.argname, request.node.name)) """, "tests/test_hooks.py": """ import pytest @pytest.fixture() def my_fixture(): return 'some' def test_func(my_fixture): print('TEST test_func') assert my_fixture == 'some' """, } ) result = pytester.runpytest("-s") assert result.ret == 0 result.stdout.fnmatch_lines( [ "*TESTS setup hook called for my_fixture from test_func*", "*ROOT setup hook called for my_fixture from test_func*", "*TEST test_func*", "*TESTS finalizer hook called for my_fixture from test_func*", "*ROOT finalizer hook called for my_fixture from test_func*", ] ) class TestScopeOrdering: """Class of tests that ensure fixtures are ordered based on their scopes (#2405)""" @pytest.mark.parametrize("variant", ["mark", "autouse"]) def test_func_closure_module_auto( self, pytester: Pytester, variant, monkeypatch ) -> None: """Semantically identical to the example posted in #2405 when ``use_mark=True``""" monkeypatch.setenv("FIXTURE_ACTIVATION_VARIANT", variant) pytester.makepyfile( """ import warnings import os import pytest VAR = 'FIXTURE_ACTIVATION_VARIANT' VALID_VARS = ('autouse', 'mark') VARIANT = os.environ.get(VAR) if VARIANT is None or VARIANT not in VALID_VARS: warnings.warn("{!r} is not in {}, assuming autouse".format(VARIANT, VALID_VARS) ) variant = 'mark' @pytest.fixture(scope='module', autouse=VARIANT == 'autouse') def m1(): pass if VARIANT=='mark': pytestmark = pytest.mark.usefixtures('m1') @pytest.fixture(scope='function', autouse=True) def f1(): pass def test_func(m1): pass """ ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) assert request.fixturenames == "m1 f1".split() def test_func_closure_with_native_fixtures( self, pytester: Pytester, monkeypatch: MonkeyPatch ) -> None: """Sanity check that verifies the order returned by the closures and the actual fixture execution order: The execution order may differ because of fixture inter-dependencies. """ monkeypatch.setattr(pytest, "FIXTURE_ORDER", [], raising=False) pytester.makepyfile( """ import pytest FIXTURE_ORDER = pytest.FIXTURE_ORDER @pytest.fixture(scope="session") def s1(): FIXTURE_ORDER.append('s1') @pytest.fixture(scope="package") def p1(): FIXTURE_ORDER.append('p1') @pytest.fixture(scope="module") def m1(): FIXTURE_ORDER.append('m1') @pytest.fixture(scope='session') def my_tmp_path_factory(): FIXTURE_ORDER.append('my_tmp_path_factory') @pytest.fixture def my_tmp_path(my_tmp_path_factory): FIXTURE_ORDER.append('my_tmp_path') @pytest.fixture def f1(my_tmp_path): FIXTURE_ORDER.append('f1') @pytest.fixture def f2(): FIXTURE_ORDER.append('f2') def test_foo(f1, p1, m1, f2, s1): pass """ ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) # order of fixtures based on their scope and position in the parameter list assert ( request.fixturenames == "s1 my_tmp_path_factory p1 m1 f1 f2 my_tmp_path".split() ) pytester.runpytest() # actual fixture execution differs: dependent fixtures must be created first ("my_tmp_path") FIXTURE_ORDER = pytest.FIXTURE_ORDER # type: ignore[attr-defined] assert FIXTURE_ORDER == "s1 my_tmp_path_factory p1 m1 my_tmp_path f1 f2".split() def test_func_closure_module(self, pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(scope='module') def m1(): pass @pytest.fixture(scope='function') def f1(): pass def test_func(f1, m1): pass """ ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) assert request.fixturenames == "m1 f1".split() def test_func_closure_scopes_reordered(self, pytester: Pytester) -> None: """Test ensures that fixtures are ordered by scope regardless of the order of the parameters, although fixtures of same scope keep the declared order """ pytester.makepyfile( """ import pytest @pytest.fixture(scope='session') def s1(): pass @pytest.fixture(scope='module') def m1(): pass @pytest.fixture(scope='function') def f1(): pass @pytest.fixture(scope='function') def f2(): pass class Test: @pytest.fixture(scope='class') def c1(cls): pass def test_func(self, f2, f1, c1, m1, s1): pass """ ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) assert request.fixturenames == "s1 m1 c1 f2 f1".split() def test_func_closure_same_scope_closer_root_first( self, pytester: Pytester ) -> None: """Auto-use fixtures of same scope are ordered by closer-to-root first""" pytester.makeconftest( """ import pytest @pytest.fixture(scope='module', autouse=True) def m_conf(): pass """ ) pytester.makepyfile( **{ "sub/conftest.py": """ import pytest @pytest.fixture(scope='package', autouse=True) def p_sub(): pass @pytest.fixture(scope='module', autouse=True) def m_sub(): pass """, "sub/__init__.py": "", "sub/test_func.py": """ import pytest @pytest.fixture(scope='module', autouse=True) def m_test(): pass @pytest.fixture(scope='function') def f1(): pass def test_func(m_test, f1): pass """, } ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) assert request.fixturenames == "p_sub m_conf m_sub m_test f1".split() def test_func_closure_all_scopes_complex(self, pytester: Pytester) -> None: """Complex test involving all scopes and mixing autouse with normal fixtures""" pytester.makeconftest( """ import pytest @pytest.fixture(scope='session') def s1(): pass @pytest.fixture(scope='package', autouse=True) def p1(): pass """ ) pytester.makepyfile(**{"__init__.py": ""}) pytester.makepyfile( """ import pytest @pytest.fixture(scope='module', autouse=True) def m1(): pass @pytest.fixture(scope='module') def m2(s1): pass @pytest.fixture(scope='function') def f1(): pass @pytest.fixture(scope='function') def f2(): pass class Test: @pytest.fixture(scope='class', autouse=True) def c1(self): pass def test_func(self, f2, f1, m2): pass """ ) items, _ = pytester.inline_genitems() assert isinstance(items[0], Function) request = TopRequest(items[0], _ispytest=True) assert request.fixturenames == "s1 p1 m1 m2 c1 f2 f1".split() def test_multiple_packages(self, pytester: Pytester) -> None: """Complex test involving multiple package fixtures. Make sure teardowns are executed in order. . └── root ├── __init__.py ├── sub1 │ ├── __init__.py │ ├── conftest.py │ └── test_1.py └── sub2 ├── __init__.py ├── conftest.py └── test_2.py """ root = pytester.mkdir("root") root.joinpath("__init__.py").write_text("values = []", encoding="utf-8") sub1 = root.joinpath("sub1") sub1.mkdir() sub1.joinpath("__init__.py").touch() sub1.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest from .. import values @pytest.fixture(scope="package") def fix(): values.append("pre-sub1") yield values assert values.pop() == "pre-sub1" """ ), encoding="utf-8", ) sub1.joinpath("test_1.py").write_text( textwrap.dedent( """\ from .. import values def test_1(fix): assert values == ["pre-sub1"] """ ), encoding="utf-8", ) sub2 = root.joinpath("sub2") sub2.mkdir() sub2.joinpath("__init__.py").touch() sub2.joinpath("conftest.py").write_text( textwrap.dedent( """\ import pytest from .. import values @pytest.fixture(scope="package") def fix(): values.append("pre-sub2") yield values assert values.pop() == "pre-sub2" """ ), encoding="utf-8", ) sub2.joinpath("test_2.py").write_text( textwrap.dedent( """\ from .. import values def test_2(fix): assert values == ["pre-sub2"] """ ), encoding="utf-8", ) reprec = pytester.inline_run() reprec.assertoutcome(passed=2) def test_class_fixture_self_instance(self, pytester: Pytester) -> None: """Check that plugin classes which implement fixtures receive the plugin instance as self (see #2270). """ pytester.makeconftest( """ import pytest def pytest_configure(config): config.pluginmanager.register(MyPlugin()) class MyPlugin(): def __init__(self): self.arg = 1 @pytest.fixture(scope='function') def myfix(self): assert isinstance(self, MyPlugin) return self.arg """ ) pytester.makepyfile( """ class TestClass(object): def test_1(self, myfix): assert myfix == 1 """ ) reprec = pytester.inline_run() reprec.assertoutcome(passed=1) def test_call_fixture_function_error(): """Check if an error is raised if a fixture function is called directly (#4545)""" @pytest.fixture def fix(): raise NotImplementedError() with pytest.raises(pytest.fail.Exception): assert fix() == 1 def test_fixture_double_decorator(pytester: Pytester) -> None: """Check if an error is raised when using @pytest.fixture twice.""" pytester.makepyfile( """ import pytest @pytest.fixture @pytest.fixture def fixt(): pass """ ) result = pytester.runpytest() result.assert_outcomes(errors=1) result.stdout.fnmatch_lines( [ "E * ValueError: @pytest.fixture is being applied more than once to the same function 'fixt'" ] ) def test_fixture_param_shadowing(pytester: Pytester) -> None: """Parametrized arguments would be shadowed if a fixture with the same name also exists (#5036)""" pytester.makepyfile( """ import pytest @pytest.fixture(params=['a', 'b']) def argroot(request): return request.param @pytest.fixture def arg(argroot): return argroot # This should only be parametrized directly @pytest.mark.parametrize("arg", [1]) def test_direct(arg): assert arg == 1 # This should be parametrized based on the fixtures def test_normal_fixture(arg): assert isinstance(arg, str) # Indirect should still work: @pytest.fixture def arg2(request): return 2*request.param @pytest.mark.parametrize("arg2", [1], indirect=True) def test_indirect(arg2): assert arg2 == 2 """ ) # Only one test should have run result = pytester.runpytest("-v") result.assert_outcomes(passed=4) result.stdout.fnmatch_lines(["*::test_direct[[]1[]]*"]) result.stdout.fnmatch_lines(["*::test_normal_fixture[[]a[]]*"]) result.stdout.fnmatch_lines(["*::test_normal_fixture[[]b[]]*"]) result.stdout.fnmatch_lines(["*::test_indirect[[]1[]]*"]) def test_fixture_named_request(pytester: Pytester) -> None: pytester.copy_example("fixtures/test_fixture_named_request.py") result = pytester.runpytest() result.stdout.fnmatch_lines( [ "*'request' is a reserved word for fixtures, use another name:", " *test_fixture_named_request.py:6", ] ) def test_indirect_fixture_does_not_break_scope(pytester: Pytester) -> None: """Ensure that fixture scope is respected when using indirect fixtures (#570)""" pytester.makepyfile( """ import pytest instantiated = [] @pytest.fixture(scope="session") def fixture_1(request): instantiated.append(("fixture_1", request.param)) @pytest.fixture(scope="session") def fixture_2(request): instantiated.append(("fixture_2", request.param)) scenarios = [ ("A", "a1"), ("A", "a2"), ("B", "b1"), ("B", "b2"), ("C", "c1"), ("C", "c2"), ] @pytest.mark.parametrize( "fixture_1,fixture_2", scenarios, indirect=["fixture_1", "fixture_2"] ) def test_create_fixtures(fixture_1, fixture_2): pass def test_check_fixture_instantiations(): assert instantiated == [ ('fixture_1', 'A'), ('fixture_2', 'a1'), ('fixture_2', 'a2'), ('fixture_1', 'B'), ('fixture_2', 'b1'), ('fixture_2', 'b2'), ('fixture_1', 'C'), ('fixture_2', 'c1'), ('fixture_2', 'c2'), ] """ ) result = pytester.runpytest() result.assert_outcomes(passed=7) def test_fixture_parametrization_nparray(pytester: Pytester) -> None: pytest.importorskip("numpy") pytester.makepyfile( """ from numpy import linspace from pytest import fixture @fixture(params=linspace(1, 10, 10)) def value(request): return request.param def test_bug(value): assert value == value """ ) result = pytester.runpytest() result.assert_outcomes(passed=10) def test_fixture_arg_ordering(pytester: Pytester) -> None: """ This test describes how fixtures in the same scope but without explicit dependencies between them are created. While users should make dependencies explicit, often they rely on this order, so this test exists to catch regressions in this regard. See #6540 and #6492. """ p1 = pytester.makepyfile( """ import pytest suffixes = [] @pytest.fixture def fix_1(): suffixes.append("fix_1") @pytest.fixture def fix_2(): suffixes.append("fix_2") @pytest.fixture def fix_3(): suffixes.append("fix_3") @pytest.fixture def fix_4(): suffixes.append("fix_4") @pytest.fixture def fix_5(): suffixes.append("fix_5") @pytest.fixture def fix_combined(fix_1, fix_2, fix_3, fix_4, fix_5): pass def test_suffix(fix_combined): assert suffixes == ["fix_1", "fix_2", "fix_3", "fix_4", "fix_5"] """ ) result = pytester.runpytest("-vv", str(p1)) assert result.ret == 0 def test_yield_fixture_with_no_value(pytester: Pytester) -> None: pytester.makepyfile( """ import pytest @pytest.fixture(name='custom') def empty_yield(): if False: yield def test_fixt(custom): pass """ ) expected = "E ValueError: custom did not yield a value" result = pytester.runpytest() result.assert_outcomes(errors=1) result.stdout.fnmatch_lines([expected]) assert result.ret == ExitCode.TESTS_FAILED def test_deduplicate_names() -> None: items = deduplicate_names("abacd") assert items == ("a", "b", "c", "d") items = deduplicate_names((*items, "g", "f", "g", "e", "b")) assert items == ("a", "b", "c", "d", "g", "f", "e") def test_staticmethod_classmethod_fixture_instance(pytester: Pytester) -> None: """Ensure that static and class methods get and have access to a fresh instance. This also ensures `setup_method` works well with static and class methods. Regression test for #12065. """ pytester.makepyfile( """ import pytest class Test: ran_setup_method = False ran_fixture = False def setup_method(self): assert not self.ran_setup_method self.ran_setup_method = True @pytest.fixture(autouse=True) def fixture(self): assert not self.ran_fixture self.ran_fixture = True def test_method(self): assert self.ran_setup_method assert self.ran_fixture @staticmethod def test_1(request): assert request.instance.ran_setup_method assert request.instance.ran_fixture @classmethod def test_2(cls, request): assert request.instance.ran_setup_method assert request.instance.ran_fixture """ ) result = pytester.runpytest() assert result.ret == ExitCode.OK result.assert_outcomes(passed=3) def test_scoped_fixture_caching(pytester: Pytester) -> None: """Make sure setup and finalization is only run once when using scoped fixture multiple times.""" pytester.makepyfile( """ from __future__ import annotations from typing import Generator import pytest executed: list[str] = [] @pytest.fixture(scope="class") def fixture_1() -> Generator[None, None, None]: executed.append("fix setup") yield executed.append("fix teardown") class TestFixtureCaching: def test_1(self, fixture_1: None) -> None: assert executed == ["fix setup"] def test_2(self, fixture_1: None) -> None: assert executed == ["fix setup"] def test_expected_setup_and_teardown() -> None: assert executed == ["fix setup", "fix teardown"] """ ) result = pytester.runpytest() assert result.ret == 0 def test_scoped_fixture_caching_exception(pytester: Pytester) -> None: """Make sure setup & finalization is only run once for scoped fixture, with a cached exception.""" pytester.makepyfile( """ from __future__ import annotations import pytest executed_crash: list[str] = [] @pytest.fixture(scope="class") def fixture_crash(request: pytest.FixtureRequest) -> None: executed_crash.append("fix_crash setup") def my_finalizer() -> None: executed_crash.append("fix_crash teardown") request.addfinalizer(my_finalizer) raise Exception("foo") class TestFixtureCachingException: @pytest.mark.xfail def test_crash_1(self, fixture_crash: None) -> None: ... @pytest.mark.xfail def test_crash_2(self, fixture_crash: None) -> None: ... def test_crash_expected_setup_and_teardown() -> None: assert executed_crash == ["fix_crash setup", "fix_crash teardown"] """ ) result = pytester.runpytest() assert result.ret == 0 def test_scoped_fixture_teardown_order(pytester: Pytester) -> None: """ Make sure teardowns happen in reverse order of setup with scoped fixtures, when a later test only depends on a subset of scoped fixtures. Regression test for https://github.com/pytest-dev/pytest/issues/1489 """ pytester.makepyfile( """ from typing import Generator import pytest last_executed = "" @pytest.fixture(scope="module") def fixture_1() -> Generator[None, None, None]: global last_executed assert last_executed == "" last_executed = "fixture_1_setup" yield assert last_executed == "fixture_2_teardown" last_executed = "fixture_1_teardown" @pytest.fixture(scope="module") def fixture_2() -> Generator[None, None, None]: global last_executed assert last_executed == "fixture_1_setup" last_executed = "fixture_2_setup" yield assert last_executed == "run_test" last_executed = "fixture_2_teardown" def test_fixture_teardown_order(fixture_1: None, fixture_2: None) -> None: global last_executed assert last_executed == "fixture_2_setup" last_executed = "run_test" def test_2(fixture_1: None) -> None: # This would previously queue an additional teardown of fixture_1, # despite fixture_1's value being cached, which caused fixture_1 to be # torn down before fixture_2 - violating the rule that teardowns should # happen in reverse order of setup. pass """ ) result = pytester.runpytest() assert result.ret == 0 def test_subfixture_teardown_order(pytester: Pytester) -> None: """ Make sure fixtures don't re-register their finalization in parent fixtures multiple times, causing ordering failure in their teardowns. Regression test for #12135 """ pytester.makepyfile( """ import pytest execution_order = [] @pytest.fixture(scope="class") def fixture_1(): ... @pytest.fixture(scope="class") def fixture_2(fixture_1): execution_order.append("setup 2") yield execution_order.append("teardown 2") @pytest.fixture(scope="class") def fixture_3(fixture_1): execution_order.append("setup 3") yield execution_order.append("teardown 3") class TestFoo: def test_initialize_fixtures(self, fixture_2, fixture_3): ... # This would previously reschedule fixture_2's finalizer in the parent fixture, # causing it to be torn down before fixture 3. def test_reschedule_fixture_2(self, fixture_2): ... # Force finalization directly on fixture_1 # Otherwise the cleanup would sequence 3&2 before 1 as normal. @pytest.mark.parametrize("fixture_1", [None], indirect=["fixture_1"]) def test_finalize_fixture_1(self, fixture_1): ... def test_result(): assert execution_order == ["setup 2", "setup 3", "teardown 3", "teardown 2"] """ ) result = pytester.runpytest() assert result.ret == 0