Make cache plugin always remember failed tests
This commit is contained in:
parent
e97fd5ec55
commit
62810f61b2
|
@ -105,27 +105,22 @@ class LFPlugin:
|
||||||
self.config = config
|
self.config = config
|
||||||
active_keys = 'lf', 'failedfirst'
|
active_keys = 'lf', 'failedfirst'
|
||||||
self.active = any(config.getvalue(key) for key in active_keys)
|
self.active = any(config.getvalue(key) for key in active_keys)
|
||||||
if self.active:
|
|
||||||
self.lastfailed = config.cache.get("cache/lastfailed", {})
|
self.lastfailed = config.cache.get("cache/lastfailed", {})
|
||||||
else:
|
|
||||||
self.lastfailed = {}
|
|
||||||
|
|
||||||
def pytest_report_header(self):
|
def pytest_report_header(self):
|
||||||
if self.active:
|
if self.active:
|
||||||
if not self.lastfailed:
|
if not self.lastfailed:
|
||||||
mode = "run all (no recorded failures)"
|
mode = "run all (no recorded failures)"
|
||||||
else:
|
else:
|
||||||
mode = "rerun last %d failures%s" % (
|
mode = "rerun previous failures%s" % (
|
||||||
len(self.lastfailed),
|
|
||||||
" first" if self.config.getvalue("failedfirst") else "")
|
" first" if self.config.getvalue("failedfirst") else "")
|
||||||
return "run-last-failure: %s" % mode
|
return "run-last-failure: %s" % mode
|
||||||
|
|
||||||
def pytest_runtest_logreport(self, report):
|
def pytest_runtest_logreport(self, report):
|
||||||
if report.failed and "xfail" not in report.keywords:
|
if report.passed and report.when == 'call':
|
||||||
self.lastfailed[report.nodeid] = True
|
|
||||||
elif not report.failed:
|
|
||||||
if report.when == "call":
|
|
||||||
self.lastfailed.pop(report.nodeid, None)
|
self.lastfailed.pop(report.nodeid, None)
|
||||||
|
elif report.failed:
|
||||||
|
self.lastfailed[report.nodeid] = True
|
||||||
|
|
||||||
def pytest_collectreport(self, report):
|
def pytest_collectreport(self, report):
|
||||||
passed = report.outcome in ('passed', 'skipped')
|
passed = report.outcome in ('passed', 'skipped')
|
||||||
|
@ -147,11 +142,11 @@ class LFPlugin:
|
||||||
previously_failed.append(item)
|
previously_failed.append(item)
|
||||||
else:
|
else:
|
||||||
previously_passed.append(item)
|
previously_passed.append(item)
|
||||||
if not previously_failed and previously_passed:
|
if not previously_failed:
|
||||||
# running a subset of all tests with recorded failures outside
|
# running a subset of all tests with recorded failures outside
|
||||||
# of the set of tests currently executing
|
# of the set of tests currently executing
|
||||||
pass
|
return
|
||||||
elif self.config.getvalue("lf"):
|
if self.config.getvalue("lf"):
|
||||||
items[:] = previously_failed
|
items[:] = previously_failed
|
||||||
config.hook.pytest_deselected(items=previously_passed)
|
config.hook.pytest_deselected(items=previously_passed)
|
||||||
else:
|
else:
|
||||||
|
@ -161,8 +156,9 @@ class LFPlugin:
|
||||||
config = self.config
|
config = self.config
|
||||||
if config.getvalue("cacheshow") or hasattr(config, "slaveinput"):
|
if config.getvalue("cacheshow") or hasattr(config, "slaveinput"):
|
||||||
return
|
return
|
||||||
prev_failed = config.cache.get("cache/lastfailed", None) is not None
|
|
||||||
if (session.testscollected and prev_failed) or self.lastfailed:
|
saved_lastfailed = config.cache.get("cache/lastfailed", {})
|
||||||
|
if saved_lastfailed != self.lastfailed:
|
||||||
config.cache.set("cache/lastfailed", self.lastfailed)
|
config.cache.set("cache/lastfailed", self.lastfailed)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
``--last-failed`` now remembers forever when a test has failed and only forgets it if it passes again. This makes it
|
||||||
|
easy to fix a test suite by selectively running files and fixing tests incrementally.
|
|
@ -437,3 +437,58 @@ class TestLastFailed(object):
|
||||||
testdir.makepyfile(test_errored='def test_error():\n assert False')
|
testdir.makepyfile(test_errored='def test_error():\n assert False')
|
||||||
testdir.runpytest('-q', '--lf')
|
testdir.runpytest('-q', '--lf')
|
||||||
assert os.path.exists('.cache')
|
assert os.path.exists('.cache')
|
||||||
|
|
||||||
|
def get_cached_last_failed(self, testdir):
|
||||||
|
config = testdir.parseconfigure()
|
||||||
|
return sorted(config.cache.get("cache/lastfailed", {}))
|
||||||
|
|
||||||
|
def test_cache_cumulative(self, testdir):
|
||||||
|
"""
|
||||||
|
Test workflow where user fixes errors gradually file by file using --lf.
|
||||||
|
"""
|
||||||
|
# 1. initial run
|
||||||
|
test_bar = testdir.makepyfile(test_bar="""
|
||||||
|
def test_bar_1():
|
||||||
|
pass
|
||||||
|
def test_bar_2():
|
||||||
|
assert 0
|
||||||
|
""")
|
||||||
|
test_foo = testdir.makepyfile(test_foo="""
|
||||||
|
def test_foo_3():
|
||||||
|
pass
|
||||||
|
def test_foo_4():
|
||||||
|
assert 0
|
||||||
|
""")
|
||||||
|
testdir.runpytest()
|
||||||
|
assert self.get_cached_last_failed(testdir) == ['test_bar.py::test_bar_2', 'test_foo.py::test_foo_4']
|
||||||
|
|
||||||
|
# 2. fix test_bar_2, run only test_bar.py
|
||||||
|
testdir.makepyfile(test_bar="""
|
||||||
|
def test_bar_1():
|
||||||
|
pass
|
||||||
|
def test_bar_2():
|
||||||
|
pass
|
||||||
|
""")
|
||||||
|
result = testdir.runpytest(test_bar)
|
||||||
|
result.stdout.fnmatch_lines('*2 passed*')
|
||||||
|
# ensure cache does not forget that test_foo_4 failed once before
|
||||||
|
assert self.get_cached_last_failed(testdir) == ['test_foo.py::test_foo_4']
|
||||||
|
|
||||||
|
result = testdir.runpytest('--last-failed')
|
||||||
|
result.stdout.fnmatch_lines('*1 failed, 3 deselected*')
|
||||||
|
assert self.get_cached_last_failed(testdir) == ['test_foo.py::test_foo_4']
|
||||||
|
|
||||||
|
# 3. fix test_foo_4, run only test_foo.py
|
||||||
|
test_foo = testdir.makepyfile(test_foo="""
|
||||||
|
def test_foo_3():
|
||||||
|
pass
|
||||||
|
def test_foo_4():
|
||||||
|
pass
|
||||||
|
""")
|
||||||
|
result = testdir.runpytest(test_foo, '--last-failed')
|
||||||
|
result.stdout.fnmatch_lines('*1 passed, 1 deselected*')
|
||||||
|
assert self.get_cached_last_failed(testdir) == []
|
||||||
|
|
||||||
|
result = testdir.runpytest('--last-failed')
|
||||||
|
result.stdout.fnmatch_lines('*4 passed*')
|
||||||
|
assert self.get_cached_last_failed(testdir) == []
|
||||||
|
|
Loading…
Reference in New Issue