2010-11-06 18:38:53 +08:00
|
|
|
""" discover and run doctests in modules and test files."""
|
2017-03-17 09:21:30 +08:00
|
|
|
from __future__ import absolute_import, division, print_function
|
2015-11-27 22:43:01 +08:00
|
|
|
|
2014-08-01 06:13:40 +08:00
|
|
|
import traceback
|
2018-02-14 01:08:39 +08:00
|
|
|
import sys
|
2018-02-14 06:51:20 +08:00
|
|
|
import platform
|
2015-11-27 22:43:01 +08:00
|
|
|
|
|
|
|
import pytest
|
2016-07-23 20:50:24 +08:00
|
|
|
from _pytest._code.code import ExceptionInfo, ReprFileLocation, TerminalRepr
|
2016-07-10 02:36:00 +08:00
|
|
|
from _pytest.fixtures import FixtureRequest
|
2015-11-27 22:43:01 +08:00
|
|
|
|
2009-02-27 18:18:27 +08:00
|
|
|
|
2016-07-23 22:18:12 +08:00
|
|
|
DOCTEST_REPORT_CHOICE_NONE = 'none'
|
|
|
|
DOCTEST_REPORT_CHOICE_CDIFF = 'cdiff'
|
|
|
|
DOCTEST_REPORT_CHOICE_NDIFF = 'ndiff'
|
|
|
|
DOCTEST_REPORT_CHOICE_UDIFF = 'udiff'
|
|
|
|
DOCTEST_REPORT_CHOICE_ONLY_FIRST_FAILURE = 'only_first_failure'
|
|
|
|
|
|
|
|
DOCTEST_REPORT_CHOICES = (
|
|
|
|
DOCTEST_REPORT_CHOICE_NONE,
|
|
|
|
DOCTEST_REPORT_CHOICE_CDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_NDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_UDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_ONLY_FIRST_FAILURE,
|
|
|
|
)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2018-05-13 18:06:09 +08:00
|
|
|
# Lazy definition of runner class
|
2018-02-23 11:00:54 +08:00
|
|
|
RUNNER_CLASS = None
|
|
|
|
|
2017-07-17 07:25:09 +08:00
|
|
|
|
2009-05-19 05:26:16 +08:00
|
|
|
def pytest_addoption(parser):
|
2014-10-08 20:31:17 +08:00
|
|
|
parser.addini('doctest_optionflags', 'option flags for doctests',
|
2017-07-17 07:25:07 +08:00
|
|
|
type="args", default=["ELLIPSIS"])
|
2016-11-30 18:43:33 +08:00
|
|
|
parser.addini("doctest_encoding", 'encoding used for doctest files', default="utf-8")
|
2010-01-03 19:41:29 +08:00
|
|
|
group = parser.getgroup("collect")
|
2010-07-27 03:15:15 +08:00
|
|
|
group.addoption("--doctest-modules",
|
2017-07-17 07:25:07 +08:00
|
|
|
action="store_true", default=False,
|
|
|
|
help="run doctests in all .py modules",
|
|
|
|
dest="doctestmodules")
|
2016-07-23 19:06:05 +08:00
|
|
|
group.addoption("--doctest-report",
|
2017-07-17 07:25:07 +08:00
|
|
|
type=str.lower, default="udiff",
|
|
|
|
help="choose another output format for diffs on doctest failure",
|
|
|
|
choices=DOCTEST_REPORT_CHOICES,
|
|
|
|
dest="doctestreport")
|
2010-01-03 06:30:46 +08:00
|
|
|
group.addoption("--doctest-glob",
|
2017-07-17 07:25:07 +08:00
|
|
|
action="append", default=[], metavar="pat",
|
|
|
|
help="doctests file matching pattern, default: test*.txt",
|
|
|
|
dest="doctestglob")
|
2015-02-08 14:25:23 +08:00
|
|
|
group.addoption("--doctest-ignore-import-errors",
|
2017-07-17 07:25:07 +08:00
|
|
|
action="store_true", default=False,
|
|
|
|
help="ignore doctest ImportErrors",
|
|
|
|
dest="doctest_ignore_import_errors")
|
2018-02-24 12:31:11 +08:00
|
|
|
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")
|
2010-01-03 06:30:46 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2009-05-19 05:26:16 +08:00
|
|
|
def pytest_collect_file(path, parent):
|
2010-01-03 06:30:46 +08:00
|
|
|
config = parent.config
|
2009-05-19 05:26:16 +08:00
|
|
|
if path.ext == ".py":
|
2017-10-13 18:57:52 +08:00
|
|
|
if config.option.doctestmodules and not _is_setup_py(config, path, parent):
|
2009-05-19 05:26:16 +08:00
|
|
|
return DoctestModule(path, parent)
|
2015-12-12 03:58:49 +08:00
|
|
|
elif _is_doctest(config, path, parent):
|
2009-05-19 05:26:16 +08:00
|
|
|
return DoctestTextfile(path, parent)
|
2009-02-27 18:18:27 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2017-10-13 18:57:52 +08:00
|
|
|
def _is_setup_py(config, path, parent):
|
|
|
|
if path.basename != "setup.py":
|
|
|
|
return False
|
2017-10-14 00:47:02 +08:00
|
|
|
contents = path.read()
|
2017-10-13 18:57:52 +08:00
|
|
|
return 'setuptools' in contents or 'distutils' in contents
|
|
|
|
|
|
|
|
|
2015-12-12 03:58:49 +08:00
|
|
|
def _is_doctest(config, path, parent):
|
|
|
|
if path.ext in ('.txt', '.rst') and parent.session.isinitpath(path):
|
|
|
|
return True
|
2015-12-31 04:19:08 +08:00
|
|
|
globs = config.getoption("doctestglob") or ['test*.txt']
|
2015-12-12 03:58:49 +08:00
|
|
|
for glob in globs:
|
|
|
|
if path.check(fnmatch=glob):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2009-08-26 02:24:43 +08:00
|
|
|
class ReprFailDoctest(TerminalRepr):
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2018-02-23 11:00:54 +08:00
|
|
|
def __init__(self, reprlocation_lines):
|
|
|
|
# List of (reprlocation, lines) tuples
|
|
|
|
self.reprlocation_lines = reprlocation_lines
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2009-02-27 18:18:27 +08:00
|
|
|
def toterminal(self, tw):
|
2018-02-23 11:00:54 +08:00
|
|
|
for reprlocation, lines in self.reprlocation_lines:
|
|
|
|
for line in lines:
|
|
|
|
tw.line(line)
|
|
|
|
reprlocation.toterminal(tw)
|
|
|
|
|
|
|
|
|
|
|
|
class MultipleDoctestFailures(Exception):
|
|
|
|
def __init__(self, failures):
|
|
|
|
super(MultipleDoctestFailures, self).__init__()
|
|
|
|
self.failures = failures
|
|
|
|
|
|
|
|
|
|
|
|
def _init_runner_class():
|
|
|
|
import doctest
|
|
|
|
|
2018-02-24 12:31:11 +08:00
|
|
|
class PytestDoctestRunner(doctest.DebugRunner):
|
2018-02-23 11:00:54 +08:00
|
|
|
"""
|
|
|
|
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):
|
2018-02-24 12:31:11 +08:00
|
|
|
doctest.DebugRunner.__init__(
|
2018-02-23 11:00:54 +08:00
|
|
|
self, checker=checker, verbose=verbose, optionflags=optionflags)
|
|
|
|
self.continue_on_failure = continue_on_failure
|
|
|
|
|
|
|
|
def report_failure(self, out, 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 = 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)
|
2010-07-27 03:15:15 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2010-11-13 16:05:11 +08:00
|
|
|
class DoctestItem(pytest.Item):
|
2013-05-17 23:18:22 +08:00
|
|
|
def __init__(self, name, parent, runner=None, dtest=None):
|
|
|
|
super(DoctestItem, self).__init__(name, parent)
|
|
|
|
self.runner = runner
|
|
|
|
self.dtest = dtest
|
2015-10-02 09:59:37 +08:00
|
|
|
self.obj = None
|
|
|
|
self.fixture_request = None
|
|
|
|
|
|
|
|
def setup(self):
|
|
|
|
if self.dtest is not None:
|
|
|
|
self.fixture_request = _setup_fixtures(self)
|
2016-06-21 18:09:55 +08:00
|
|
|
globs = dict(getfixture=self.fixture_request.getfixturevalue)
|
2016-09-01 19:19:11 +08:00
|
|
|
for name, value in self.fixture_request.getfixturevalue('doctest_namespace').items():
|
2016-03-02 20:43:57 +08:00
|
|
|
globs[name] = value
|
2015-10-02 09:59:37 +08:00
|
|
|
self.dtest.globs.update(globs)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
|
|
|
def runtest(self):
|
2015-08-26 10:12:02 +08:00
|
|
|
_check_all_skipped(self.dtest)
|
2018-02-15 22:17:33 +08:00
|
|
|
self._disable_output_capturing_for_darwin()
|
2018-02-23 11:00:54 +08:00
|
|
|
failures = []
|
|
|
|
self.runner.run(self.dtest, out=failures)
|
|
|
|
if failures:
|
|
|
|
raise MultipleDoctestFailures(failures)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
2018-02-15 22:17:33 +08:00
|
|
|
def _disable_output_capturing_for_darwin(self):
|
2018-02-14 01:08:39 +08:00
|
|
|
"""
|
|
|
|
Disable output capturing. Otherwise, stdout is lost to doctest (#985)
|
|
|
|
"""
|
2018-02-14 06:51:20 +08:00
|
|
|
if platform.system() != 'Darwin':
|
|
|
|
return
|
2018-02-14 01:08:39 +08:00
|
|
|
capman = self.config.pluginmanager.getplugin("capturemanager")
|
|
|
|
if capman:
|
|
|
|
out, err = capman.suspend_global_capture(in_=True)
|
|
|
|
sys.stdout.write(out)
|
2018-02-18 01:13:33 +08:00
|
|
|
sys.stderr.write(err)
|
2018-02-14 01:08:39 +08:00
|
|
|
|
2009-07-26 00:09:01 +08:00
|
|
|
def repr_failure(self, excinfo):
|
2014-08-01 06:13:40 +08:00
|
|
|
import doctest
|
2018-02-23 11:00:54 +08:00
|
|
|
failures = None
|
2010-12-07 02:00:30 +08:00
|
|
|
if excinfo.errisinstance((doctest.DocTestFailure,
|
|
|
|
doctest.UnexpectedException)):
|
2018-02-23 11:00:54 +08:00
|
|
|
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)
|
2010-07-27 03:15:15 +08:00
|
|
|
else:
|
2009-07-26 00:09:01 +08:00
|
|
|
return super(DoctestItem, self).repr_failure(excinfo)
|
2009-02-27 18:18:27 +08:00
|
|
|
|
2010-09-26 22:23:44 +08:00
|
|
|
def reportinfo(self):
|
2017-07-24 12:52:24 +08:00
|
|
|
return self.fspath, self.dtest.lineno, "[doctest] %s" % self.name
|
2010-09-26 22:23:44 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2014-10-08 20:31:17 +08:00
|
|
|
def _get_flag_lookup():
|
|
|
|
import doctest
|
|
|
|
return dict(DONT_ACCEPT_TRUE_FOR_1=doctest.DONT_ACCEPT_TRUE_FOR_1,
|
|
|
|
DONT_ACCEPT_BLANKLINE=doctest.DONT_ACCEPT_BLANKLINE,
|
|
|
|
NORMALIZE_WHITESPACE=doctest.NORMALIZE_WHITESPACE,
|
|
|
|
ELLIPSIS=doctest.ELLIPSIS,
|
|
|
|
IGNORE_EXCEPTION_DETAIL=doctest.IGNORE_EXCEPTION_DETAIL,
|
2015-08-13 08:41:31 +08:00
|
|
|
COMPARISON_FLAGS=doctest.COMPARISON_FLAGS,
|
2015-12-30 06:55:19 +08:00
|
|
|
ALLOW_UNICODE=_get_allow_unicode_flag(),
|
|
|
|
ALLOW_BYTES=_get_allow_bytes_flag(),
|
|
|
|
)
|
2014-10-08 20:31:17 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
2014-10-08 20:31:17 +08:00
|
|
|
def get_optionflags(parent):
|
|
|
|
optionflags_str = parent.config.getini("doctest_optionflags")
|
|
|
|
flag_lookup_table = _get_flag_lookup()
|
|
|
|
flag_acc = 0
|
|
|
|
for flag in optionflags_str:
|
|
|
|
flag_acc |= flag_lookup_table[flag]
|
|
|
|
return flag_acc
|
|
|
|
|
2017-07-17 07:25:09 +08:00
|
|
|
|
2018-02-24 12:31:11 +08:00
|
|
|
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
|
|
|
|
|
|
|
|
|
2016-06-02 00:16:05 +08:00
|
|
|
class DoctestTextfile(pytest.Module):
|
|
|
|
obj = None
|
2015-08-13 08:41:31 +08:00
|
|
|
|
2016-06-02 00:16:05 +08:00
|
|
|
def collect(self):
|
2014-08-01 06:13:40 +08:00
|
|
|
import doctest
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
# inspired by doctest.testfile; ideally we would use it directly,
|
|
|
|
# but it doesn't support passing a custom checker
|
2016-11-30 18:43:33 +08:00
|
|
|
encoding = self.config.getini("doctest_encoding")
|
|
|
|
text = self.fspath.read_text(encoding)
|
2015-08-13 08:41:31 +08:00
|
|
|
filename = str(self.fspath)
|
|
|
|
name = self.fspath.basename
|
2016-06-02 00:16:05 +08:00
|
|
|
globs = {'__name__': '__main__'}
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
optionflags = get_optionflags(self)
|
2018-02-24 12:31:11 +08:00
|
|
|
|
|
|
|
runner = _get_runner(
|
|
|
|
verbose=0, optionflags=optionflags,
|
|
|
|
checker=_get_checker(),
|
|
|
|
continue_on_failure=_get_continue_on_failure(self.config))
|
2017-05-26 04:08:32 +08:00
|
|
|
_fix_spoof_python2(runner, encoding)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
parser = doctest.DocTestParser()
|
|
|
|
test = parser.get_doctest(text, globs, name, filename, 0)
|
2016-06-02 00:16:05 +08:00
|
|
|
if test.examples:
|
|
|
|
yield DoctestItem(test.name, self, runner, test)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
2009-02-27 18:18:27 +08:00
|
|
|
|
2015-08-26 10:12:02 +08:00
|
|
|
def _check_all_skipped(test):
|
|
|
|
"""raises pytest.skip() if all examples in the given DocTest have the SKIP
|
|
|
|
option set.
|
|
|
|
"""
|
|
|
|
import doctest
|
|
|
|
all_skipped = all(x.options.get(doctest.SKIP, False) for x in test.examples)
|
|
|
|
if all_skipped:
|
|
|
|
pytest.skip('all tests skipped by +SKIP option')
|
|
|
|
|
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
class DoctestModule(pytest.Module):
|
2013-05-17 23:18:22 +08:00
|
|
|
def collect(self):
|
2014-08-01 06:13:40 +08:00
|
|
|
import doctest
|
2010-11-18 22:31:58 +08:00
|
|
|
if self.fspath.basename == "conftest.py":
|
2015-07-13 10:43:33 +08:00
|
|
|
module = self.config.pluginmanager._importconftest(self.fspath)
|
2010-11-18 22:31:58 +08:00
|
|
|
else:
|
2015-02-08 14:25:23 +08:00
|
|
|
try:
|
|
|
|
module = self.fspath.pyimport()
|
|
|
|
except ImportError:
|
|
|
|
if self.config.getvalue('doctest_ignore_import_errors'):
|
|
|
|
pytest.skip('unable to import module %r' % self.fspath)
|
|
|
|
else:
|
|
|
|
raise
|
2013-05-17 23:18:22 +08:00
|
|
|
# uses internal doctest module parsing mechanism
|
|
|
|
finder = doctest.DocTestFinder()
|
2015-02-08 14:25:23 +08:00
|
|
|
optionflags = get_optionflags(self)
|
2018-02-24 12:31:11 +08:00
|
|
|
runner = _get_runner(
|
|
|
|
verbose=0, optionflags=optionflags,
|
|
|
|
checker=_get_checker(),
|
|
|
|
continue_on_failure=_get_continue_on_failure(self.config))
|
2017-05-26 04:08:32 +08:00
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
for test in finder.find(module, module.__name__):
|
2015-02-08 14:25:23 +08:00
|
|
|
if test.examples: # skip empty doctests
|
2013-05-17 23:18:22 +08:00
|
|
|
yield DoctestItem(test.name, self, runner, test)
|
2015-07-13 10:43:33 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _setup_fixtures(doctest_item):
|
|
|
|
"""
|
2015-10-02 09:59:37 +08:00
|
|
|
Used by DoctestTextfile and DoctestItem to setup fixture information.
|
2015-07-13 10:43:33 +08:00
|
|
|
"""
|
|
|
|
def func():
|
|
|
|
pass
|
|
|
|
|
|
|
|
doctest_item.funcargs = {}
|
|
|
|
fm = doctest_item.session._fixturemanager
|
|
|
|
doctest_item._fixtureinfo = fm.getfixtureinfo(node=doctest_item, func=func,
|
|
|
|
cls=None, funcargs=False)
|
|
|
|
fixture_request = FixtureRequest(doctest_item)
|
|
|
|
fixture_request._fillfixtures()
|
|
|
|
return fixture_request
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
def _get_checker():
|
2015-08-13 08:41:31 +08:00
|
|
|
"""
|
|
|
|
Returns a doctest.OutputChecker subclass that takes in account the
|
2015-12-30 06:55:19 +08:00
|
|
|
ALLOW_UNICODE option to ignore u'' prefixes in strings and ALLOW_BYTES
|
|
|
|
to strip b'' prefixes.
|
|
|
|
Useful when the same doctest should run in Python 2 and Python 3.
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
An inner class is used to avoid importing "doctest" at the module
|
|
|
|
level.
|
|
|
|
"""
|
2015-12-30 06:55:19 +08:00
|
|
|
if hasattr(_get_checker, 'LiteralsOutputChecker'):
|
|
|
|
return _get_checker.LiteralsOutputChecker()
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
import doctest
|
|
|
|
import re
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
class LiteralsOutputChecker(doctest.OutputChecker):
|
2015-08-13 08:41:31 +08:00
|
|
|
"""
|
|
|
|
Copied from doctest_nose_plugin.py from the nltk project:
|
|
|
|
https://github.com/nltk/nltk
|
2015-12-30 06:55:19 +08:00
|
|
|
|
|
|
|
Further extended to also support byte literals.
|
2015-08-13 08:41:31 +08:00
|
|
|
"""
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
_unicode_literal_re = re.compile(r"(\W|^)[uU]([rR]?[\'\"])", re.UNICODE)
|
|
|
|
_bytes_literal_re = re.compile(r"(\W|^)[bB]([rR]?[\'\"])", re.UNICODE)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
def check_output(self, want, got, optionflags):
|
2015-08-13 09:46:13 +08:00
|
|
|
res = doctest.OutputChecker.check_output(self, want, got,
|
|
|
|
optionflags)
|
2015-08-13 08:41:31 +08:00
|
|
|
if res:
|
|
|
|
return True
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
allow_unicode = optionflags & _get_allow_unicode_flag()
|
|
|
|
allow_bytes = optionflags & _get_allow_bytes_flag()
|
|
|
|
if not allow_unicode and not allow_bytes:
|
2015-08-13 08:41:31 +08:00
|
|
|
return False
|
|
|
|
|
2015-08-13 09:46:13 +08:00
|
|
|
else: # pragma: no cover
|
2015-12-30 06:55:19 +08:00
|
|
|
def remove_prefixes(regex, txt):
|
|
|
|
return re.sub(regex, r'\1\2', txt)
|
|
|
|
|
|
|
|
if allow_unicode:
|
|
|
|
want = remove_prefixes(self._unicode_literal_re, want)
|
|
|
|
got = remove_prefixes(self._unicode_literal_re, got)
|
|
|
|
if allow_bytes:
|
|
|
|
want = remove_prefixes(self._bytes_literal_re, want)
|
|
|
|
got = remove_prefixes(self._bytes_literal_re, got)
|
2015-08-13 09:46:13 +08:00
|
|
|
res = doctest.OutputChecker.check_output(self, want, got,
|
|
|
|
optionflags)
|
|
|
|
return res
|
2015-08-13 08:41:31 +08:00
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
_get_checker.LiteralsOutputChecker = LiteralsOutputChecker
|
|
|
|
return _get_checker.LiteralsOutputChecker()
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _get_allow_unicode_flag():
|
|
|
|
"""
|
|
|
|
Registers and returns the ALLOW_UNICODE flag.
|
|
|
|
"""
|
|
|
|
import doctest
|
|
|
|
return doctest.register_optionflag('ALLOW_UNICODE')
|
2015-12-30 06:55:19 +08:00
|
|
|
|
|
|
|
|
|
|
|
def _get_allow_bytes_flag():
|
|
|
|
"""
|
|
|
|
Registers and returns the ALLOW_BYTES flag.
|
|
|
|
"""
|
|
|
|
import doctest
|
|
|
|
return doctest.register_optionflag('ALLOW_BYTES')
|
2016-03-02 20:43:57 +08:00
|
|
|
|
|
|
|
|
2016-07-23 22:18:12 +08:00
|
|
|
def _get_report_choice(key):
|
2016-07-23 21:19:18 +08:00
|
|
|
"""
|
2016-07-23 22:18:12 +08:00
|
|
|
This function returns the actual `doctest` module flag value, we want to do it as late as possible to avoid
|
|
|
|
importing `doctest` and all its dependencies when parsing options, as it adds overhead and breaks tests.
|
2016-07-23 21:19:18 +08:00
|
|
|
"""
|
2016-07-23 19:06:05 +08:00
|
|
|
import doctest
|
2016-07-23 22:18:12 +08:00
|
|
|
|
|
|
|
return {
|
|
|
|
DOCTEST_REPORT_CHOICE_UDIFF: doctest.REPORT_UDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_CDIFF: doctest.REPORT_CDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_NDIFF: doctest.REPORT_NDIFF,
|
|
|
|
DOCTEST_REPORT_CHOICE_ONLY_FIRST_FAILURE: doctest.REPORT_ONLY_FIRST_FAILURE,
|
|
|
|
DOCTEST_REPORT_CHOICE_NONE: 0,
|
|
|
|
}[key]
|
2016-07-23 19:06:05 +08:00
|
|
|
|
2017-05-26 04:08:32 +08:00
|
|
|
|
|
|
|
def _fix_spoof_python2(runner, encoding):
|
|
|
|
"""
|
2017-06-14 06:34:05 +08:00
|
|
|
Installs a "SpoofOut" into the given DebugRunner so it properly deals with unicode output. This
|
|
|
|
should patch only doctests for text files because they don't have a way to declare their
|
|
|
|
encoding. Doctests in docstrings from Python modules don't have the same problem given that
|
|
|
|
Python already decoded the strings.
|
2017-07-17 07:25:06 +08:00
|
|
|
|
2017-05-26 04:08:32 +08:00
|
|
|
This fixes the problem related in issue #2434.
|
|
|
|
"""
|
|
|
|
from _pytest.compat import _PY2
|
|
|
|
if not _PY2:
|
|
|
|
return
|
|
|
|
|
|
|
|
from doctest import _SpoofOut
|
|
|
|
|
|
|
|
class UnicodeSpoof(_SpoofOut):
|
|
|
|
|
|
|
|
def getvalue(self):
|
|
|
|
result = _SpoofOut.getvalue(self)
|
|
|
|
if encoding:
|
|
|
|
result = result.decode(encoding)
|
|
|
|
return result
|
|
|
|
|
|
|
|
runner._fakeout = UnicodeSpoof()
|
|
|
|
|
|
|
|
|
2016-03-02 20:43:57 +08:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def doctest_namespace():
|
|
|
|
"""
|
2018-02-09 06:08:44 +08:00
|
|
|
Fixture that returns a :py:class:`dict` that will be injected into the namespace of doctests.
|
2016-03-02 20:43:57 +08:00
|
|
|
"""
|
|
|
|
return dict()
|