From fbc45be83f279f936121355649728f7aeec6e6a6 Mon Sep 17 00:00:00 2001 From: William Lee Date: Thu, 22 Feb 2018 21:00:54 -0600 Subject: [PATCH 1/6] Fixed #3149 where doctest does not continue to run when there is a failure --- AUTHORS | 1 + _pytest/doctest.py | 186 ++++++++++++++++++++++++++++++---------- testing/test_doctest.py | 19 ++++ 3 files changed, 161 insertions(+), 45 deletions(-) diff --git a/AUTHORS b/AUTHORS index cda6511a0..c3a9cfd2a 100644 --- a/AUTHORS +++ b/AUTHORS @@ -195,6 +195,7 @@ Victor Uriarte Vidar T. Fauske Vitaly Lashmanov Vlad Dragos +William Lee Wouter van Ackooy Xuan Luong Xuecong Liao diff --git a/_pytest/doctest.py b/_pytest/doctest.py index f54f833ec..1345d5868 100644 --- a/_pytest/doctest.py +++ b/_pytest/doctest.py @@ -24,6 +24,9 @@ DOCTEST_REPORT_CHOICES = ( DOCTEST_REPORT_CHOICE_ONLY_FIRST_FAILURE, ) +# Lazy definiton of runner class +RUNNER_CLASS = None + def pytest_addoption(parser): parser.addini('doctest_optionflags', 'option flags for doctests', @@ -77,14 +80,91 @@ def _is_doctest(config, path, parent): class ReprFailDoctest(TerminalRepr): - def __init__(self, reprlocation, lines): - self.reprlocation = reprlocation - self.lines = lines + def __init__(self, reprlocation_lines): + # List of (reprlocation, lines) tuples + self.reprlocation_lines = reprlocation_lines def toterminal(self, tw): - for line in self.lines: - tw.line(line) - self.reprlocation.toterminal(tw) + for reprlocation, lines in self.reprlocation_lines: + for line in lines: + tw.line(line) + reprlocation.toterminal(tw) + + +# class DoctestFailureContainer(object): +# +# NAME = 'DocTestFailure' +# +# def __init__(self, test, example, got): +# self.test = test +# self.example = example +# self.got = got +# +# +# class DoctestUnexpectedExceptionContainer(object): +# +# NAME = 'DoctestUnexpectedException' +# +# def __init__(self, test, example, exc_info): +# self.test = test +# self.example = example +# self.exc_info = exc_info + + +class MultipleDoctestFailures(Exception): + def __init__(self, failures): + super(MultipleDoctestFailures, self).__init__() + self.failures = failures + + +def _init_runner_class(): + import doctest + + class PytestDoctestRunner(doctest.DocTestRunner): + """ + Runner to collect failures. Note that the out variable in this case is + a list instead of a stdout-like object + """ + def __init__(self, checker=None, verbose=None, optionflags=0, + continue_on_failure=True): + doctest.DocTestRunner.__init__( + self, checker=checker, verbose=verbose, optionflags=optionflags) + self.continue_on_failure = continue_on_failure + + def report_start(self, out, test, example): + pass + + def report_success(self, out, test, example, got): + pass + + def report_failure(self, out, test, example, got): + # failure = DoctestFailureContainer(test, example, got) + failure = doctest.DocTestFailure(test, example, got) + if self.continue_on_failure: + out.append(failure) + else: + raise failure + + def report_unexpected_exception(self, out, test, example, exc_info): + # failure = DoctestUnexpectedExceptionContainer(test, example, exc_info) + failure = doctest.UnexpectedException(test, example, exc_info) + if self.continue_on_failure: + out.append(failure) + else: + raise failure + + return PytestDoctestRunner + + +def _get_runner(checker=None, verbose=None, optionflags=0, + continue_on_failure=True): + # We need this in order to do a lazy import on doctest + global RUNNER_CLASS + if RUNNER_CLASS is None: + RUNNER_CLASS = _init_runner_class() + return RUNNER_CLASS( + checker=checker, verbose=verbose, optionflags=optionflags, + continue_on_failure=continue_on_failure) class DoctestItem(pytest.Item): @@ -106,7 +186,10 @@ class DoctestItem(pytest.Item): def runtest(self): _check_all_skipped(self.dtest) self._disable_output_capturing_for_darwin() - self.runner.run(self.dtest) + failures = [] + self.runner.run(self.dtest, out=failures) + if failures: + raise MultipleDoctestFailures(failures) def _disable_output_capturing_for_darwin(self): """ @@ -122,42 +205,51 @@ class DoctestItem(pytest.Item): def repr_failure(self, excinfo): import doctest + failures = None if excinfo.errisinstance((doctest.DocTestFailure, doctest.UnexpectedException)): - doctestfailure = excinfo.value - example = doctestfailure.example - test = doctestfailure.test - filename = test.filename - if test.lineno is None: - lineno = None - else: - lineno = test.lineno + example.lineno + 1 - message = excinfo.type.__name__ - reprlocation = ReprFileLocation(filename, lineno, message) - checker = _get_checker() - report_choice = _get_report_choice(self.config.getoption("doctestreport")) - if lineno is not None: - lines = doctestfailure.test.docstring.splitlines(False) - # add line numbers to the left of the error message - lines = ["%03d %s" % (i + test.lineno + 1, x) - for (i, x) in enumerate(lines)] - # trim docstring error lines to 10 - lines = lines[max(example.lineno - 9, 0):example.lineno + 1] - else: - lines = ['EXAMPLE LOCATION UNKNOWN, not showing all tests of that example'] - indent = '>>>' - for line in example.source.splitlines(): - lines.append('??? %s %s' % (indent, line)) - indent = '...' - if excinfo.errisinstance(doctest.DocTestFailure): - lines += checker.output_difference(example, - doctestfailure.got, report_choice).split("\n") - else: - inner_excinfo = ExceptionInfo(excinfo.value.exc_info) - lines += ["UNEXPECTED EXCEPTION: %s" % - repr(inner_excinfo.value)] - lines += traceback.format_exception(*excinfo.value.exc_info) - return ReprFailDoctest(reprlocation, lines) + failures = [excinfo.value] + elif excinfo.errisinstance(MultipleDoctestFailures): + failures = excinfo.value.failures + + if failures is not None: + reprlocation_lines = [] + for failure in failures: + example = failure.example + test = failure.test + filename = test.filename + if test.lineno is None: + lineno = None + else: + lineno = test.lineno + example.lineno + 1 + message = type(failure).__name__ + reprlocation = ReprFileLocation(filename, lineno, message) + checker = _get_checker() + report_choice = _get_report_choice(self.config.getoption("doctestreport")) + if lineno is not None: + lines = failure.test.docstring.splitlines(False) + # add line numbers to the left of the error message + lines = ["%03d %s" % (i + test.lineno + 1, x) + for (i, x) in enumerate(lines)] + # trim docstring error lines to 10 + lines = lines[max(example.lineno - 9, 0):example.lineno + 1] + else: + lines = ['EXAMPLE LOCATION UNKNOWN, not showing all tests of that example'] + indent = '>>>' + for line in example.source.splitlines(): + lines.append('??? %s %s' % (indent, line)) + indent = '...' + if isinstance(failure, doctest.DocTestFailure): + lines += checker.output_difference(example, + failure.got, + report_choice).split("\n") + else: + inner_excinfo = ExceptionInfo(failure.exc_info) + lines += ["UNEXPECTED EXCEPTION: %s" % + repr(inner_excinfo.value)] + lines += traceback.format_exception(*failure.exc_info) + reprlocation_lines.append((reprlocation, lines)) + return ReprFailDoctest(reprlocation_lines) else: return super(DoctestItem, self).repr_failure(excinfo) @@ -202,8 +294,10 @@ class DoctestTextfile(pytest.Module): globs = {'__name__': '__main__'} optionflags = get_optionflags(self) - runner = doctest.DebugRunner(verbose=0, optionflags=optionflags, - checker=_get_checker()) + continue_on_failure = not self.config.getvalue("usepdb") + runner = _get_runner(verbose=0, optionflags=optionflags, + checker=_get_checker(), + continue_on_failure=continue_on_failure) _fix_spoof_python2(runner, encoding) parser = doctest.DocTestParser() @@ -238,8 +332,10 @@ class DoctestModule(pytest.Module): # uses internal doctest module parsing mechanism finder = doctest.DocTestFinder() optionflags = get_optionflags(self) - runner = doctest.DebugRunner(verbose=0, optionflags=optionflags, - checker=_get_checker()) + continue_on_failure = not self.config.getvalue("usepdb") + runner = _get_runner(verbose=0, optionflags=optionflags, + checker=_get_checker(), + continue_on_failure=continue_on_failure) for test in finder.find(module, module.__name__): if test.examples: # skip empty doctests diff --git a/testing/test_doctest.py b/testing/test_doctest.py index b15067f15..8c8f45224 100644 --- a/testing/test_doctest.py +++ b/testing/test_doctest.py @@ -756,6 +756,25 @@ class TestDoctestSkips(object): reprec = testdir.inline_run("--doctest-modules") reprec.assertoutcome(passed=0, skipped=0) + def test_continue_on_failure(self, testdir): + testdir.maketxtfile(test_something=""" + >>> i = 5 + >>> def foo(): + ... raise ValueError('error1') + >>> foo() + >>> i + >>> i + 2 + 7 + >>> i + 1 + """) + result = testdir.runpytest("--doctest-modules") + result.assert_outcomes(passed=0, failed=1) + # We need to make sure we have two failure lines (4, 5, and 8) instead of + # one. + result.stdout.fnmatch_lines("*test_something.txt:4: DoctestUnexpectedException*") + result.stdout.fnmatch_lines("*test_something.txt:5: DocTestFailure*") + result.stdout.fnmatch_lines("*test_something.txt:8: DocTestFailure*") + class TestDoctestAutoUseFixtures(object): From 14cd1e9d9438e86b0c0af9a9437abe578072aedc Mon Sep 17 00:00:00 2001 From: William Lee Date: Thu, 22 Feb 2018 21:01:19 -0600 Subject: [PATCH 2/6] Added the feature in change log for #3149 --- changelog/3149.feature | 1 + 1 file changed, 1 insertion(+) create mode 100644 changelog/3149.feature diff --git a/changelog/3149.feature b/changelog/3149.feature new file mode 100644 index 000000000..ed71b5a19 --- /dev/null +++ b/changelog/3149.feature @@ -0,0 +1 @@ +Doctest runs now show multiple failures for each doctest snippet, instead of stopping at the first failure. From 7f2dd74ae9ba23cc8c8a33e933b87df9bf708be0 Mon Sep 17 00:00:00 2001 From: William Lee Date: Fri, 23 Feb 2018 21:20:14 -0600 Subject: [PATCH 3/6] Fixed test for the continue run --- _pytest/doctest.py | 22 ---------------------- testing/test_doctest.py | 12 +++++++----- 2 files changed, 7 insertions(+), 27 deletions(-) diff --git a/_pytest/doctest.py b/_pytest/doctest.py index 1345d5868..1c16f4c84 100644 --- a/_pytest/doctest.py +++ b/_pytest/doctest.py @@ -91,26 +91,6 @@ class ReprFailDoctest(TerminalRepr): reprlocation.toterminal(tw) -# class DoctestFailureContainer(object): -# -# NAME = 'DocTestFailure' -# -# def __init__(self, test, example, got): -# self.test = test -# self.example = example -# self.got = got -# -# -# class DoctestUnexpectedExceptionContainer(object): -# -# NAME = 'DoctestUnexpectedException' -# -# def __init__(self, test, example, exc_info): -# self.test = test -# self.example = example -# self.exc_info = exc_info - - class MultipleDoctestFailures(Exception): def __init__(self, failures): super(MultipleDoctestFailures, self).__init__() @@ -138,7 +118,6 @@ def _init_runner_class(): pass def report_failure(self, out, test, example, got): - # failure = DoctestFailureContainer(test, example, got) failure = doctest.DocTestFailure(test, example, got) if self.continue_on_failure: out.append(failure) @@ -146,7 +125,6 @@ def _init_runner_class(): raise failure def report_unexpected_exception(self, out, test, example, exc_info): - # failure = DoctestUnexpectedExceptionContainer(test, example, exc_info) failure = doctest.UnexpectedException(test, example, exc_info) if self.continue_on_failure: out.append(failure) diff --git a/testing/test_doctest.py b/testing/test_doctest.py index 8c8f45224..931279820 100644 --- a/testing/test_doctest.py +++ b/testing/test_doctest.py @@ -769,11 +769,13 @@ class TestDoctestSkips(object): """) result = testdir.runpytest("--doctest-modules") result.assert_outcomes(passed=0, failed=1) - # We need to make sure we have two failure lines (4, 5, and 8) instead of - # one. - result.stdout.fnmatch_lines("*test_something.txt:4: DoctestUnexpectedException*") - result.stdout.fnmatch_lines("*test_something.txt:5: DocTestFailure*") - result.stdout.fnmatch_lines("*test_something.txt:8: DocTestFailure*") + # The lines that contains the failure are 4, 5, and 8. The first one + # is a stack trace and the other two are mismatches. + result.stdout.fnmatch_lines([ + "*4: UnexpectedException*", + "*5: DocTestFailure*", + "*8: DocTestFailure*", + ]) class TestDoctestAutoUseFixtures(object): From f4cc45bb41b29f1aeab61dc3c3219d1e1baadc66 Mon Sep 17 00:00:00 2001 From: William Lee Date: Fri, 23 Feb 2018 22:31:11 -0600 Subject: [PATCH 4/6] Turn on the continue on failure only when the flag is given --- _pytest/doctest.py | 41 +++++++++++++++++++++++++---------------- testing/test_doctest.py | 2 +- 2 files changed, 26 insertions(+), 17 deletions(-) diff --git a/_pytest/doctest.py b/_pytest/doctest.py index 1c16f4c84..03775a09a 100644 --- a/_pytest/doctest.py +++ b/_pytest/doctest.py @@ -50,6 +50,10 @@ def pytest_addoption(parser): action="store_true", default=False, help="ignore doctest ImportErrors", dest="doctest_ignore_import_errors") + group.addoption("--doctest-continue-on-failure", + action="store_true", default=False, + help="for a given doctest, continue to run after the first failure", + dest="doctest_continue_on_failure") def pytest_collect_file(path, parent): @@ -100,23 +104,17 @@ class MultipleDoctestFailures(Exception): def _init_runner_class(): import doctest - class PytestDoctestRunner(doctest.DocTestRunner): + class PytestDoctestRunner(doctest.DebugRunner): """ Runner to collect failures. Note that the out variable in this case is a list instead of a stdout-like object """ def __init__(self, checker=None, verbose=None, optionflags=0, continue_on_failure=True): - doctest.DocTestRunner.__init__( + doctest.DebugRunner.__init__( self, checker=checker, verbose=verbose, optionflags=optionflags) self.continue_on_failure = continue_on_failure - def report_start(self, out, test, example): - pass - - def report_success(self, out, test, example, got): - pass - def report_failure(self, out, test, example, got): failure = doctest.DocTestFailure(test, example, got) if self.continue_on_failure: @@ -257,6 +255,16 @@ def get_optionflags(parent): return flag_acc +def _get_continue_on_failure(config): + continue_on_failure = config.getvalue('doctest_continue_on_failure') + if continue_on_failure: + # We need to turn off this if we use pdb since we should stop at + # the first failure + if config.getvalue("usepdb"): + continue_on_failure = False + return continue_on_failure + + class DoctestTextfile(pytest.Module): obj = None @@ -272,10 +280,11 @@ class DoctestTextfile(pytest.Module): globs = {'__name__': '__main__'} optionflags = get_optionflags(self) - continue_on_failure = not self.config.getvalue("usepdb") - runner = _get_runner(verbose=0, optionflags=optionflags, - checker=_get_checker(), - continue_on_failure=continue_on_failure) + + runner = _get_runner( + verbose=0, optionflags=optionflags, + checker=_get_checker(), + continue_on_failure=_get_continue_on_failure(self.config)) _fix_spoof_python2(runner, encoding) parser = doctest.DocTestParser() @@ -310,10 +319,10 @@ class DoctestModule(pytest.Module): # uses internal doctest module parsing mechanism finder = doctest.DocTestFinder() optionflags = get_optionflags(self) - continue_on_failure = not self.config.getvalue("usepdb") - runner = _get_runner(verbose=0, optionflags=optionflags, - checker=_get_checker(), - continue_on_failure=continue_on_failure) + runner = _get_runner( + verbose=0, optionflags=optionflags, + checker=_get_checker(), + continue_on_failure=_get_continue_on_failure(self.config)) for test in finder.find(module, module.__name__): if test.examples: # skip empty doctests diff --git a/testing/test_doctest.py b/testing/test_doctest.py index 931279820..314398395 100644 --- a/testing/test_doctest.py +++ b/testing/test_doctest.py @@ -767,7 +767,7 @@ class TestDoctestSkips(object): 7 >>> i + 1 """) - result = testdir.runpytest("--doctest-modules") + result = testdir.runpytest("--doctest-modules", "--doctest-continue-on-failure") result.assert_outcomes(passed=0, failed=1) # The lines that contains the failure are 4, 5, and 8. The first one # is a stack trace and the other two are mismatches. From c21eb7292451aa53b5fd24bc7e8f2ea9fb7ab912 Mon Sep 17 00:00:00 2001 From: William Lee Date: Fri, 23 Feb 2018 22:42:22 -0600 Subject: [PATCH 5/6] Added documentation for the continue-on-failure flag --- doc/en/doctest.rst | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/doc/en/doctest.rst b/doc/en/doctest.rst index 61fbe04d4..cdbc34682 100644 --- a/doc/en/doctest.rst +++ b/doc/en/doctest.rst @@ -115,6 +115,11 @@ itself:: >>> get_unicode_greeting() # doctest: +ALLOW_UNICODE 'Hello' +By default, pytest would report only the first failure for a given doctest. If +you want to continue the test even when you have failures, do:: + + pytest --doctest-modules --doctest-continue-on-failure + The 'doctest_namespace' fixture ------------------------------- From e8f9a910563f72b599ad6b97903553373cbed433 Mon Sep 17 00:00:00 2001 From: Bruno Oliveira Date: Mon, 26 Feb 2018 17:10:59 -0300 Subject: [PATCH 6/6] Small adjustment to CHANGELOG entry --- changelog/3149.feature | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/changelog/3149.feature b/changelog/3149.feature index ed71b5a19..0431f76ce 100644 --- a/changelog/3149.feature +++ b/changelog/3149.feature @@ -1 +1 @@ -Doctest runs now show multiple failures for each doctest snippet, instead of stopping at the first failure. +New ``--doctest-continue-on-failure`` command-line option to enable doctests to show multiple failures for each snippet, instead of stopping at the first failure.