Extend the incremental marker for parametrize
The incremental marker is adapted to handle properly test classes with parametrize defined at class level. Fix #3125
This commit is contained in:
parent
a9eab07739
commit
d848a20563
|
@ -461,21 +461,49 @@ an ``incremental`` marker which is to be used on classes:
|
||||||
|
|
||||||
# content of conftest.py
|
# content of conftest.py
|
||||||
|
|
||||||
import pytest
|
# store history of failures per test class name and per index in parametrize (if parametrize used)
|
||||||
|
_test_failed_incremental: Dict[str, Dict[Tuple[int, ...], str]] = {}
|
||||||
|
|
||||||
|
|
||||||
def pytest_runtest_makereport(item, call):
|
def pytest_runtest_makereport(item, call):
|
||||||
if "incremental" in item.keywords:
|
if "incremental" in item.keywords:
|
||||||
|
# incremental marker is used
|
||||||
if call.excinfo is not None:
|
if call.excinfo is not None:
|
||||||
parent = item.parent
|
# the test has failed
|
||||||
parent._previousfailed = item
|
# retrieve the class name of the test
|
||||||
|
cls_name = str(item.cls)
|
||||||
|
# retrieve the index of the test (if parametrize is used in combination with incremental)
|
||||||
|
parametrize_index = (
|
||||||
|
tuple(item.callspec.indices.values())
|
||||||
|
if hasattr(item, "callspec")
|
||||||
|
else ()
|
||||||
|
)
|
||||||
|
# retrieve the name of the test function
|
||||||
|
test_name = item.originalname or item.name
|
||||||
|
# store in _test_failed_incremental the original name of the failed test
|
||||||
|
_test_failed_incremental.setdefault(cls_name, {}).setdefault(
|
||||||
|
parametrize_index, test_name
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def pytest_runtest_setup(item):
|
def pytest_runtest_setup(item):
|
||||||
if "incremental" in item.keywords:
|
if "incremental" in item.keywords:
|
||||||
previousfailed = getattr(item.parent, "_previousfailed", None)
|
# retrieve the class name of the test
|
||||||
if previousfailed is not None:
|
cls_name = str(item.cls)
|
||||||
pytest.xfail("previous test failed ({})".format(previousfailed.name))
|
# check if a previous test has failed for this class
|
||||||
|
if cls_name in _test_failed_incremental:
|
||||||
|
# retrieve the index of the test (if parametrize is used in combination with incremental)
|
||||||
|
parametrize_index = (
|
||||||
|
tuple(item.callspec.indices.values())
|
||||||
|
if hasattr(item, "callspec")
|
||||||
|
else ()
|
||||||
|
)
|
||||||
|
# retrieve the name of the first test function to fail for this class name and index
|
||||||
|
test_name = _test_failed_incremental[cls_name].get(parametrize_index, None)
|
||||||
|
# if name found, test has failed for the combination of class name & test name
|
||||||
|
if test_name is not None:
|
||||||
|
pytest.xfail("previous test failed ({})".format(test_name))
|
||||||
|
|
||||||
|
|
||||||
These two hook implementations work together to abort incremental-marked
|
These two hook implementations work together to abort incremental-marked
|
||||||
tests in a class. Here is a test module example:
|
tests in a class. Here is a test module example:
|
||||||
|
|
Loading…
Reference in New Issue