Make cache plugin always remember failed tests

This commit is contained in:
Bruno Oliveira 2017-07-26 21:06:08 -03:00
parent e97fd5ec55
commit 62810f61b2
3 changed files with 68 additions and 15 deletions

View File

@ -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.pop(report.nodeid, None)
elif report.failed:
self.lastfailed[report.nodeid] = True self.lastfailed[report.nodeid] = True
elif not report.failed:
if report.when == "call":
self.lastfailed.pop(report.nodeid, None)
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)

2
changelog/2621.feature Normal file
View File

@ -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.

View File

@ -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) == []