unittest: make `obj` work more like `Function`/`Class`
Previously, the `obj` of a `TestCaseFunction` (the unittest plugin item type) was the unbound method. This is unlike regular `Class` where the `obj` is a bound method to a fresh instance. This difference necessitated several special cases in in places outside of the unittest plugin, such as `FixtureDef` and `FixtureRequest`, and made things a bit harder to understand. Instead, match how the python plugin does it, including collecting fixtures from a fresh instance. The downside is that now this instance for fixture-collection is kept around in memory, but it's the same as `Class` so nothing new. Users should only initialize stuff in `setUp`/`setUpClass` and similar methods, and not in `__init__` which is generally off-limits in `TestCase` subclasses. I am not sure why there was a difference in the first place, though I will say the previous unittest approach is probably the preferable one, but first let's get consistency.
This commit is contained in:
parent
03e54712dd
commit
1a5e0eb71d
|
@ -86,7 +86,6 @@ def getfuncargnames(
|
||||||
function: Callable[..., object],
|
function: Callable[..., object],
|
||||||
*,
|
*,
|
||||||
name: str = "",
|
name: str = "",
|
||||||
is_method: bool = False,
|
|
||||||
cls: type | None = None,
|
cls: type | None = None,
|
||||||
) -> tuple[str, ...]:
|
) -> tuple[str, ...]:
|
||||||
"""Return the names of a function's mandatory arguments.
|
"""Return the names of a function's mandatory arguments.
|
||||||
|
@ -97,9 +96,8 @@ def getfuncargnames(
|
||||||
* Aren't bound with functools.partial.
|
* Aren't bound with functools.partial.
|
||||||
* Aren't replaced with mocks.
|
* Aren't replaced with mocks.
|
||||||
|
|
||||||
The is_method and cls arguments indicate that the function should
|
The cls arguments indicate that the function should be treated as a bound
|
||||||
be treated as a bound method even though it's not unless, only in
|
method even though it's not unless the function is a static method.
|
||||||
the case of cls, the function is a static method.
|
|
||||||
|
|
||||||
The name parameter should be the original name in which the function was collected.
|
The name parameter should be the original name in which the function was collected.
|
||||||
"""
|
"""
|
||||||
|
@ -137,7 +135,7 @@ def getfuncargnames(
|
||||||
# If this function should be treated as a bound method even though
|
# If this function should be treated as a bound method even though
|
||||||
# it's passed as an unbound method or function, remove the first
|
# it's passed as an unbound method or function, remove the first
|
||||||
# parameter name.
|
# parameter name.
|
||||||
if is_method or (
|
if (
|
||||||
# Not using `getattr` because we don't want to resolve the staticmethod.
|
# Not using `getattr` because we don't want to resolve the staticmethod.
|
||||||
# Not using `cls.__dict__` because we want to check the entire MRO.
|
# Not using `cls.__dict__` because we want to check the entire MRO.
|
||||||
cls
|
cls
|
||||||
|
|
|
@ -462,12 +462,8 @@ class FixtureRequest(abc.ABC):
|
||||||
@property
|
@property
|
||||||
def instance(self):
|
def instance(self):
|
||||||
"""Instance (can be None) on which test function was collected."""
|
"""Instance (can be None) on which test function was collected."""
|
||||||
# unittest support hack, see _pytest.unittest.TestCaseFunction.
|
function = getattr(self, "function", None)
|
||||||
try:
|
return getattr(function, "__self__", None)
|
||||||
return self._pyfuncitem._testcase # type: ignore[attr-defined]
|
|
||||||
except AttributeError:
|
|
||||||
function = getattr(self, "function", None)
|
|
||||||
return getattr(function, "__self__", None)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def module(self):
|
def module(self):
|
||||||
|
@ -965,7 +961,6 @@ class FixtureDef(Generic[FixtureValue]):
|
||||||
func: "_FixtureFunc[FixtureValue]",
|
func: "_FixtureFunc[FixtureValue]",
|
||||||
scope: Union[Scope, _ScopeName, Callable[[str, Config], _ScopeName], None],
|
scope: Union[Scope, _ScopeName, Callable[[str, Config], _ScopeName], None],
|
||||||
params: Optional[Sequence[object]],
|
params: Optional[Sequence[object]],
|
||||||
unittest: bool = False,
|
|
||||||
ids: Optional[
|
ids: Optional[
|
||||||
Union[Tuple[Optional[object], ...], Callable[[Any], Optional[object]]]
|
Union[Tuple[Optional[object], ...], Callable[[Any], Optional[object]]]
|
||||||
] = None,
|
] = None,
|
||||||
|
@ -1011,9 +1006,7 @@ class FixtureDef(Generic[FixtureValue]):
|
||||||
# a parameter value.
|
# a parameter value.
|
||||||
self.ids: Final = ids
|
self.ids: Final = ids
|
||||||
# The names requested by the fixtures.
|
# The names requested by the fixtures.
|
||||||
self.argnames: Final = getfuncargnames(func, name=argname, is_method=unittest)
|
self.argnames: Final = getfuncargnames(func, name=argname)
|
||||||
# Whether the fixture was collected from a unittest TestCase class.
|
|
||||||
self.unittest: Final = unittest
|
|
||||||
# If the fixture was executed, the current value of the fixture.
|
# If the fixture was executed, the current value of the fixture.
|
||||||
# Can change if the fixture is executed with different parameters.
|
# Can change if the fixture is executed with different parameters.
|
||||||
self.cached_result: Optional[_FixtureCachedResult[FixtureValue]] = None
|
self.cached_result: Optional[_FixtureCachedResult[FixtureValue]] = None
|
||||||
|
@ -1092,25 +1085,20 @@ def resolve_fixture_function(
|
||||||
"""Get the actual callable that can be called to obtain the fixture
|
"""Get the actual callable that can be called to obtain the fixture
|
||||||
value, dealing with unittest-specific instances and bound methods."""
|
value, dealing with unittest-specific instances and bound methods."""
|
||||||
fixturefunc = fixturedef.func
|
fixturefunc = fixturedef.func
|
||||||
if fixturedef.unittest:
|
# The fixture function needs to be bound to the actual
|
||||||
if request.instance is not None:
|
# request.instance so that code working with "fixturedef" behaves
|
||||||
# Bind the unbound method to the TestCase instance.
|
# as expected.
|
||||||
fixturefunc = fixturedef.func.__get__(request.instance) # type: ignore[union-attr]
|
if request.instance is not None:
|
||||||
else:
|
# Handle the case where fixture is defined not in a test class, but some other class
|
||||||
# The fixture function needs to be bound to the actual
|
# (for example a plugin class with a fixture), see #2270.
|
||||||
# request.instance so that code working with "fixturedef" behaves
|
if hasattr(fixturefunc, "__self__") and not isinstance(
|
||||||
# as expected.
|
request.instance,
|
||||||
if request.instance is not None:
|
fixturefunc.__self__.__class__, # type: ignore[union-attr]
|
||||||
# Handle the case where fixture is defined not in a test class, but some other class
|
):
|
||||||
# (for example a plugin class with a fixture), see #2270.
|
return fixturefunc
|
||||||
if hasattr(fixturefunc, "__self__") and not isinstance(
|
fixturefunc = getimfunc(fixturedef.func)
|
||||||
request.instance,
|
if fixturefunc != fixturedef.func:
|
||||||
fixturefunc.__self__.__class__, # type: ignore[union-attr]
|
fixturefunc = fixturefunc.__get__(request.instance) # type: ignore[union-attr]
|
||||||
):
|
|
||||||
return fixturefunc
|
|
||||||
fixturefunc = getimfunc(fixturedef.func)
|
|
||||||
if fixturefunc != fixturedef.func:
|
|
||||||
fixturefunc = fixturefunc.__get__(request.instance) # type: ignore[union-attr]
|
|
||||||
return fixturefunc
|
return fixturefunc
|
||||||
|
|
||||||
|
|
||||||
|
@ -1614,7 +1602,6 @@ class FixtureManager:
|
||||||
Union[Tuple[Optional[object], ...], Callable[[Any], Optional[object]]]
|
Union[Tuple[Optional[object], ...], Callable[[Any], Optional[object]]]
|
||||||
] = None,
|
] = None,
|
||||||
autouse: bool = False,
|
autouse: bool = False,
|
||||||
unittest: bool = False,
|
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Register a fixture
|
"""Register a fixture
|
||||||
|
|
||||||
|
@ -1635,8 +1622,6 @@ class FixtureManager:
|
||||||
The fixture's IDs.
|
The fixture's IDs.
|
||||||
:param autouse:
|
:param autouse:
|
||||||
Whether this is an autouse fixture.
|
Whether this is an autouse fixture.
|
||||||
:param unittest:
|
|
||||||
Set this if this is a unittest fixture.
|
|
||||||
"""
|
"""
|
||||||
fixture_def = FixtureDef(
|
fixture_def = FixtureDef(
|
||||||
config=self.config,
|
config=self.config,
|
||||||
|
@ -1645,7 +1630,6 @@ class FixtureManager:
|
||||||
func=func,
|
func=func,
|
||||||
scope=scope,
|
scope=scope,
|
||||||
params=params,
|
params=params,
|
||||||
unittest=unittest,
|
|
||||||
ids=ids,
|
ids=ids,
|
||||||
_ispytest=True,
|
_ispytest=True,
|
||||||
)
|
)
|
||||||
|
@ -1667,8 +1651,6 @@ class FixtureManager:
|
||||||
def parsefactories(
|
def parsefactories(
|
||||||
self,
|
self,
|
||||||
node_or_obj: nodes.Node,
|
node_or_obj: nodes.Node,
|
||||||
*,
|
|
||||||
unittest: bool = ...,
|
|
||||||
) -> None:
|
) -> None:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
@ -1677,8 +1659,6 @@ class FixtureManager:
|
||||||
self,
|
self,
|
||||||
node_or_obj: object,
|
node_or_obj: object,
|
||||||
nodeid: Optional[str],
|
nodeid: Optional[str],
|
||||||
*,
|
|
||||||
unittest: bool = ...,
|
|
||||||
) -> None:
|
) -> None:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
@ -1686,8 +1666,6 @@ class FixtureManager:
|
||||||
self,
|
self,
|
||||||
node_or_obj: Union[nodes.Node, object],
|
node_or_obj: Union[nodes.Node, object],
|
||||||
nodeid: Union[str, NotSetType, None] = NOTSET,
|
nodeid: Union[str, NotSetType, None] = NOTSET,
|
||||||
*,
|
|
||||||
unittest: bool = False,
|
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Collect fixtures from a collection node or object.
|
"""Collect fixtures from a collection node or object.
|
||||||
|
|
||||||
|
@ -1739,7 +1717,6 @@ class FixtureManager:
|
||||||
func=func,
|
func=func,
|
||||||
scope=marker.scope,
|
scope=marker.scope,
|
||||||
params=marker.params,
|
params=marker.params,
|
||||||
unittest=unittest,
|
|
||||||
ids=marker.ids,
|
ids=marker.ids,
|
||||||
autouse=marker.autouse,
|
autouse=marker.autouse,
|
||||||
)
|
)
|
||||||
|
|
|
@ -1314,7 +1314,6 @@ class Metafunc:
|
||||||
func=get_direct_param_fixture_func,
|
func=get_direct_param_fixture_func,
|
||||||
scope=scope_,
|
scope=scope_,
|
||||||
params=None,
|
params=None,
|
||||||
unittest=False,
|
|
||||||
ids=None,
|
ids=None,
|
||||||
_ispytest=True,
|
_ispytest=True,
|
||||||
)
|
)
|
||||||
|
|
|
@ -15,7 +15,6 @@ from typing import TYPE_CHECKING
|
||||||
from typing import Union
|
from typing import Union
|
||||||
|
|
||||||
import _pytest._code
|
import _pytest._code
|
||||||
from _pytest.compat import getimfunc
|
|
||||||
from _pytest.compat import is_async_function
|
from _pytest.compat import is_async_function
|
||||||
from _pytest.config import hookimpl
|
from _pytest.config import hookimpl
|
||||||
from _pytest.fixtures import FixtureRequest
|
from _pytest.fixtures import FixtureRequest
|
||||||
|
@ -63,6 +62,14 @@ class UnitTestCase(Class):
|
||||||
# to declare that our children do not support funcargs.
|
# to declare that our children do not support funcargs.
|
||||||
nofuncargs = True
|
nofuncargs = True
|
||||||
|
|
||||||
|
def newinstance(self):
|
||||||
|
# TestCase __init__ takes the method (test) name. The TestCase
|
||||||
|
# constructor treats the name "runTest" as a special no-op, so it can be
|
||||||
|
# used when a dummy instance is needed. While unittest.TestCase has a
|
||||||
|
# default, some subclasses omit the default (#9610), so always supply
|
||||||
|
# it.
|
||||||
|
return self.obj("runTest")
|
||||||
|
|
||||||
def collect(self) -> Iterable[Union[Item, Collector]]:
|
def collect(self) -> Iterable[Union[Item, Collector]]:
|
||||||
from unittest import TestLoader
|
from unittest import TestLoader
|
||||||
|
|
||||||
|
@ -76,15 +83,15 @@ class UnitTestCase(Class):
|
||||||
self._register_unittest_setup_class_fixture(cls)
|
self._register_unittest_setup_class_fixture(cls)
|
||||||
self._register_setup_class_fixture()
|
self._register_setup_class_fixture()
|
||||||
|
|
||||||
self.session._fixturemanager.parsefactories(self, unittest=True)
|
self.session._fixturemanager.parsefactories(self.newinstance(), self.nodeid)
|
||||||
|
|
||||||
loader = TestLoader()
|
loader = TestLoader()
|
||||||
foundsomething = False
|
foundsomething = False
|
||||||
for name in loader.getTestCaseNames(self.obj):
|
for name in loader.getTestCaseNames(self.obj):
|
||||||
x = getattr(self.obj, name)
|
x = getattr(self.obj, name)
|
||||||
if not getattr(x, "__test__", True):
|
if not getattr(x, "__test__", True):
|
||||||
continue
|
continue
|
||||||
funcobj = getimfunc(x)
|
yield TestCaseFunction.from_parent(self, name=name)
|
||||||
yield TestCaseFunction.from_parent(self, name=name, callobj=funcobj)
|
|
||||||
foundsomething = True
|
foundsomething = True
|
||||||
|
|
||||||
if not foundsomething:
|
if not foundsomething:
|
||||||
|
@ -169,23 +176,21 @@ class UnitTestCase(Class):
|
||||||
class TestCaseFunction(Function):
|
class TestCaseFunction(Function):
|
||||||
nofuncargs = True
|
nofuncargs = True
|
||||||
_excinfo: Optional[List[_pytest._code.ExceptionInfo[BaseException]]] = None
|
_excinfo: Optional[List[_pytest._code.ExceptionInfo[BaseException]]] = None
|
||||||
_testcase: Optional["unittest.TestCase"] = None
|
|
||||||
|
|
||||||
def _getobj(self):
|
def _getobj(self):
|
||||||
assert self.parent is not None
|
assert isinstance(self.parent, UnitTestCase)
|
||||||
# Unlike a regular Function in a Class, where `item.obj` returns
|
testcase = self.parent.obj(self.name)
|
||||||
# a *bound* method (attached to an instance), TestCaseFunction's
|
return getattr(testcase, self.name)
|
||||||
# `obj` returns an *unbound* method (not attached to an instance).
|
|
||||||
# This inconsistency is probably not desirable, but needs some
|
# Backward compat for pytest-django; can be removed after pytest-django
|
||||||
# consideration before changing.
|
# updates + some slack.
|
||||||
return getattr(self.parent.obj, self.originalname) # type: ignore[attr-defined]
|
@property
|
||||||
|
def _testcase(self):
|
||||||
|
return self._obj.__self__
|
||||||
|
|
||||||
def setup(self) -> None:
|
def setup(self) -> None:
|
||||||
# A bound method to be called during teardown() if set (see 'runtest()').
|
# A bound method to be called during teardown() if set (see 'runtest()').
|
||||||
self._explicit_tearDown: Optional[Callable[[], None]] = None
|
self._explicit_tearDown: Optional[Callable[[], None]] = None
|
||||||
assert self.parent is not None
|
|
||||||
self._testcase = self.parent.obj(self.name) # type: ignore[attr-defined]
|
|
||||||
self._obj = getattr(self._testcase, self.name)
|
|
||||||
super().setup()
|
super().setup()
|
||||||
|
|
||||||
def teardown(self) -> None:
|
def teardown(self) -> None:
|
||||||
|
@ -193,7 +198,6 @@ class TestCaseFunction(Function):
|
||||||
if self._explicit_tearDown is not None:
|
if self._explicit_tearDown is not None:
|
||||||
self._explicit_tearDown()
|
self._explicit_tearDown()
|
||||||
self._explicit_tearDown = None
|
self._explicit_tearDown = None
|
||||||
self._testcase = None
|
|
||||||
self._obj = None
|
self._obj = None
|
||||||
|
|
||||||
def startTest(self, testcase: "unittest.TestCase") -> None:
|
def startTest(self, testcase: "unittest.TestCase") -> None:
|
||||||
|
@ -292,14 +296,14 @@ class TestCaseFunction(Function):
|
||||||
def runtest(self) -> None:
|
def runtest(self) -> None:
|
||||||
from _pytest.debugging import maybe_wrap_pytest_function_for_tracing
|
from _pytest.debugging import maybe_wrap_pytest_function_for_tracing
|
||||||
|
|
||||||
assert self._testcase is not None
|
testcase = self.obj.__self__
|
||||||
|
|
||||||
maybe_wrap_pytest_function_for_tracing(self)
|
maybe_wrap_pytest_function_for_tracing(self)
|
||||||
|
|
||||||
# Let the unittest framework handle async functions.
|
# Let the unittest framework handle async functions.
|
||||||
if is_async_function(self.obj):
|
if is_async_function(self.obj):
|
||||||
# Type ignored because self acts as the TestResult, but is not actually one.
|
# Type ignored because self acts as the TestResult, but is not actually one.
|
||||||
self._testcase(result=self) # type: ignore[arg-type]
|
testcase(result=self) # type: ignore[arg-type]
|
||||||
else:
|
else:
|
||||||
# When --pdb is given, we want to postpone calling tearDown() otherwise
|
# When --pdb is given, we want to postpone calling tearDown() otherwise
|
||||||
# when entering the pdb prompt, tearDown() would have probably cleaned up
|
# when entering the pdb prompt, tearDown() would have probably cleaned up
|
||||||
|
@ -311,16 +315,16 @@ class TestCaseFunction(Function):
|
||||||
assert isinstance(self.parent, UnitTestCase)
|
assert isinstance(self.parent, UnitTestCase)
|
||||||
skipped = _is_skipped(self.obj) or _is_skipped(self.parent.obj)
|
skipped = _is_skipped(self.obj) or _is_skipped(self.parent.obj)
|
||||||
if self.config.getoption("usepdb") and not skipped:
|
if self.config.getoption("usepdb") and not skipped:
|
||||||
self._explicit_tearDown = self._testcase.tearDown
|
self._explicit_tearDown = testcase.tearDown
|
||||||
setattr(self._testcase, "tearDown", lambda *args: None)
|
setattr(testcase, "tearDown", lambda *args: None)
|
||||||
|
|
||||||
# We need to update the actual bound method with self.obj, because
|
# We need to update the actual bound method with self.obj, because
|
||||||
# wrap_pytest_function_for_tracing replaces self.obj by a wrapper.
|
# wrap_pytest_function_for_tracing replaces self.obj by a wrapper.
|
||||||
setattr(self._testcase, self.name, self.obj)
|
setattr(testcase, self.name, self.obj)
|
||||||
try:
|
try:
|
||||||
self._testcase(result=self) # type: ignore[arg-type]
|
testcase(result=self) # type: ignore[arg-type]
|
||||||
finally:
|
finally:
|
||||||
delattr(self._testcase, self.name)
|
delattr(testcase, self.name)
|
||||||
|
|
||||||
def _traceback_filter(
|
def _traceback_filter(
|
||||||
self, excinfo: _pytest._code.ExceptionInfo[BaseException]
|
self, excinfo: _pytest._code.ExceptionInfo[BaseException]
|
||||||
|
|
|
@ -208,10 +208,14 @@ def test_teardown_issue1649(pytester: Pytester) -> None:
|
||||||
|
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
|
||||||
pytester.inline_run("-s", testpath)
|
pytester.inline_run("-s", testpath)
|
||||||
gc.collect()
|
gc.collect()
|
||||||
|
|
||||||
|
# Either already destroyed, or didn't run setUp.
|
||||||
for obj in gc.get_objects():
|
for obj in gc.get_objects():
|
||||||
assert type(obj).__name__ != "TestCaseObjectsShouldBeCleanedUp"
|
if type(obj).__name__ == "TestCaseObjectsShouldBeCleanedUp":
|
||||||
|
assert not hasattr(obj, "an_expensive_obj")
|
||||||
|
|
||||||
|
|
||||||
def test_unittest_skip_issue148(pytester: Pytester) -> None:
|
def test_unittest_skip_issue148(pytester: Pytester) -> None:
|
||||||
|
|
Loading…
Reference in New Issue