2019-06-08 10:39:59 +08:00
|
|
|
|
import inspect
|
2018-08-24 00:06:17 +08:00
|
|
|
|
import textwrap
|
2018-10-25 15:01:29 +08:00
|
|
|
|
|
2015-08-13 08:41:31 +08:00
|
|
|
|
import pytest
|
2018-10-25 15:01:29 +08:00
|
|
|
|
from _pytest.compat import MODULE_NOT_FOUND_ERROR
|
2019-07-08 22:41:33 +08:00
|
|
|
|
from _pytest.doctest import _get_checker
|
2019-06-08 10:39:59 +08:00
|
|
|
|
from _pytest.doctest import _is_mocked
|
|
|
|
|
from _pytest.doctest import _patch_unwrap_mock_aware
|
2018-10-25 15:01:29 +08:00
|
|
|
|
from _pytest.doctest import DoctestItem
|
|
|
|
|
from _pytest.doctest import DoctestModule
|
|
|
|
|
from _pytest.doctest import DoctestTextfile
|
2009-09-06 22:59:39 +08:00
|
|
|
|
|
2016-12-13 20:49:44 +08:00
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestDoctests:
|
2009-09-06 22:59:39 +08:00
|
|
|
|
def test_collect_testtextfile(self, testdir):
|
2010-11-17 21:33:21 +08:00
|
|
|
|
w = testdir.maketxtfile(whatever="")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
checkfile = testdir.maketxtfile(
|
|
|
|
|
test_something="""
|
2009-09-06 22:59:39 +08:00
|
|
|
|
alskdjalsdk
|
|
|
|
|
>>> i = 5
|
|
|
|
|
>>> i-1
|
|
|
|
|
4
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2016-06-02 00:16:05 +08:00
|
|
|
|
|
2010-07-27 03:15:15 +08:00
|
|
|
|
for x in (testdir.tmpdir, checkfile):
|
2017-07-17 07:25:09 +08:00
|
|
|
|
# print "checking that %s returns custom items" % (x,)
|
2010-01-03 06:30:46 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(x)
|
2009-09-06 22:59:39 +08:00
|
|
|
|
assert len(items) == 1
|
2016-06-02 00:16:05 +08:00
|
|
|
|
assert isinstance(items[0], DoctestItem)
|
|
|
|
|
assert isinstance(items[0].parent, DoctestTextfile)
|
|
|
|
|
# Empty file has no items.
|
2010-11-17 21:33:21 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(w)
|
2016-06-02 00:16:05 +08:00
|
|
|
|
assert len(items) == 0
|
2009-09-06 22:59:39 +08:00
|
|
|
|
|
2013-05-17 23:18:22 +08:00
|
|
|
|
def test_collect_module_empty(self, testdir):
|
2009-09-06 22:59:39 +08:00
|
|
|
|
path = testdir.makepyfile(whatever="#")
|
2013-05-17 23:18:22 +08:00
|
|
|
|
for p in (path, testdir.tmpdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(p, "--doctest-modules")
|
2013-05-17 23:18:22 +08:00
|
|
|
|
assert len(items) == 0
|
|
|
|
|
|
|
|
|
|
def test_collect_module_single_modulelevel_doctest(self, testdir):
|
|
|
|
|
path = testdir.makepyfile(whatever='""">>> pass"""')
|
2010-07-27 03:15:15 +08:00
|
|
|
|
for p in (path, testdir.tmpdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(p, "--doctest-modules")
|
2009-09-06 22:59:39 +08:00
|
|
|
|
assert len(items) == 1
|
2013-05-17 23:18:22 +08:00
|
|
|
|
assert isinstance(items[0], DoctestItem)
|
|
|
|
|
assert isinstance(items[0].parent, DoctestModule)
|
|
|
|
|
|
|
|
|
|
def test_collect_module_two_doctest_one_modulelevel(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
path = testdir.makepyfile(
|
|
|
|
|
whatever="""
|
2013-05-17 23:18:22 +08:00
|
|
|
|
'>>> x = None'
|
|
|
|
|
def my_func():
|
|
|
|
|
">>> magic = 42 "
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
for p in (path, testdir.tmpdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(p, "--doctest-modules")
|
2013-05-17 23:18:22 +08:00
|
|
|
|
assert len(items) == 2
|
|
|
|
|
assert isinstance(items[0], DoctestItem)
|
|
|
|
|
assert isinstance(items[1], DoctestItem)
|
|
|
|
|
assert isinstance(items[0].parent, DoctestModule)
|
|
|
|
|
assert items[0].parent is items[1].parent
|
|
|
|
|
|
|
|
|
|
def test_collect_module_two_doctest_no_modulelevel(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
path = testdir.makepyfile(
|
|
|
|
|
whatever="""
|
2013-05-17 23:18:22 +08:00
|
|
|
|
'# Empty'
|
|
|
|
|
def my_func():
|
|
|
|
|
">>> magic = 42 "
|
|
|
|
|
def unuseful():
|
|
|
|
|
'''
|
|
|
|
|
# This is a function
|
|
|
|
|
# >>> # it doesn't have any doctest
|
|
|
|
|
'''
|
|
|
|
|
def another():
|
|
|
|
|
'''
|
|
|
|
|
# This is another function
|
|
|
|
|
>>> import os # this one does have a doctest
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
for p in (path, testdir.tmpdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
items, reprec = testdir.inline_genitems(p, "--doctest-modules")
|
2013-05-17 23:18:22 +08:00
|
|
|
|
assert len(items) == 2
|
|
|
|
|
assert isinstance(items[0], DoctestItem)
|
|
|
|
|
assert isinstance(items[1], DoctestItem)
|
|
|
|
|
assert isinstance(items[0].parent, DoctestModule)
|
|
|
|
|
assert items[0].parent is items[1].parent
|
2009-09-06 22:59:39 +08:00
|
|
|
|
|
|
|
|
|
def test_simple_doctestfile(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2009-09-06 22:59:39 +08:00
|
|
|
|
>>> x = 1
|
|
|
|
|
>>> x == 1
|
|
|
|
|
False
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run(p)
|
2010-01-03 06:30:46 +08:00
|
|
|
|
reprec.assertoutcome(failed=1)
|
|
|
|
|
|
|
|
|
|
def test_new_pattern(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
xdoc="""
|
2010-01-03 06:30:46 +08:00
|
|
|
|
>>> x = 1
|
|
|
|
|
>>> x == 1
|
|
|
|
|
False
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2010-01-03 06:30:46 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-glob=x*.txt")
|
2009-09-06 22:59:39 +08:00
|
|
|
|
reprec.assertoutcome(failed=1)
|
|
|
|
|
|
2015-12-31 04:19:08 +08:00
|
|
|
|
def test_multiple_patterns(self, testdir):
|
|
|
|
|
"""Test support for multiple --doctest-glob arguments (#1255).
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
xdoc="""
|
2015-12-31 04:19:08 +08:00
|
|
|
|
>>> 1
|
|
|
|
|
1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
testdir.makefile(
|
|
|
|
|
".foo",
|
|
|
|
|
test="""
|
2015-12-31 04:19:08 +08:00
|
|
|
|
>>> 1
|
|
|
|
|
1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""",
|
|
|
|
|
)
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_normal="""
|
2015-12-31 04:19:08 +08:00
|
|
|
|
>>> 1
|
|
|
|
|
1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
expected = {"xdoc.txt", "test.foo", "test_normal.txt"}
|
2018-05-18 05:31:16 +08:00
|
|
|
|
assert {x.basename for x in testdir.tmpdir.listdir()} == expected
|
2015-12-31 04:19:08 +08:00
|
|
|
|
args = ["--doctest-glob=xdoc*.txt", "--doctest-glob=*.foo"]
|
|
|
|
|
result = testdir.runpytest(*args)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*test.foo *", "*xdoc.txt *", "*2 passed*"])
|
2015-12-31 04:19:08 +08:00
|
|
|
|
result = testdir.runpytest()
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*test_normal.txt *", "*1 passed*"])
|
2015-12-31 04:19:08 +08:00
|
|
|
|
|
2016-11-30 18:43:33 +08:00
|
|
|
|
@pytest.mark.parametrize(
|
2018-05-23 22:48:46 +08:00
|
|
|
|
" test_string, encoding",
|
2019-06-03 06:32:00 +08:00
|
|
|
|
[("foo", "ascii"), ("öäü", "latin1"), ("öäü", "utf-8")],
|
2016-11-30 18:43:33 +08:00
|
|
|
|
)
|
|
|
|
|
def test_encoding(self, testdir, test_string, encoding):
|
|
|
|
|
"""Test support for doctest_encoding ini option.
|
2016-11-29 19:18:41 +08:00
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2016-11-30 18:43:33 +08:00
|
|
|
|
[pytest]
|
2018-05-18 05:31:16 +08:00
|
|
|
|
doctest_encoding={}
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
encoding
|
|
|
|
|
)
|
|
|
|
|
)
|
2019-06-03 06:32:00 +08:00
|
|
|
|
doctest = """
|
2019-06-05 08:48:06 +08:00
|
|
|
|
>>> "{}"
|
2018-05-18 05:31:16 +08:00
|
|
|
|
{}
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
test_string, repr(test_string)
|
|
|
|
|
)
|
2016-11-30 18:43:33 +08:00
|
|
|
|
testdir._makefile(".txt", [doctest], {}, encoding=encoding)
|
2016-11-29 19:18:41 +08:00
|
|
|
|
|
|
|
|
|
result = testdir.runpytest()
|
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*1 passed*"])
|
2016-11-29 19:18:41 +08:00
|
|
|
|
|
2009-09-06 22:59:39 +08:00
|
|
|
|
def test_doctest_unexpected_exception(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
"""
|
2009-09-06 22:59:39 +08:00
|
|
|
|
>>> i = 0
|
2010-12-07 02:00:30 +08:00
|
|
|
|
>>> 0 / i
|
2009-09-06 22:59:39 +08:00
|
|
|
|
2
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2010-12-07 02:00:30 +08:00
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
2020-02-24 22:18:08 +08:00
|
|
|
|
"test_doctest_unexpected_exception.txt F *",
|
|
|
|
|
"",
|
|
|
|
|
"*= FAILURES =*",
|
|
|
|
|
"*_ [[]doctest[]] test_doctest_unexpected_exception.txt _*",
|
|
|
|
|
"001 >>> i = 0",
|
|
|
|
|
"002 >>> 0 / i",
|
|
|
|
|
"UNEXPECTED EXCEPTION: ZeroDivisionError*",
|
|
|
|
|
"Traceback (most recent call last):",
|
|
|
|
|
' File "*/doctest.py", line *, in __run',
|
|
|
|
|
" *",
|
|
|
|
|
' File "<doctest test_doctest_unexpected_exception.txt[1]>", line 1, in <module>',
|
|
|
|
|
"ZeroDivisionError: division by zero",
|
|
|
|
|
"*/test_doctest_unexpected_exception.txt:2: UnexpectedException",
|
|
|
|
|
],
|
|
|
|
|
consecutive=True,
|
2018-05-23 22:48:46 +08:00
|
|
|
|
)
|
2009-09-06 22:59:39 +08:00
|
|
|
|
|
2020-02-19 20:16:37 +08:00
|
|
|
|
def test_doctest_outcomes(self, testdir):
|
2019-03-15 10:06:57 +08:00
|
|
|
|
testdir.maketxtfile(
|
2020-02-19 20:16:37 +08:00
|
|
|
|
test_skip="""
|
2019-03-15 10:06:57 +08:00
|
|
|
|
>>> 1
|
|
|
|
|
1
|
|
|
|
|
>>> import pytest
|
|
|
|
|
>>> pytest.skip("")
|
2020-02-19 20:16:37 +08:00
|
|
|
|
>>> 2
|
|
|
|
|
3
|
|
|
|
|
""",
|
|
|
|
|
test_xfail="""
|
|
|
|
|
>>> import pytest
|
|
|
|
|
>>> pytest.xfail("xfail_reason")
|
|
|
|
|
>>> foo
|
|
|
|
|
bar
|
|
|
|
|
""",
|
|
|
|
|
test_importorskip="""
|
|
|
|
|
>>> import pytest
|
|
|
|
|
>>> pytest.importorskip("doesnotexist")
|
|
|
|
|
>>> foo
|
|
|
|
|
bar
|
|
|
|
|
""",
|
2019-03-15 10:06:57 +08:00
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2020-02-19 20:16:37 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"collected 3 items",
|
|
|
|
|
"",
|
|
|
|
|
"test_importorskip.txt s *",
|
|
|
|
|
"test_skip.txt s *",
|
|
|
|
|
"test_xfail.txt x *",
|
|
|
|
|
"",
|
|
|
|
|
"*= 2 skipped, 1 xfailed in *",
|
|
|
|
|
]
|
|
|
|
|
)
|
2019-03-15 10:06:57 +08:00
|
|
|
|
|
2017-11-04 02:37:18 +08:00
|
|
|
|
def test_docstring_partial_context_around_error(self, testdir):
|
2016-01-09 08:19:37 +08:00
|
|
|
|
"""Test that we show some context before the actual line of a failing
|
|
|
|
|
doctest.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
'''
|
2016-01-09 08:19:37 +08:00
|
|
|
|
def foo():
|
|
|
|
|
"""
|
|
|
|
|
text-line-1
|
|
|
|
|
text-line-2
|
|
|
|
|
text-line-3
|
|
|
|
|
text-line-4
|
|
|
|
|
text-line-5
|
|
|
|
|
text-line-6
|
|
|
|
|
text-line-7
|
|
|
|
|
text-line-8
|
|
|
|
|
text-line-9
|
|
|
|
|
text-line-10
|
|
|
|
|
text-line-11
|
|
|
|
|
>>> 1 + 1
|
|
|
|
|
3
|
|
|
|
|
|
|
|
|
|
text-line-after
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
'''
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*docstring_partial_context_around_error*",
|
|
|
|
|
"005*text-line-3",
|
|
|
|
|
"006*text-line-4",
|
|
|
|
|
"013*text-line-11",
|
|
|
|
|
"014*>>> 1 + 1",
|
|
|
|
|
"Expected:",
|
|
|
|
|
" 3",
|
|
|
|
|
"Got:",
|
|
|
|
|
" 2",
|
|
|
|
|
]
|
|
|
|
|
)
|
2016-01-09 08:19:37 +08:00
|
|
|
|
# lines below should be trimmed out
|
2019-10-06 01:18:51 +08:00
|
|
|
|
result.stdout.no_fnmatch_line("*text-line-2*")
|
|
|
|
|
result.stdout.no_fnmatch_line("*text-line-after*")
|
2016-01-09 08:19:37 +08:00
|
|
|
|
|
2017-11-04 02:37:18 +08:00
|
|
|
|
def test_docstring_full_context_around_error(self, testdir):
|
|
|
|
|
"""Test that we show the whole context before the actual line of a failing
|
|
|
|
|
doctest, provided that the context is up to 10 lines long.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
'''
|
2017-11-04 02:37:18 +08:00
|
|
|
|
def foo():
|
|
|
|
|
"""
|
|
|
|
|
text-line-1
|
|
|
|
|
text-line-2
|
|
|
|
|
|
|
|
|
|
>>> 1 + 1
|
|
|
|
|
3
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
'''
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*docstring_full_context_around_error*",
|
|
|
|
|
"003*text-line-1",
|
|
|
|
|
"004*text-line-2",
|
|
|
|
|
"006*>>> 1 + 1",
|
|
|
|
|
"Expected:",
|
|
|
|
|
" 3",
|
|
|
|
|
"Got:",
|
|
|
|
|
" 2",
|
|
|
|
|
]
|
|
|
|
|
)
|
2017-11-04 02:37:18 +08:00
|
|
|
|
|
2013-03-25 03:05:29 +08:00
|
|
|
|
def test_doctest_linedata_missing(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.tmpdir.join("hello.py").write(
|
2018-08-24 00:06:17 +08:00
|
|
|
|
textwrap.dedent(
|
|
|
|
|
"""\
|
|
|
|
|
class Fun(object):
|
|
|
|
|
@property
|
|
|
|
|
def test(self):
|
|
|
|
|
'''
|
|
|
|
|
>>> a = 1
|
|
|
|
|
>>> 1/0
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
)
|
2013-03-25 03:05:29 +08:00
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2019-10-28 05:28:21 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
["*hello*", "006*>>> 1/0*", "*UNEXPECTED*ZeroDivision*", "*1 failed*"]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def test_doctest_linedata_on_property(self, testdir):
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
"""
|
|
|
|
|
class Sample(object):
|
|
|
|
|
@property
|
|
|
|
|
def some_property(self):
|
|
|
|
|
'''
|
|
|
|
|
>>> Sample().some_property
|
|
|
|
|
'another thing'
|
|
|
|
|
'''
|
|
|
|
|
return 'something'
|
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
2019-10-28 05:28:21 +08:00
|
|
|
|
"*= FAILURES =*",
|
|
|
|
|
"*_ [[]doctest[]] test_doctest_linedata_on_property.Sample.some_property _*",
|
|
|
|
|
"004 ",
|
|
|
|
|
"005 >>> Sample().some_property",
|
|
|
|
|
"Expected:",
|
|
|
|
|
" 'another thing'",
|
|
|
|
|
"Got:",
|
|
|
|
|
" 'something'",
|
|
|
|
|
"",
|
|
|
|
|
"*/test_doctest_linedata_on_property.py:5: DocTestFailure",
|
|
|
|
|
"*= 1 failed in *",
|
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def test_doctest_no_linedata_on_overriden_property(self, testdir):
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
"""
|
|
|
|
|
class Sample(object):
|
|
|
|
|
@property
|
|
|
|
|
def some_property(self):
|
|
|
|
|
'''
|
|
|
|
|
>>> Sample().some_property
|
|
|
|
|
'another thing'
|
|
|
|
|
'''
|
|
|
|
|
return 'something'
|
|
|
|
|
some_property = property(some_property.__get__, None, None, some_property.__doc__)
|
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*= FAILURES =*",
|
|
|
|
|
"*_ [[]doctest[]] test_doctest_no_linedata_on_overriden_property.Sample.some_property _*",
|
|
|
|
|
"EXAMPLE LOCATION UNKNOWN, not showing all tests of that example",
|
|
|
|
|
"[?][?][?] >>> Sample().some_property",
|
|
|
|
|
"Expected:",
|
|
|
|
|
" 'another thing'",
|
|
|
|
|
"Got:",
|
|
|
|
|
" 'something'",
|
|
|
|
|
"",
|
|
|
|
|
"*/test_doctest_no_linedata_on_overriden_property.py:None: DocTestFailure",
|
|
|
|
|
"*= 1 failed in *",
|
2018-05-23 22:48:46 +08:00
|
|
|
|
]
|
|
|
|
|
)
|
2013-03-25 03:05:29 +08:00
|
|
|
|
|
2016-06-20 21:05:50 +08:00
|
|
|
|
def test_doctest_unex_importerror_only_txt(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
"""
|
2016-06-20 21:05:50 +08:00
|
|
|
|
>>> import asdalsdkjaslkdjasd
|
|
|
|
|
>>>
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2016-06-20 21:05:50 +08:00
|
|
|
|
result = testdir.runpytest()
|
|
|
|
|
# doctest is never executed because of error during hello.py collection
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*>>> import asdals*",
|
|
|
|
|
"*UNEXPECTED*{e}*".format(e=MODULE_NOT_FOUND_ERROR),
|
|
|
|
|
"{e}: No module named *asdal*".format(e=MODULE_NOT_FOUND_ERROR),
|
|
|
|
|
]
|
|
|
|
|
)
|
2013-03-25 03:05:29 +08:00
|
|
|
|
|
2016-06-20 21:05:50 +08:00
|
|
|
|
def test_doctest_unex_importerror_with_module(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.tmpdir.join("hello.py").write(
|
2018-08-24 00:06:17 +08:00
|
|
|
|
textwrap.dedent(
|
|
|
|
|
"""\
|
|
|
|
|
import asdalsdkjaslkdjasd
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
"""
|
2011-05-28 20:38:15 +08:00
|
|
|
|
>>> import hello
|
|
|
|
|
>>>
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2011-05-28 20:38:15 +08:00
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2016-06-20 21:05:50 +08:00
|
|
|
|
# doctest is never executed because of error during hello.py collection
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*ERROR collecting hello.py*",
|
|
|
|
|
"*{e}: No module named *asdals*".format(e=MODULE_NOT_FOUND_ERROR),
|
2019-10-27 23:02:37 +08:00
|
|
|
|
"*Interrupted: 1 error during collection*",
|
2018-05-23 22:48:46 +08:00
|
|
|
|
]
|
|
|
|
|
)
|
2011-05-28 20:38:15 +08:00
|
|
|
|
|
2009-09-06 22:59:39 +08:00
|
|
|
|
def test_doctestmodule(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2009-09-06 22:59:39 +08:00
|
|
|
|
'''
|
|
|
|
|
>>> x = 1
|
|
|
|
|
>>> x == 1
|
|
|
|
|
False
|
|
|
|
|
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2010-01-03 06:30:46 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
2010-07-27 03:15:15 +08:00
|
|
|
|
reprec.assertoutcome(failed=1)
|
2009-09-06 22:59:39 +08:00
|
|
|
|
|
2010-09-04 15:21:35 +08:00
|
|
|
|
def test_doctestmodule_external_and_issue116(self, testdir):
|
|
|
|
|
p = testdir.mkpydir("hello")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p.join("__init__.py").write(
|
2018-08-24 00:06:17 +08:00
|
|
|
|
textwrap.dedent(
|
|
|
|
|
"""\
|
|
|
|
|
def somefunc():
|
|
|
|
|
'''
|
|
|
|
|
>>> i = 0
|
|
|
|
|
>>> i + 1
|
|
|
|
|
2
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
)
|
2010-01-03 06:30:46 +08:00
|
|
|
|
result = testdir.runpytest(p, "--doctest-modules")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
2018-08-24 00:06:17 +08:00
|
|
|
|
"003 *>>> i = 0",
|
|
|
|
|
"004 *>>> i + 1",
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"*Expected:",
|
|
|
|
|
"* 2",
|
|
|
|
|
"*Got:",
|
|
|
|
|
"* 1",
|
2018-08-24 00:06:17 +08:00
|
|
|
|
"*:4: DocTestFailure",
|
2018-05-23 22:48:46 +08:00
|
|
|
|
]
|
|
|
|
|
)
|
2010-07-27 03:15:15 +08:00
|
|
|
|
|
2009-09-06 22:59:39 +08:00
|
|
|
|
def test_txtfile_failing(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
"""
|
2009-09-06 22:59:39 +08:00
|
|
|
|
>>> i = 0
|
|
|
|
|
>>> i + 1
|
|
|
|
|
2
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2011-05-29 15:21:48 +08:00
|
|
|
|
result = testdir.runpytest(p, "-s")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"001 >>> i = 0",
|
|
|
|
|
"002 >>> i + 1",
|
|
|
|
|
"Expected:",
|
|
|
|
|
" 2",
|
|
|
|
|
"Got:",
|
|
|
|
|
" 1",
|
|
|
|
|
"*test_txtfile_failing.txt:2: DocTestFailure",
|
|
|
|
|
]
|
|
|
|
|
)
|
2013-03-21 00:14:28 +08:00
|
|
|
|
|
2013-03-21 08:03:59 +08:00
|
|
|
|
def test_txtfile_with_fixtures(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
"""
|
2013-03-21 19:04:14 +08:00
|
|
|
|
>>> dir = getfixture('tmpdir')
|
2013-03-21 00:14:28 +08:00
|
|
|
|
>>> type(dir).__name__
|
|
|
|
|
'LocalPath'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run(p)
|
2013-03-21 00:14:28 +08:00
|
|
|
|
reprec.assertoutcome(passed=1)
|
2013-03-21 00:32:48 +08:00
|
|
|
|
|
2013-11-22 22:35:20 +08:00
|
|
|
|
def test_txtfile_with_usefixtures_in_ini(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2013-11-22 22:35:20 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
usefixtures = myfixture
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2013-11-22 22:35:20 +08:00
|
|
|
|
import pytest
|
|
|
|
|
@pytest.fixture
|
|
|
|
|
def myfixture(monkeypatch):
|
|
|
|
|
monkeypatch.setenv("HELLO", "WORLD")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-11-22 22:35:20 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
"""
|
2013-11-22 22:35:20 +08:00
|
|
|
|
>>> import os
|
|
|
|
|
>>> os.environ["HELLO"]
|
|
|
|
|
'WORLD'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run(p)
|
2013-11-22 22:35:20 +08:00
|
|
|
|
reprec.assertoutcome(passed=1)
|
2013-09-06 16:07:06 +08:00
|
|
|
|
|
2013-03-21 08:03:59 +08:00
|
|
|
|
def test_doctestmodule_with_fixtures(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2013-03-21 00:32:48 +08:00
|
|
|
|
'''
|
2013-03-21 19:04:14 +08:00
|
|
|
|
>>> dir = getfixture('tmpdir')
|
2013-03-21 00:32:48 +08:00
|
|
|
|
>>> type(dir).__name__
|
|
|
|
|
'LocalPath'
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-03-21 00:32:48 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
|
|
|
|
|
def test_doctestmodule_three_tests(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2013-05-17 23:18:22 +08:00
|
|
|
|
'''
|
|
|
|
|
>>> dir = getfixture('tmpdir')
|
|
|
|
|
>>> type(dir).__name__
|
|
|
|
|
'LocalPath'
|
|
|
|
|
'''
|
|
|
|
|
def my_func():
|
|
|
|
|
'''
|
|
|
|
|
>>> magic = 42
|
|
|
|
|
>>> magic - 42
|
|
|
|
|
0
|
|
|
|
|
'''
|
|
|
|
|
def unuseful():
|
|
|
|
|
pass
|
|
|
|
|
def another():
|
|
|
|
|
'''
|
|
|
|
|
>>> import os
|
|
|
|
|
>>> os is os
|
|
|
|
|
True
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=3)
|
|
|
|
|
|
|
|
|
|
def test_doctestmodule_two_tests_one_fail(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2017-02-17 02:41:51 +08:00
|
|
|
|
class MyClass(object):
|
2013-05-17 23:18:22 +08:00
|
|
|
|
def bad_meth(self):
|
|
|
|
|
'''
|
|
|
|
|
>>> magic = 42
|
|
|
|
|
>>> magic
|
|
|
|
|
0
|
|
|
|
|
'''
|
|
|
|
|
def nice_meth(self):
|
|
|
|
|
'''
|
|
|
|
|
>>> magic = 42
|
|
|
|
|
>>> magic - 42
|
|
|
|
|
0
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2013-05-17 23:18:22 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(failed=1, passed=1)
|
2014-10-08 20:31:17 +08:00
|
|
|
|
|
|
|
|
|
def test_ignored_whitespace(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2014-10-08 20:31:17 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ELLIPSIS NORMALIZE_WHITESPACE
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2017-02-17 02:41:51 +08:00
|
|
|
|
class MyClass(object):
|
2014-10-08 20:31:17 +08:00
|
|
|
|
'''
|
|
|
|
|
>>> a = "foo "
|
|
|
|
|
>>> print(a)
|
|
|
|
|
foo
|
|
|
|
|
'''
|
|
|
|
|
pass
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2014-10-08 20:31:17 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
|
|
|
|
def test_non_ignored_whitespace(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2014-10-08 20:31:17 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ELLIPSIS
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2017-02-17 02:41:51 +08:00
|
|
|
|
class MyClass(object):
|
2014-10-08 20:31:17 +08:00
|
|
|
|
'''
|
|
|
|
|
>>> a = "foo "
|
|
|
|
|
>>> print(a)
|
|
|
|
|
foo
|
|
|
|
|
'''
|
|
|
|
|
pass
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2014-10-08 20:31:17 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(failed=1, passed=0)
|
2014-10-08 21:54:08 +08:00
|
|
|
|
|
|
|
|
|
def test_ignored_whitespace_glob(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2014-10-08 21:54:08 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ELLIPSIS NORMALIZE_WHITESPACE
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
xdoc="""
|
2014-10-08 21:54:08 +08:00
|
|
|
|
>>> a = "foo "
|
|
|
|
|
>>> print(a)
|
|
|
|
|
foo
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2014-10-08 21:54:08 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-glob=x*.txt")
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
|
|
|
|
def test_non_ignored_whitespace_glob(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2014-10-08 21:54:08 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ELLIPSIS
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
xdoc="""
|
2014-10-08 21:54:08 +08:00
|
|
|
|
>>> a = "foo "
|
|
|
|
|
>>> print(a)
|
|
|
|
|
foo
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2014-10-08 21:54:08 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-glob=x*.txt")
|
|
|
|
|
reprec.assertoutcome(failed=1, passed=0)
|
2015-02-27 01:39:36 +08:00
|
|
|
|
|
2016-01-09 08:19:37 +08:00
|
|
|
|
def test_contains_unicode(self, testdir):
|
|
|
|
|
"""Fix internal error with docstrings containing non-ascii characters.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makepyfile(
|
2019-06-03 06:40:34 +08:00
|
|
|
|
'''\
|
2016-01-09 08:19:37 +08:00
|
|
|
|
def foo():
|
|
|
|
|
"""
|
|
|
|
|
>>> name = 'с' # not letter 'c' but instead Cyrillic 's'.
|
|
|
|
|
'anything'
|
|
|
|
|
"""
|
2019-06-03 06:40:34 +08:00
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(["Got nothing", "* 1 failed in*"])
|
2016-01-09 08:19:37 +08:00
|
|
|
|
|
2015-02-27 01:39:36 +08:00
|
|
|
|
def test_ignore_import_errors_on_doctest(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2015-02-27 01:39:36 +08:00
|
|
|
|
import asdf
|
|
|
|
|
|
|
|
|
|
def add_one(x):
|
|
|
|
|
'''
|
|
|
|
|
>>> add_one(1)
|
|
|
|
|
2
|
|
|
|
|
'''
|
|
|
|
|
return x + 1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-02-27 01:39:36 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
reprec = testdir.inline_run(
|
|
|
|
|
p, "--doctest-modules", "--doctest-ignore-import-errors"
|
|
|
|
|
)
|
2015-02-27 01:39:36 +08:00
|
|
|
|
reprec.assertoutcome(skipped=1, failed=1, passed=0)
|
2015-06-17 04:35:31 +08:00
|
|
|
|
|
|
|
|
|
def test_junit_report_for_doctest(self, testdir):
|
2015-06-17 07:42:49 +08:00
|
|
|
|
"""
|
|
|
|
|
#713: Fix --junit-xml option when used with --doctest-modules.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2015-06-17 04:35:31 +08:00
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> 1 + 1
|
|
|
|
|
3
|
|
|
|
|
'''
|
|
|
|
|
pass
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules", "--junit-xml=junit.xml")
|
2015-06-17 04:35:31 +08:00
|
|
|
|
reprec.assertoutcome(failed=1)
|
2015-07-13 10:43:33 +08:00
|
|
|
|
|
2017-05-26 04:08:32 +08:00
|
|
|
|
def test_unicode_doctest(self, testdir):
|
|
|
|
|
"""
|
|
|
|
|
Test case for issue 2434: DecodeError on Python 2 when doctest contains non-ascii
|
|
|
|
|
characters.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
test_unicode_doctest="""
|
2017-05-26 04:08:32 +08:00
|
|
|
|
.. doctest::
|
|
|
|
|
|
|
|
|
|
>>> print(
|
|
|
|
|
... "Hi\\n\\nByé")
|
|
|
|
|
Hi
|
|
|
|
|
...
|
|
|
|
|
Byé
|
|
|
|
|
>>> 1/0 # Byé
|
|
|
|
|
1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2017-05-26 04:08:32 +08:00
|
|
|
|
result = testdir.runpytest(p)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
["*UNEXPECTED EXCEPTION: ZeroDivisionError*", "*1 failed*"]
|
|
|
|
|
)
|
2017-05-26 04:08:32 +08:00
|
|
|
|
|
2017-06-14 06:34:05 +08:00
|
|
|
|
def test_unicode_doctest_module(self, testdir):
|
|
|
|
|
"""
|
|
|
|
|
Test case for issue 2434: DecodeError on Python 2 when doctest docstring
|
|
|
|
|
contains non-ascii characters.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
test_unicode_doctest_module="""
|
2017-06-14 06:34:05 +08:00
|
|
|
|
def fix_bad_unicode(text):
|
|
|
|
|
'''
|
|
|
|
|
>>> print(fix_bad_unicode('único'))
|
|
|
|
|
único
|
|
|
|
|
'''
|
|
|
|
|
return "único"
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest(p, "--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(["* 1 passed *"])
|
2017-06-14 06:34:05 +08:00
|
|
|
|
|
2018-06-15 04:00:38 +08:00
|
|
|
|
def test_print_unicode_value(self, testdir):
|
|
|
|
|
"""
|
|
|
|
|
Test case for issue 3583: Printing Unicode in doctest under Python 2.7
|
|
|
|
|
doesn't work
|
|
|
|
|
"""
|
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
test_print_unicode_value=r"""
|
|
|
|
|
Here is a doctest::
|
|
|
|
|
|
2019-06-05 08:48:06 +08:00
|
|
|
|
>>> print('\xE5\xE9\xEE\xF8\xFC')
|
2018-06-15 04:00:38 +08:00
|
|
|
|
åéîøü
|
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest(p)
|
|
|
|
|
result.stdout.fnmatch_lines(["* 1 passed *"])
|
|
|
|
|
|
2017-07-25 00:07:45 +08:00
|
|
|
|
def test_reportinfo(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
2017-07-25 00:07:45 +08:00
|
|
|
|
Test case to make sure that DoctestItem.reportinfo() returns lineno.
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
test_reportinfo="""
|
2017-07-25 00:07:45 +08:00
|
|
|
|
def foo(x):
|
|
|
|
|
'''
|
|
|
|
|
>>> foo('a')
|
|
|
|
|
'b'
|
|
|
|
|
'''
|
|
|
|
|
return 'c'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
items, reprec = testdir.inline_genitems(p, "--doctest-modules")
|
2017-07-25 00:07:45 +08:00
|
|
|
|
reportinfo = items[0].reportinfo()
|
|
|
|
|
assert reportinfo[1] == 1
|
|
|
|
|
|
2017-10-14 00:47:02 +08:00
|
|
|
|
def test_valid_setup_py(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
2017-10-14 00:47:02 +08:00
|
|
|
|
Test to make sure that pytest ignores valid setup.py files when ran
|
|
|
|
|
with --doctest-modules
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
setup="""
|
2017-10-14 00:47:02 +08:00
|
|
|
|
from setuptools import setup, find_packages
|
|
|
|
|
setup(name='sample',
|
|
|
|
|
version='0.0',
|
|
|
|
|
description='description',
|
|
|
|
|
packages=find_packages()
|
|
|
|
|
)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest(p, "--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(["*collected 0 items*"])
|
2017-10-14 00:47:02 +08:00
|
|
|
|
|
|
|
|
|
def test_invalid_setup_py(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
2017-10-14 00:47:02 +08:00
|
|
|
|
Test to make sure that pytest reads setup.py files that are not used
|
|
|
|
|
for python packages when ran with --doctest-modules
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
setup="""
|
2017-10-14 00:47:02 +08:00
|
|
|
|
def test_foo():
|
|
|
|
|
return 'bar'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest(p, "--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(["*collected 1 item*"])
|
2017-10-14 00:47:02 +08:00
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestLiterals:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("config_mode", ["ini", "comment"])
|
2015-08-13 08:41:31 +08:00
|
|
|
|
def test_allow_unicode(self, testdir, config_mode):
|
|
|
|
|
"""Test that doctests which output unicode work in all python versions
|
|
|
|
|
tested by pytest when the ALLOW_UNICODE option is used (either in
|
|
|
|
|
the ini file or by an inline comment).
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
if config_mode == "ini":
|
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2015-08-13 08:41:31 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ALLOW_UNICODE
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
comment = ""
|
2015-08-13 08:41:31 +08:00
|
|
|
|
else:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
comment = "#doctest: +ALLOW_UNICODE"
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-08-13 08:41:31 +08:00
|
|
|
|
>>> b'12'.decode('ascii') {comment}
|
|
|
|
|
'12'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
comment=comment
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
foo="""
|
2015-08-13 08:41:31 +08:00
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> b'12'.decode('ascii') {comment}
|
|
|
|
|
'12'
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
comment=comment
|
|
|
|
|
)
|
|
|
|
|
)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=2)
|
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("config_mode", ["ini", "comment"])
|
2015-12-30 06:55:19 +08:00
|
|
|
|
def test_allow_bytes(self, testdir, config_mode):
|
|
|
|
|
"""Test that doctests which output bytes work in all python versions
|
|
|
|
|
tested by pytest when the ALLOW_BYTES option is used (either in
|
|
|
|
|
the ini file or by an inline comment)(#1287).
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
if config_mode == "ini":
|
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
2015-12-30 06:55:19 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = ALLOW_BYTES
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
comment = ""
|
2015-12-30 06:55:19 +08:00
|
|
|
|
else:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
comment = "#doctest: +ALLOW_BYTES"
|
2015-12-30 06:55:19 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-12-30 06:55:19 +08:00
|
|
|
|
>>> b'foo' {comment}
|
|
|
|
|
'foo'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
comment=comment
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
foo="""
|
2015-12-30 06:55:19 +08:00
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> b'foo' {comment}
|
|
|
|
|
'foo'
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
comment=comment
|
|
|
|
|
)
|
|
|
|
|
)
|
2015-12-30 06:55:19 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=2)
|
|
|
|
|
|
2015-08-13 09:46:13 +08:00
|
|
|
|
def test_unicode_string(self, testdir):
|
2015-08-13 08:41:31 +08:00
|
|
|
|
"""Test that doctests which output unicode fail in Python 2 when
|
2015-08-13 09:46:13 +08:00
|
|
|
|
the ALLOW_UNICODE option is not used. The same test should pass
|
|
|
|
|
in Python 3.
|
2015-08-13 08:41:31 +08:00
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-08-13 09:46:13 +08:00
|
|
|
|
>>> b'12'.decode('ascii')
|
2015-08-13 08:41:31 +08:00
|
|
|
|
'12'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
reprec = testdir.inline_run()
|
2019-05-28 07:31:52 +08:00
|
|
|
|
reprec.assertoutcome(passed=1)
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
2015-12-30 06:55:19 +08:00
|
|
|
|
def test_bytes_literal(self, testdir):
|
|
|
|
|
"""Test that doctests which output bytes fail in Python 3 when
|
2019-05-28 07:31:52 +08:00
|
|
|
|
the ALLOW_BYTES option is not used. (#1287).
|
2015-12-30 06:55:19 +08:00
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-12-30 06:55:19 +08:00
|
|
|
|
>>> b'foo'
|
|
|
|
|
'foo'
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-12-30 06:55:19 +08:00
|
|
|
|
reprec = testdir.inline_run()
|
2019-05-28 07:31:52 +08:00
|
|
|
|
reprec.assertoutcome(failed=1)
|
2015-12-30 06:55:19 +08:00
|
|
|
|
|
2019-08-30 15:35:08 +08:00
|
|
|
|
def test_number_re(self) -> None:
|
|
|
|
|
_number_re = _get_checker()._number_re # type: ignore
|
2019-07-08 22:41:33 +08:00
|
|
|
|
for s in [
|
|
|
|
|
"1.",
|
|
|
|
|
"+1.",
|
|
|
|
|
"-1.",
|
|
|
|
|
".1",
|
|
|
|
|
"+.1",
|
|
|
|
|
"-.1",
|
|
|
|
|
"0.1",
|
|
|
|
|
"+0.1",
|
|
|
|
|
"-0.1",
|
|
|
|
|
"1e5",
|
|
|
|
|
"+1e5",
|
|
|
|
|
"1e+5",
|
|
|
|
|
"+1e+5",
|
|
|
|
|
"1e-5",
|
|
|
|
|
"+1e-5",
|
|
|
|
|
"-1e-5",
|
|
|
|
|
"1.2e3",
|
|
|
|
|
"-1.2e-3",
|
|
|
|
|
]:
|
|
|
|
|
print(s)
|
2019-08-30 15:35:08 +08:00
|
|
|
|
m = _number_re.match(s)
|
2019-07-08 22:41:33 +08:00
|
|
|
|
assert m is not None
|
|
|
|
|
assert float(m.group()) == pytest.approx(float(s))
|
|
|
|
|
for s in ["1", "abc"]:
|
|
|
|
|
print(s)
|
2019-08-30 15:35:08 +08:00
|
|
|
|
assert _number_re.match(s) is None
|
2019-07-08 22:41:33 +08:00
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("config_mode", ["ini", "comment"])
|
|
|
|
|
def test_number_precision(self, testdir, config_mode):
|
|
|
|
|
"""Test the NUMBER option."""
|
|
|
|
|
if config_mode == "ini":
|
|
|
|
|
testdir.makeini(
|
|
|
|
|
"""
|
|
|
|
|
[pytest]
|
|
|
|
|
doctest_optionflags = NUMBER
|
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
comment = ""
|
|
|
|
|
else:
|
|
|
|
|
comment = "#doctest: +NUMBER"
|
|
|
|
|
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
|
|
|
|
|
|
|
|
|
Scalars:
|
|
|
|
|
|
|
|
|
|
>>> import math
|
|
|
|
|
>>> math.pi {comment}
|
|
|
|
|
3.141592653589793
|
|
|
|
|
>>> math.pi {comment}
|
|
|
|
|
3.1416
|
|
|
|
|
>>> math.pi {comment}
|
|
|
|
|
3.14
|
|
|
|
|
>>> -math.pi {comment}
|
|
|
|
|
-3.14
|
|
|
|
|
>>> math.pi {comment}
|
|
|
|
|
3.
|
|
|
|
|
>>> 3. {comment}
|
|
|
|
|
3.0
|
|
|
|
|
>>> 3. {comment}
|
|
|
|
|
3.
|
|
|
|
|
>>> 3. {comment}
|
|
|
|
|
3.01
|
|
|
|
|
>>> 3. {comment}
|
|
|
|
|
2.99
|
|
|
|
|
>>> .299 {comment}
|
|
|
|
|
.3
|
|
|
|
|
>>> .301 {comment}
|
|
|
|
|
.3
|
|
|
|
|
>>> 951. {comment}
|
|
|
|
|
1e3
|
|
|
|
|
>>> 1049. {comment}
|
|
|
|
|
1e3
|
|
|
|
|
>>> -1049. {comment}
|
|
|
|
|
-1e3
|
|
|
|
|
>>> 1e3 {comment}
|
|
|
|
|
1e3
|
|
|
|
|
>>> 1e3 {comment}
|
|
|
|
|
1000.
|
|
|
|
|
|
|
|
|
|
Lists:
|
|
|
|
|
|
|
|
|
|
>>> [3.1415, 0.097, 13.1, 7, 8.22222e5, 0.598e-2] {comment}
|
|
|
|
|
[3.14, 0.1, 13., 7, 8.22e5, 6.0e-3]
|
|
|
|
|
>>> [[0.333, 0.667], [0.999, 1.333]] {comment}
|
|
|
|
|
[[0.33, 0.667], [0.999, 1.333]]
|
|
|
|
|
>>> [[[0.101]]] {comment}
|
|
|
|
|
[[[0.1]]]
|
|
|
|
|
|
|
|
|
|
Doesn't barf on non-numbers:
|
|
|
|
|
|
|
|
|
|
>>> 'abc' {comment}
|
|
|
|
|
'abc'
|
|
|
|
|
>>> None {comment}
|
|
|
|
|
""".format(
|
|
|
|
|
comment=comment
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run()
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
|
"expression,output",
|
|
|
|
|
[
|
|
|
|
|
# ints shouldn't match floats:
|
|
|
|
|
("3.0", "3"),
|
|
|
|
|
("3e0", "3"),
|
|
|
|
|
("1e3", "1000"),
|
|
|
|
|
("3", "3.0"),
|
|
|
|
|
# Rounding:
|
|
|
|
|
("3.1", "3.0"),
|
|
|
|
|
("3.1", "3.2"),
|
|
|
|
|
("3.1", "4.0"),
|
|
|
|
|
("8.22e5", "810000.0"),
|
|
|
|
|
# Only the actual output is rounded up, not the expected output:
|
2019-07-11 16:57:44 +08:00
|
|
|
|
("3.0", "2.98"),
|
2019-07-08 22:41:33 +08:00
|
|
|
|
("1e3", "999"),
|
2019-07-11 17:04:43 +08:00
|
|
|
|
# The current implementation doesn't understand that numbers inside
|
|
|
|
|
# strings shouldn't be treated as numbers:
|
|
|
|
|
pytest.param("'3.1416'", "'3.14'", marks=pytest.mark.xfail),
|
2019-07-08 22:41:33 +08:00
|
|
|
|
],
|
|
|
|
|
)
|
|
|
|
|
def test_number_non_matches(self, testdir, expression, output):
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
|
|
|
|
>>> {expression} #doctest: +NUMBER
|
|
|
|
|
{output}
|
2019-07-11 16:57:44 +08:00
|
|
|
|
""".format(
|
|
|
|
|
expression=expression, output=output
|
|
|
|
|
)
|
2019-07-08 22:41:33 +08:00
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run()
|
|
|
|
|
reprec.assertoutcome(passed=0, failed=1)
|
|
|
|
|
|
|
|
|
|
def test_number_and_allow_unicode(self, testdir):
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
|
|
|
|
>>> from collections import namedtuple
|
|
|
|
|
>>> T = namedtuple('T', 'a b c')
|
|
|
|
|
>>> T(a=0.2330000001, b=u'str', c=b'bytes') # doctest: +ALLOW_UNICODE, +ALLOW_BYTES, +NUMBER
|
|
|
|
|
T(a=0.233, b=u'str', c='bytes')
|
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
reprec = testdir.inline_run()
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
2015-08-13 08:41:31 +08:00
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestDoctestSkips:
|
2015-08-26 10:12:02 +08:00
|
|
|
|
"""
|
|
|
|
|
If all examples in a doctest are skipped due to the SKIP option, then
|
|
|
|
|
the tests should be SKIPPED rather than PASSED. (#957)
|
|
|
|
|
"""
|
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.fixture(params=["text", "module"])
|
2015-08-26 10:12:02 +08:00
|
|
|
|
def makedoctest(self, testdir, request):
|
|
|
|
|
def makeit(doctest):
|
|
|
|
|
mode = request.param
|
2018-05-23 22:48:46 +08:00
|
|
|
|
if mode == "text":
|
2015-08-26 10:12:02 +08:00
|
|
|
|
testdir.maketxtfile(doctest)
|
|
|
|
|
else:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
assert mode == "module"
|
2015-08-26 10:12:02 +08:00
|
|
|
|
testdir.makepyfile('"""\n%s"""' % doctest)
|
|
|
|
|
|
|
|
|
|
return makeit
|
|
|
|
|
|
|
|
|
|
def test_one_skipped(self, testdir, makedoctest):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
makedoctest(
|
|
|
|
|
"""
|
2015-08-26 10:12:02 +08:00
|
|
|
|
>>> 1 + 1 # doctest: +SKIP
|
|
|
|
|
2
|
|
|
|
|
>>> 2 + 2
|
|
|
|
|
4
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-08-26 10:12:02 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
|
|
|
|
def test_one_skipped_failed(self, testdir, makedoctest):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
makedoctest(
|
|
|
|
|
"""
|
2015-08-26 10:12:02 +08:00
|
|
|
|
>>> 1 + 1 # doctest: +SKIP
|
|
|
|
|
2
|
|
|
|
|
>>> 2 + 2
|
|
|
|
|
200
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-08-26 10:12:02 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(failed=1)
|
|
|
|
|
|
|
|
|
|
def test_all_skipped(self, testdir, makedoctest):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
makedoctest(
|
|
|
|
|
"""
|
2015-08-26 10:12:02 +08:00
|
|
|
|
>>> 1 + 1 # doctest: +SKIP
|
|
|
|
|
2
|
|
|
|
|
>>> 2 + 2 # doctest: +SKIP
|
|
|
|
|
200
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-08-26 10:12:02 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(skipped=1)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
2016-06-02 00:16:05 +08:00
|
|
|
|
def test_vacuous_all_skipped(self, testdir, makedoctest):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
makedoctest("")
|
2016-06-02 00:16:05 +08:00
|
|
|
|
reprec = testdir.inline_run("--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=0, skipped=0)
|
|
|
|
|
|
2018-02-23 11:00:54 +08:00
|
|
|
|
def test_continue_on_failure(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_something="""
|
2018-02-23 11:00:54 +08:00
|
|
|
|
>>> i = 5
|
|
|
|
|
>>> def foo():
|
|
|
|
|
... raise ValueError('error1')
|
|
|
|
|
>>> foo()
|
|
|
|
|
>>> i
|
|
|
|
|
>>> i + 2
|
|
|
|
|
7
|
|
|
|
|
>>> i + 1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2018-02-24 12:31:11 +08:00
|
|
|
|
result = testdir.runpytest("--doctest-modules", "--doctest-continue-on-failure")
|
2018-02-23 11:00:54 +08:00
|
|
|
|
result.assert_outcomes(passed=0, failed=1)
|
2018-02-24 11:20:14 +08:00
|
|
|
|
# The lines that contains the failure are 4, 5, and 8. The first one
|
|
|
|
|
# is a stack trace and the other two are mismatches.
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
["*4: UnexpectedException*", "*5: DocTestFailure*", "*8: DocTestFailure*"]
|
|
|
|
|
)
|
2018-02-23 11:00:54 +08:00
|
|
|
|
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestDoctestAutoUseFixtures:
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
SCOPES = ["module", "session", "class", "function"]
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
|
|
|
|
def test_doctest_module_session_fixture(self, testdir):
|
|
|
|
|
"""Test that session fixtures are initialized for doctest modules (#768)
|
|
|
|
|
"""
|
|
|
|
|
# session fixture which changes some global data, which will
|
|
|
|
|
# be accessed by doctests in a module
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
import pytest
|
|
|
|
|
import sys
|
|
|
|
|
|
|
|
|
|
@pytest.yield_fixture(autouse=True, scope='session')
|
|
|
|
|
def myfixture():
|
|
|
|
|
assert not hasattr(sys, 'pytest_session_data')
|
|
|
|
|
sys.pytest_session_data = 1
|
|
|
|
|
yield
|
|
|
|
|
del sys.pytest_session_data
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
foo="""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
import sys
|
|
|
|
|
|
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> assert sys.pytest_session_data == 1
|
|
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
def bar():
|
|
|
|
|
'''
|
|
|
|
|
>>> assert sys.pytest_session_data == 1
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2019-03-23 18:36:18 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*2 passed*"])
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("scope", SCOPES)
|
|
|
|
|
@pytest.mark.parametrize("enable_doctest", [True, False])
|
2015-10-02 09:59:37 +08:00
|
|
|
|
def test_fixture_scopes(self, testdir, scope, enable_doctest):
|
|
|
|
|
"""Test that auto-use fixtures work properly with doctest modules.
|
|
|
|
|
See #1057 and #1100.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="{scope}")
|
|
|
|
|
def auto(request):
|
|
|
|
|
return 99
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
scope=scope
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
test_1='''
|
2015-10-02 09:59:37 +08:00
|
|
|
|
def test_foo():
|
|
|
|
|
"""
|
|
|
|
|
>>> getfixture('auto') + 1
|
|
|
|
|
100
|
|
|
|
|
"""
|
|
|
|
|
def test_bar():
|
|
|
|
|
assert 1
|
2018-05-23 22:48:46 +08:00
|
|
|
|
'''
|
|
|
|
|
)
|
|
|
|
|
params = ("--doctest-modules",) if enable_doctest else ()
|
2015-10-02 09:59:37 +08:00
|
|
|
|
passes = 3 if enable_doctest else 2
|
|
|
|
|
result = testdir.runpytest(*params)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*=== %d passed in *" % passes])
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("scope", SCOPES)
|
|
|
|
|
@pytest.mark.parametrize("autouse", [True, False])
|
|
|
|
|
@pytest.mark.parametrize("use_fixture_in_doctest", [True, False])
|
|
|
|
|
def test_fixture_module_doctest_scopes(
|
|
|
|
|
self, testdir, scope, autouse, use_fixture_in_doctest
|
|
|
|
|
):
|
2015-10-02 09:59:37 +08:00
|
|
|
|
"""Test that auto-use fixtures work properly with doctest files.
|
|
|
|
|
See #1057 and #1100.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse={autouse}, scope="{scope}")
|
|
|
|
|
def auto(request):
|
|
|
|
|
return 99
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
scope=scope, autouse=autouse
|
|
|
|
|
)
|
|
|
|
|
)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
if use_fixture_in_doctest:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
>>> getfixture('auto')
|
|
|
|
|
99
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
else:
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
>>> 1 + 1
|
|
|
|
|
2
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2019-10-06 01:18:51 +08:00
|
|
|
|
result.stdout.no_fnmatch_line("*FAILURES*")
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*=== 1 passed in *"])
|
2015-10-02 09:59:37 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("scope", SCOPES)
|
2015-10-02 09:59:37 +08:00
|
|
|
|
def test_auto_use_request_attributes(self, testdir, scope):
|
|
|
|
|
"""Check that all attributes of a request in an autouse fixture
|
|
|
|
|
behave as expected when requested for a doctest item.
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="{scope}")
|
|
|
|
|
def auto(request):
|
|
|
|
|
if "{scope}" == 'module':
|
|
|
|
|
assert request.module is None
|
|
|
|
|
if "{scope}" == 'class':
|
|
|
|
|
assert request.cls is None
|
|
|
|
|
if "{scope}" == 'function':
|
|
|
|
|
assert request.function is None
|
|
|
|
|
return 99
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
scope=scope
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
testdir.maketxtfile(
|
|
|
|
|
test_doc="""
|
2015-10-02 09:59:37 +08:00
|
|
|
|
>>> 1 + 1
|
|
|
|
|
2
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
2019-10-06 01:18:51 +08:00
|
|
|
|
str(result.stdout.no_fnmatch_line("*FAILURES*"))
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(["*=== 1 passed in *"])
|
2016-03-02 20:43:57 +08:00
|
|
|
|
|
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestDoctestNamespaceFixture:
|
2016-03-02 20:43:57 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
SCOPES = ["module", "session", "class", "function"]
|
2016-03-02 20:43:57 +08:00
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("scope", SCOPES)
|
2016-03-02 20:43:57 +08:00
|
|
|
|
def test_namespace_doctestfile(self, testdir, scope):
|
|
|
|
|
"""
|
|
|
|
|
Check that inserting something into the namespace works in a
|
|
|
|
|
simple text file doctest
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2016-03-02 20:43:57 +08:00
|
|
|
|
import pytest
|
|
|
|
|
import contextlib
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="{scope}")
|
|
|
|
|
def add_contextlib(doctest_namespace):
|
|
|
|
|
doctest_namespace['cl'] = contextlib
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
scope=scope
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
p = testdir.maketxtfile(
|
|
|
|
|
"""
|
2016-03-02 20:43:57 +08:00
|
|
|
|
>>> print(cl.__name__)
|
|
|
|
|
contextlib
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2016-03-02 20:43:57 +08:00
|
|
|
|
reprec = testdir.inline_run(p)
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("scope", SCOPES)
|
2016-03-02 20:43:57 +08:00
|
|
|
|
def test_namespace_pyfile(self, testdir, scope):
|
|
|
|
|
"""
|
|
|
|
|
Check that inserting something into the namespace works in a
|
|
|
|
|
simple Python file docstring doctest
|
|
|
|
|
"""
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makeconftest(
|
|
|
|
|
"""
|
2016-03-02 20:43:57 +08:00
|
|
|
|
import pytest
|
|
|
|
|
import contextlib
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(autouse=True, scope="{scope}")
|
|
|
|
|
def add_contextlib(doctest_namespace):
|
|
|
|
|
doctest_namespace['cl'] = contextlib
|
2018-05-23 22:48:46 +08:00
|
|
|
|
""".format(
|
|
|
|
|
scope=scope
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
p = testdir.makepyfile(
|
|
|
|
|
"""
|
2016-03-02 20:43:57 +08:00
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> print(cl.__name__)
|
|
|
|
|
contextlib
|
|
|
|
|
'''
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2016-03-02 20:43:57 +08:00
|
|
|
|
reprec = testdir.inline_run(p, "--doctest-modules")
|
|
|
|
|
reprec.assertoutcome(passed=1)
|
2016-07-23 21:16:23 +08:00
|
|
|
|
|
|
|
|
|
|
2019-06-03 06:32:00 +08:00
|
|
|
|
class TestDoctestReportingOption:
|
2016-07-23 21:16:23 +08:00
|
|
|
|
def _run_doctest_report(self, testdir, format):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
"""
|
2016-07-23 21:16:23 +08:00
|
|
|
|
def foo():
|
|
|
|
|
'''
|
|
|
|
|
>>> foo()
|
|
|
|
|
a b
|
|
|
|
|
0 1 4
|
|
|
|
|
1 2 4
|
|
|
|
|
2 3 6
|
|
|
|
|
'''
|
|
|
|
|
print(' a b\\n'
|
|
|
|
|
'0 1 4\\n'
|
|
|
|
|
'1 2 5\\n'
|
|
|
|
|
'2 3 6')
|
2018-05-23 22:48:46 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2016-07-23 21:16:23 +08:00
|
|
|
|
return testdir.runpytest("--doctest-modules", "--doctest-report", format)
|
|
|
|
|
|
2018-05-23 22:48:46 +08:00
|
|
|
|
@pytest.mark.parametrize("format", ["udiff", "UDIFF", "uDiFf"])
|
2016-07-23 21:21:59 +08:00
|
|
|
|
def test_doctest_report_udiff(self, testdir, format):
|
2016-07-23 21:16:23 +08:00
|
|
|
|
result = self._run_doctest_report(testdir, format)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[" 0 1 4", " -1 2 4", " +1 2 5", " 2 3 6"]
|
|
|
|
|
)
|
2016-07-23 21:16:23 +08:00
|
|
|
|
|
|
|
|
|
def test_doctest_report_cdiff(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result = self._run_doctest_report(testdir, "cdiff")
|
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
" a b",
|
|
|
|
|
" 0 1 4",
|
|
|
|
|
" ! 1 2 4",
|
|
|
|
|
" 2 3 6",
|
|
|
|
|
" --- 1,4 ----",
|
|
|
|
|
" a b",
|
|
|
|
|
" 0 1 4",
|
|
|
|
|
" ! 1 2 5",
|
|
|
|
|
" 2 3 6",
|
|
|
|
|
]
|
|
|
|
|
)
|
2016-07-23 21:16:23 +08:00
|
|
|
|
|
|
|
|
|
def test_doctest_report_ndiff(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result = self._run_doctest_report(testdir, "ndiff")
|
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
" a b",
|
|
|
|
|
" 0 1 4",
|
|
|
|
|
" - 1 2 4",
|
|
|
|
|
" ? ^",
|
|
|
|
|
" + 1 2 5",
|
|
|
|
|
" ? ^",
|
|
|
|
|
" 2 3 6",
|
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("format", ["none", "only_first_failure"])
|
2016-07-23 21:58:13 +08:00
|
|
|
|
def test_doctest_report_none_or_only_first_failure(self, testdir, format):
|
|
|
|
|
result = self._run_doctest_report(testdir, format)
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result.stdout.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"Expected:",
|
|
|
|
|
" a b",
|
|
|
|
|
" 0 1 4",
|
|
|
|
|
" 1 2 4",
|
|
|
|
|
" 2 3 6",
|
|
|
|
|
"Got:",
|
|
|
|
|
" a b",
|
|
|
|
|
" 0 1 4",
|
|
|
|
|
" 1 2 5",
|
|
|
|
|
" 2 3 6",
|
|
|
|
|
]
|
|
|
|
|
)
|
2016-07-23 21:16:23 +08:00
|
|
|
|
|
|
|
|
|
def test_doctest_report_invalid(self, testdir):
|
2018-05-23 22:48:46 +08:00
|
|
|
|
result = self._run_doctest_report(testdir, "obviously_invalid_format")
|
|
|
|
|
result.stderr.fnmatch_lines(
|
|
|
|
|
[
|
|
|
|
|
"*error: argument --doctest-report: invalid choice: 'obviously_invalid_format' (choose from*"
|
|
|
|
|
]
|
|
|
|
|
)
|
2018-10-22 22:30:25 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("mock_module", ["mock", "unittest.mock"])
|
|
|
|
|
def test_doctest_mock_objects_dont_recurse_missbehaved(mock_module, testdir):
|
|
|
|
|
pytest.importorskip(mock_module)
|
|
|
|
|
testdir.makepyfile(
|
|
|
|
|
"""
|
|
|
|
|
from {mock_module} import call
|
|
|
|
|
class Example(object):
|
|
|
|
|
'''
|
|
|
|
|
>>> 1 + 1
|
|
|
|
|
2
|
|
|
|
|
'''
|
|
|
|
|
""".format(
|
|
|
|
|
mock_module=mock_module
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
result = testdir.runpytest("--doctest-modules")
|
|
|
|
|
result.stdout.fnmatch_lines(["* 1 passed *"])
|
2019-06-08 10:39:59 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Broken:
|
|
|
|
|
def __getattr__(self, _):
|
|
|
|
|
raise KeyError("This should be an AttributeError")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize( # pragma: no branch (lambdas are not called)
|
|
|
|
|
"stop", [None, _is_mocked, lambda f: None, lambda f: False, lambda f: True]
|
|
|
|
|
)
|
|
|
|
|
def test_warning_on_unwrap_of_broken_object(stop):
|
|
|
|
|
bad_instance = Broken()
|
|
|
|
|
assert inspect.unwrap.__module__ == "inspect"
|
|
|
|
|
with _patch_unwrap_mock_aware():
|
|
|
|
|
assert inspect.unwrap.__module__ != "inspect"
|
|
|
|
|
with pytest.warns(
|
|
|
|
|
pytest.PytestWarning, match="^Got KeyError.* when unwrapping"
|
|
|
|
|
):
|
|
|
|
|
with pytest.raises(KeyError):
|
|
|
|
|
inspect.unwrap(bad_instance, stop=stop)
|
|
|
|
|
assert inspect.unwrap.__module__ == "inspect"
|