Adding Failed exception to manage maxfail behavior

This commit is contained in:
Jesús Espino 2017-10-17 17:53:47 +02:00
parent 71c76d96d3
commit 00d3abe6dc
4 changed files with 16 additions and 6 deletions

View File

@ -102,6 +102,8 @@ def wrap_session(config, doit):
session.exitstatus = doit(config, session) or 0 session.exitstatus = doit(config, session) or 0
except UsageError: except UsageError:
raise raise
except Failed:
session.exitstatus = EXIT_TESTSFAILED
except KeyboardInterrupt: except KeyboardInterrupt:
excinfo = _pytest._code.ExceptionInfo() excinfo = _pytest._code.ExceptionInfo()
if initstate < 2 and isinstance(excinfo.value, exit.Exception): if initstate < 2 and isinstance(excinfo.value, exit.Exception):
@ -159,6 +161,8 @@ def pytest_runtestloop(session):
for i, item in enumerate(session.items): for i, item in enumerate(session.items):
nextitem = session.items[i + 1] if i + 1 < len(session.items) else None nextitem = session.items[i + 1] if i + 1 < len(session.items) else None
item.config.hook.pytest_runtest_protocol(item=item, nextitem=nextitem) item.config.hook.pytest_runtest_protocol(item=item, nextitem=nextitem)
if session.shouldfail:
raise session.Failed(session.shouldfail)
if session.shouldstop: if session.shouldstop:
raise session.Interrupted(session.shouldstop) raise session.Interrupted(session.shouldstop)
return True return True
@ -564,8 +568,13 @@ class Interrupted(KeyboardInterrupt):
__module__ = 'builtins' # for py3 __module__ = 'builtins' # for py3
class Failed(Exception):
""" signals an stop as failed test run. """
class Session(FSCollector): class Session(FSCollector):
Interrupted = Interrupted Interrupted = Interrupted
Failed = Failed
def __init__(self, config): def __init__(self, config):
FSCollector.__init__(self, config.rootdir, parent=None, FSCollector.__init__(self, config.rootdir, parent=None,
@ -573,6 +582,7 @@ class Session(FSCollector):
self.testsfailed = 0 self.testsfailed = 0
self.testscollected = 0 self.testscollected = 0
self.shouldstop = False self.shouldstop = False
self.shouldfail = False
self.trace = config.trace.root.get("collection") self.trace = config.trace.root.get("collection")
self._norecursepatterns = config.getini("norecursedirs") self._norecursepatterns = config.getini("norecursedirs")
self.startdir = py.path.local() self.startdir = py.path.local()
@ -583,6 +593,8 @@ class Session(FSCollector):
@hookimpl(tryfirst=True) @hookimpl(tryfirst=True)
def pytest_collectstart(self): def pytest_collectstart(self):
if self.shouldfail:
raise self.Failed(self.shouldfail)
if self.shouldstop: if self.shouldstop:
raise self.Interrupted(self.shouldstop) raise self.Interrupted(self.shouldstop)
@ -592,7 +604,7 @@ class Session(FSCollector):
self.testsfailed += 1 self.testsfailed += 1
maxfail = self.config.getvalue("maxfail") maxfail = self.config.getvalue("maxfail")
if maxfail and self.testsfailed >= maxfail: if maxfail and self.testsfailed >= maxfail:
self.shouldstop = "stopping after %d failures" % ( self.shouldfail = "stopping after %d failures" % (
self.testsfailed) self.testsfailed)
pytest_collectreport = pytest_runtest_logreport pytest_collectreport = pytest_runtest_logreport

1
changelog/2845.bugfix Normal file
View File

@ -0,0 +1 @@
Change return value of pytest command when maxfail is reached from 2 to 1.

View File

@ -767,12 +767,11 @@ def test_exit_on_collection_with_maxfail_smaller_than_n_errors(testdir):
testdir.makepyfile(**COLLECTION_ERROR_PY_FILES) testdir.makepyfile(**COLLECTION_ERROR_PY_FILES)
res = testdir.runpytest("--maxfail=1") res = testdir.runpytest("--maxfail=1")
assert res.ret == 2 assert res.ret == 1
res.stdout.fnmatch_lines([ res.stdout.fnmatch_lines([
"*ERROR collecting test_02_import_error.py*", "*ERROR collecting test_02_import_error.py*",
"*No module named *asdfa*", "*No module named *asdfa*",
"*Interrupted: stopping after 1 failures*",
]) ])
assert 'test_03' not in res.stdout.str() assert 'test_03' not in res.stdout.str()
@ -824,10 +823,9 @@ def test_continue_on_collection_errors_maxfail(testdir):
testdir.makepyfile(**COLLECTION_ERROR_PY_FILES) testdir.makepyfile(**COLLECTION_ERROR_PY_FILES)
res = testdir.runpytest("--continue-on-collection-errors", "--maxfail=3") res = testdir.runpytest("--continue-on-collection-errors", "--maxfail=3")
assert res.ret == 2 assert res.ret == 1
res.stdout.fnmatch_lines([ res.stdout.fnmatch_lines([
"collected 2 items / 2 errors", "collected 2 items / 2 errors",
"*Interrupted: stopping after 3 failures*",
"*1 failed, 2 error*", "*1 failed, 2 error*",
]) ])

View File

@ -757,7 +757,6 @@ class TestGenericReporting(object):
result.stdout.fnmatch_lines([ result.stdout.fnmatch_lines([
"*def test_1():*", "*def test_1():*",
"*def test_2():*", "*def test_2():*",
"*!! Interrupted: stopping after 2 failures*!!*",
"*2 failed*", "*2 failed*",
]) ])