Commit 5abd76a7 authored by R David Murray's avatar R David Murray

#14649: clarify DocTestSuite error when there are no docstrings.

Also adds tests to verify the documented behavior (which is probably a bug, as
indicated in the added comments).

Patch by Chris Jerdonek.
parent 01beb69c
...@@ -1024,6 +1024,16 @@ from text files and modules with doctests: ...@@ -1024,6 +1024,16 @@ from text files and modules with doctests:
This function uses the same search technique as :func:`testmod`. This function uses the same search technique as :func:`testmod`.
.. note::
Unlike :func:`testmod` and :class:`DocTestFinder`, this function raises
a :exc:`ValueError` if *module* contains no docstrings. You can prevent
this error by passing a :class:`DocTestFinder` instance as the
*test_finder* argument with its *exclude_empty* keyword argument set
to ``False``::
>>> finder = doctest.DocTestFinder(exclude_empty=False)
>>> suite = doctest.DocTestSuite(test_finder=finder)
Under the covers, :func:`DocTestSuite` creates a :class:`unittest.TestSuite` out Under the covers, :func:`DocTestSuite` creates a :class:`unittest.TestSuite` out
of :class:`doctest.DocTestCase` instances, and :class:`DocTestCase` is a of :class:`doctest.DocTestCase` instances, and :class:`DocTestCase` is a
......
...@@ -2333,7 +2333,12 @@ def DocTestSuite(module=None, globs=None, extraglobs=None, test_finder=None, ...@@ -2333,7 +2333,12 @@ def DocTestSuite(module=None, globs=None, extraglobs=None, test_finder=None,
elif not tests: elif not tests:
# Why do we want to do this? Because it reveals a bug that might # Why do we want to do this? Because it reveals a bug that might
# otherwise be hidden. # otherwise be hidden.
raise ValueError(module, "has no tests") # It is probably a bug that this exception is not also raised if the
# number of doctest examples in tests is zero (i.e. if no doctest
# examples were found). However, we should probably not be raising
# an exception at all here, though it is too late to make this change
# for a maintenance release. See also issue #14649.
raise ValueError(module, "has no docstrings")
tests.sort() tests.sort()
suite = unittest.TestSuite() suite = unittest.TestSuite()
......
# This is a sample module used for testing doctest.
#
# This module is for testing how doctest handles a module with no
# docstrings.
class Foo(object):
# A class with no docstring.
def __init__(self):
pass
"""This is a sample module used for testing doctest.
This module is for testing how doctest handles a module with docstrings
but no doctest examples.
"""
class Foo(object):
"""A docstring with no doctest examples.
"""
def __init__(self):
pass
...@@ -1984,6 +1984,31 @@ def test_DocTestSuite(): ...@@ -1984,6 +1984,31 @@ def test_DocTestSuite():
>>> suite.run(unittest.TestResult()) >>> suite.run(unittest.TestResult())
<unittest.result.TestResult run=9 errors=0 failures=4> <unittest.result.TestResult run=9 errors=0 failures=4>
The module need not contain any doctest examples:
>>> suite = doctest.DocTestSuite('test.sample_doctest_no_doctests')
>>> suite.run(unittest.TestResult())
<unittest.result.TestResult run=0 errors=0 failures=0>
However, if DocTestSuite finds no docstrings, it raises an error:
>>> try:
... doctest.DocTestSuite('test.sample_doctest_no_docstrings')
... except ValueError as e:
... error = e
>>> print(error.args[1])
has no docstrings
You can prevent this error by passing a DocTestFinder instance with
the `exclude_empty` keyword argument set to False:
>>> finder = doctest.DocTestFinder(exclude_empty=False)
>>> suite = doctest.DocTestSuite('test.sample_doctest_no_docstrings',
... test_finder=finder)
>>> suite.run(unittest.TestResult())
<unittest.result.TestResult run=0 errors=0 failures=0>
We can use the current module: We can use the current module:
>>> suite = test.sample_doctest.test_suite() >>> suite = test.sample_doctest.test_suite()
......
...@@ -29,7 +29,8 @@ verbose = test.support.verbose ...@@ -29,7 +29,8 @@ verbose = test.support.verbose
# test_cmd_line_script (covers the zipimport support in runpy) # test_cmd_line_script (covers the zipimport support in runpy)
# Retrieve some helpers from other test cases # Retrieve some helpers from other test cases
from test import test_doctest, sample_doctest from test import (test_doctest, sample_doctest, sample_doctest_no_doctests,
sample_doctest_no_docstrings)
def _run_object_doctest(obj, module): def _run_object_doctest(obj, module):
...@@ -105,16 +106,26 @@ class ZipSupportTests(unittest.TestCase): ...@@ -105,16 +106,26 @@ class ZipSupportTests(unittest.TestCase):
"test_zipped_doctest") "test_zipped_doctest")
test_src = test_src.replace("test.sample_doctest", test_src = test_src.replace("test.sample_doctest",
"sample_zipped_doctest") "sample_zipped_doctest")
sample_src = inspect.getsource(sample_doctest) # The sample doctest files rewritten to include in the zipped version.
sample_src = sample_src.replace("test.test_doctest", sample_sources = {}
"test_zipped_doctest") for mod in [sample_doctest, sample_doctest_no_doctests,
sample_doctest_no_docstrings]:
src = inspect.getsource(mod)
src = src.replace("test.test_doctest", "test_zipped_doctest")
# Rewrite the module name so that, for example,
# "test.sample_doctest" becomes "sample_zipped_doctest".
mod_name = mod.__name__.split(".")[-1]
mod_name = mod_name.replace("sample_", "sample_zipped_")
sample_sources[mod_name] = src
with temp_dir() as d: with temp_dir() as d:
script_name = make_script(d, 'test_zipped_doctest', script_name = make_script(d, 'test_zipped_doctest',
test_src) test_src)
zip_name, run_name = make_zip_script(d, 'test_zip', zip_name, run_name = make_zip_script(d, 'test_zip',
script_name) script_name)
z = zipfile.ZipFile(zip_name, 'a') z = zipfile.ZipFile(zip_name, 'a')
z.writestr("sample_zipped_doctest.py", sample_src) for mod_name, src in sample_sources.items():
z.writestr(mod_name + ".py", src)
z.close() z.close()
if verbose: if verbose:
zip_file = zipfile.ZipFile(zip_name, 'r') zip_file = zipfile.ZipFile(zip_name, 'r')
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment