|
|
#!/usr/bin/env python
|
|
|
"""IPython-enhanced doctest module with unittest integration.
|
|
|
|
|
|
This module is heavily based on the standard library's doctest module, but
|
|
|
enhances it with IPython support. This enables docstrings to contain
|
|
|
unmodified IPython input and output pasted from real IPython sessions.
|
|
|
|
|
|
It should be possible to use this module as a drop-in replacement for doctest
|
|
|
whenever you wish to use IPython input.
|
|
|
|
|
|
Since the module absorbs all normal doctest functionality, you can use a mix of
|
|
|
both plain Python and IPython examples in any given module, though not in the
|
|
|
same docstring.
|
|
|
|
|
|
See a simple example at the bottom of this code which serves as self-test and
|
|
|
demonstration code. Simply run this file (use -v for details) to run the
|
|
|
tests.
|
|
|
|
|
|
This module also contains routines to ease the integration of doctests with
|
|
|
regular unittest-based testing. In particular, see the DocTestLoader class and
|
|
|
the makeTestSuite utility function.
|
|
|
|
|
|
|
|
|
Limitations:
|
|
|
|
|
|
- When generating examples for use as doctests, make sure that you have
|
|
|
pretty-printing OFF. This can be done either by starting ipython with the
|
|
|
flag '--nopprint', by setting pprint to 0 in your ipythonrc file, or by
|
|
|
interactively disabling it with %Pprint. This is required so that IPython
|
|
|
output matches that of normal Python, which is used by doctest for internal
|
|
|
execution.
|
|
|
|
|
|
- Do not rely on specific prompt numbers for results (such as using
|
|
|
'_34==True', for example). For IPython tests run via an external process
|
|
|
the prompt numbers may be different, and IPython tests run as normal python
|
|
|
code won't even have these special _NN variables set at all.
|
|
|
|
|
|
- IPython functions that produce output as a side-effect of calling a system
|
|
|
process (e.g. 'ls') can be doc-tested, but they must be handled in an
|
|
|
external IPython process. Such doctests must be tagged with:
|
|
|
|
|
|
# ipdoctest: EXTERNAL
|
|
|
|
|
|
so that the testing machinery handles them differently. Since these are run
|
|
|
via pexpect in an external process, they can't deal with exceptions or other
|
|
|
fancy featurs of regular doctests. You must limit such tests to simple
|
|
|
matching of the output. For this reason, I recommend you limit these kinds
|
|
|
of doctests to features that truly require a separate process, and use the
|
|
|
normal IPython ones (which have all the features of normal doctests) for
|
|
|
everything else. See the examples at the bottom of this file for a
|
|
|
comparison of what can be done with both types.
|
|
|
"""
|
|
|
|
|
|
# Standard library imports
|
|
|
import __builtin__
|
|
|
import doctest
|
|
|
import inspect
|
|
|
import os
|
|
|
import re
|
|
|
import sys
|
|
|
import unittest
|
|
|
|
|
|
from doctest import *
|
|
|
|
|
|
# Our own imports
|
|
|
from IPython.tools import utils
|
|
|
|
|
|
###########################################################################
|
|
|
#
|
|
|
# We must start our own ipython object and heavily muck with it so that all the
|
|
|
# modifications IPython makes to system behavior don't send the doctest
|
|
|
# machinery into a fit. This code should be considered a gross hack, but it
|
|
|
# gets the job done.
|
|
|
|
|
|
import IPython
|
|
|
|
|
|
# Hack to restore __main__, which ipython modifies upon startup
|
|
|
_main = sys.modules.get('__main__')
|
|
|
ipython = IPython.Shell.IPShell(['--classic','--noterm_title']).IP
|
|
|
sys.modules['__main__'] = _main
|
|
|
|
|
|
# Deactivate the various python system hooks added by ipython for
|
|
|
# interactive convenience so we don't confuse the doctest system
|
|
|
sys.displayhook = sys.__displayhook__
|
|
|
sys.excepthook = sys.__excepthook__
|
|
|
|
|
|
# So that ipython magics and aliases can be doctested
|
|
|
__builtin__._ip = IPython.ipapi.get()
|
|
|
|
|
|
# for debugging only!!!
|
|
|
#from IPython.Shell import IPShellEmbed;ipshell=IPShellEmbed(['--noterm_title']) # dbg
|
|
|
|
|
|
|
|
|
# runner
|
|
|
from IPython.irunner import IPythonRunner
|
|
|
iprunner = IPythonRunner(echo=False)
|
|
|
|
|
|
###########################################################################
|
|
|
|
|
|
# A simple subclassing of the original with a different class name, so we can
|
|
|
# distinguish and treat differently IPython examples from pure python ones.
|
|
|
class IPExample(doctest.Example): pass
|
|
|
|
|
|
class IPExternalExample(doctest.Example):
|
|
|
"""Doctest examples to be run in an external process."""
|
|
|
|
|
|
def __init__(self, source, want, exc_msg=None, lineno=0, indent=0,
|
|
|
options=None):
|
|
|
# Parent constructor
|
|
|
doctest.Example.__init__(self,source,want,exc_msg,lineno,indent,options)
|
|
|
|
|
|
# An EXTRA newline is needed to prevent pexpect hangs
|
|
|
self.source += '\n'
|
|
|
|
|
|
class IPDocTestParser(doctest.DocTestParser):
|
|
|
"""
|
|
|
A class used to parse strings containing doctest examples.
|
|
|
|
|
|
Note: This is a version modified to properly recognize IPython input and
|
|
|
convert any IPython examples into valid Python ones.
|
|
|
"""
|
|
|
# This regular expression is used to find doctest examples in a
|
|
|
# string. It defines three groups: `source` is the source code
|
|
|
# (including leading indentation and prompts); `indent` is the
|
|
|
# indentation of the first (PS1) line of the source code; and
|
|
|
# `want` is the expected output (including leading indentation).
|
|
|
|
|
|
# Classic Python prompts or default IPython ones
|
|
|
_PS1_PY = r'>>>'
|
|
|
_PS2_PY = r'\.\.\.'
|
|
|
|
|
|
_PS1_IP = r'In\ \[\d+\]:'
|
|
|
_PS2_IP = r'\ \ \ \.\.\.+:'
|
|
|
|
|
|
_RE_TPL = r'''
|
|
|
# Source consists of a PS1 line followed by zero or more PS2 lines.
|
|
|
(?P<source>
|
|
|
(?:^(?P<indent> [ ]*) (?P<ps1> %s) .*) # PS1 line
|
|
|
(?:\n [ ]* (?P<ps2> %s) .*)*) # PS2 lines
|
|
|
\n? # a newline
|
|
|
# Want consists of any non-blank lines that do not start with PS1.
|
|
|
(?P<want> (?:(?![ ]*$) # Not a blank line
|
|
|
(?![ ]*%s) # Not a line starting with PS1
|
|
|
(?![ ]*%s) # Not a line starting with PS2
|
|
|
.*$\n? # But any other line
|
|
|
)*)
|
|
|
'''
|
|
|
|
|
|
_EXAMPLE_RE_PY = re.compile( _RE_TPL % (_PS1_PY,_PS2_PY,_PS1_PY,_PS2_PY),
|
|
|
re.MULTILINE | re.VERBOSE)
|
|
|
|
|
|
_EXAMPLE_RE_IP = re.compile( _RE_TPL % (_PS1_IP,_PS2_IP,_PS1_IP,_PS2_IP),
|
|
|
re.MULTILINE | re.VERBOSE)
|
|
|
|
|
|
def ip2py(self,source):
|
|
|
"""Convert input IPython source into valid Python."""
|
|
|
out = []
|
|
|
newline = out.append
|
|
|
for line in source.splitlines():
|
|
|
newline(ipython.prefilter(line,True))
|
|
|
newline('') # ensure a closing newline, needed by doctest
|
|
|
return '\n'.join(out)
|
|
|
|
|
|
def parse(self, string, name='<string>'):
|
|
|
"""
|
|
|
Divide the given string into examples and intervening text,
|
|
|
and return them as a list of alternating Examples and strings.
|
|
|
Line numbers for the Examples are 0-based. The optional
|
|
|
argument `name` is a name identifying this string, and is only
|
|
|
used for error messages.
|
|
|
"""
|
|
|
string = string.expandtabs()
|
|
|
# If all lines begin with the same indentation, then strip it.
|
|
|
min_indent = self._min_indent(string)
|
|
|
if min_indent > 0:
|
|
|
string = '\n'.join([l[min_indent:] for l in string.split('\n')])
|
|
|
|
|
|
output = []
|
|
|
charno, lineno = 0, 0
|
|
|
|
|
|
# Whether to convert the input from ipython to python syntax
|
|
|
ip2py = False
|
|
|
# Find all doctest examples in the string. First, try them as Python
|
|
|
# examples, then as IPython ones
|
|
|
terms = list(self._EXAMPLE_RE_PY.finditer(string))
|
|
|
if terms:
|
|
|
# Normal Python example
|
|
|
Example = doctest.Example
|
|
|
else:
|
|
|
# It's an ipython example. Note that IPExamples are run
|
|
|
# in-process, so their syntax must be turned into valid python.
|
|
|
# IPExternalExamples are run out-of-process (via pexpect) so they
|
|
|
# don't need any filtering (a real ipython will be executing them).
|
|
|
terms = list(self._EXAMPLE_RE_IP.finditer(string))
|
|
|
if re.search(r'#\s*ipdoctest:\s*EXTERNAL',string):
|
|
|
#print '-'*70 # dbg
|
|
|
#print 'IPExternalExample, Source:\n',string # dbg
|
|
|
#print '-'*70 # dbg
|
|
|
Example = IPExternalExample
|
|
|
else:
|
|
|
#print '-'*70 # dbg
|
|
|
#print 'IPExample, Source:\n',string # dbg
|
|
|
#print '-'*70 # dbg
|
|
|
Example = IPExample
|
|
|
ip2py = True
|
|
|
|
|
|
for m in terms:
|
|
|
# Add the pre-example text to `output`.
|
|
|
output.append(string[charno:m.start()])
|
|
|
# Update lineno (lines before this example)
|
|
|
lineno += string.count('\n', charno, m.start())
|
|
|
# Extract info from the regexp match.
|
|
|
(source, options, want, exc_msg) = \
|
|
|
self._parse_example(m, name, lineno,ip2py)
|
|
|
if Example is IPExternalExample:
|
|
|
options[doctest.NORMALIZE_WHITESPACE] = True
|
|
|
# Create an Example, and add it to the list.
|
|
|
if not self._IS_BLANK_OR_COMMENT(source):
|
|
|
output.append(Example(source, want, exc_msg,
|
|
|
lineno=lineno,
|
|
|
indent=min_indent+len(m.group('indent')),
|
|
|
options=options))
|
|
|
# Update lineno (lines inside this example)
|
|
|
lineno += string.count('\n', m.start(), m.end())
|
|
|
# Update charno.
|
|
|
charno = m.end()
|
|
|
# Add any remaining post-example text to `output`.
|
|
|
output.append(string[charno:])
|
|
|
|
|
|
return output
|
|
|
|
|
|
def _parse_example(self, m, name, lineno,ip2py=False):
|
|
|
"""
|
|
|
Given a regular expression match from `_EXAMPLE_RE` (`m`),
|
|
|
return a pair `(source, want)`, where `source` is the matched
|
|
|
example's source code (with prompts and indentation stripped);
|
|
|
and `want` is the example's expected output (with indentation
|
|
|
stripped).
|
|
|
|
|
|
`name` is the string's name, and `lineno` is the line number
|
|
|
where the example starts; both are used for error messages.
|
|
|
|
|
|
Optional:
|
|
|
`ip2py`: if true, filter the input via IPython to convert the syntax
|
|
|
into valid python.
|
|
|
"""
|
|
|
|
|
|
# Get the example's indentation level.
|
|
|
indent = len(m.group('indent'))
|
|
|
|
|
|
# Divide source into lines; check that they're properly
|
|
|
# indented; and then strip their indentation & prompts.
|
|
|
source_lines = m.group('source').split('\n')
|
|
|
|
|
|
# We're using variable-length input prompts
|
|
|
ps1 = m.group('ps1')
|
|
|
ps2 = m.group('ps2')
|
|
|
ps1_len = len(ps1)
|
|
|
|
|
|
self._check_prompt_blank(source_lines, indent, name, lineno,ps1_len)
|
|
|
if ps2:
|
|
|
self._check_prefix(source_lines[1:], ' '*indent + ps2, name, lineno)
|
|
|
|
|
|
source = '\n'.join([sl[indent+ps1_len+1:] for sl in source_lines])
|
|
|
|
|
|
if ip2py:
|
|
|
# Convert source input from IPython into valid Python syntax
|
|
|
source = self.ip2py(source)
|
|
|
|
|
|
# Divide want into lines; check that it's properly indented; and
|
|
|
# then strip the indentation. Spaces before the last newline should
|
|
|
# be preserved, so plain rstrip() isn't good enough.
|
|
|
want = m.group('want')
|
|
|
want_lines = want.split('\n')
|
|
|
if len(want_lines) > 1 and re.match(r' *$', want_lines[-1]):
|
|
|
del want_lines[-1] # forget final newline & spaces after it
|
|
|
self._check_prefix(want_lines, ' '*indent, name,
|
|
|
lineno + len(source_lines))
|
|
|
|
|
|
# Remove ipython output prompt that might be present in the first line
|
|
|
want_lines[0] = re.sub(r'Out\[\d+\]: \s*?\n?','',want_lines[0])
|
|
|
|
|
|
want = '\n'.join([wl[indent:] for wl in want_lines])
|
|
|
|
|
|
# If `want` contains a traceback message, then extract it.
|
|
|
m = self._EXCEPTION_RE.match(want)
|
|
|
if m:
|
|
|
exc_msg = m.group('msg')
|
|
|
else:
|
|
|
exc_msg = None
|
|
|
|
|
|
# Extract options from the source.
|
|
|
options = self._find_options(source, name, lineno)
|
|
|
|
|
|
return source, options, want, exc_msg
|
|
|
|
|
|
def _check_prompt_blank(self, lines, indent, name, lineno, ps1_len):
|
|
|
"""
|
|
|
Given the lines of a source string (including prompts and
|
|
|
leading indentation), check to make sure that every prompt is
|
|
|
followed by a space character. If any line is not followed by
|
|
|
a space character, then raise ValueError.
|
|
|
|
|
|
Note: IPython-modified version which takes the input prompt length as a
|
|
|
parameter, so that prompts of variable length can be dealt with.
|
|
|
"""
|
|
|
space_idx = indent+ps1_len
|
|
|
min_len = space_idx+1
|
|
|
for i, line in enumerate(lines):
|
|
|
if len(line) >= min_len and line[space_idx] != ' ':
|
|
|
raise ValueError('line %r of the docstring for %s '
|
|
|
'lacks blank after %s: %r' %
|
|
|
(lineno+i+1, name,
|
|
|
line[indent:space_idx], line))
|
|
|
|
|
|
|
|
|
SKIP = register_optionflag('SKIP')
|
|
|
|
|
|
class IPDocTestRunner(doctest.DocTestRunner):
|
|
|
"""Modified DocTestRunner which can also run IPython tests.
|
|
|
|
|
|
This runner is capable of handling IPython doctests that require
|
|
|
out-of-process output capture (such as system calls via !cmd or aliases).
|
|
|
Note however that because these tests are run in a separate process, many
|
|
|
of doctest's fancier capabilities (such as detailed exception analysis) are
|
|
|
not available. So try to limit such tests to simple cases of matching
|
|
|
actual output.
|
|
|
"""
|
|
|
|
|
|
#/////////////////////////////////////////////////////////////////
|
|
|
# DocTest Running
|
|
|
#/////////////////////////////////////////////////////////////////
|
|
|
|
|
|
def _run_iptest(self, test, out):
|
|
|
"""
|
|
|
Run the examples in `test`. Write the outcome of each example with one
|
|
|
of the `DocTestRunner.report_*` methods, using the writer function
|
|
|
`out`. Return a tuple `(f, t)`, where `t` is the number of examples
|
|
|
tried, and `f` is the number of examples that failed. The examples are
|
|
|
run in the namespace `test.globs`.
|
|
|
|
|
|
IPython note: this is a modified version of the original __run()
|
|
|
private method to handle out-of-process examples.
|
|
|
"""
|
|
|
|
|
|
if out is None:
|
|
|
out = sys.stdout.write
|
|
|
|
|
|
# Keep track of the number of failures and tries.
|
|
|
failures = tries = 0
|
|
|
|
|
|
# Save the option flags (since option directives can be used
|
|
|
# to modify them).
|
|
|
original_optionflags = self.optionflags
|
|
|
|
|
|
SUCCESS, FAILURE, BOOM = range(3) # `outcome` state
|
|
|
|
|
|
check = self._checker.check_output
|
|
|
|
|
|
# Process each example.
|
|
|
for examplenum, example in enumerate(test.examples):
|
|
|
|
|
|
# If REPORT_ONLY_FIRST_FAILURE is set, then supress
|
|
|
# reporting after the first failure.
|
|
|
quiet = (self.optionflags & REPORT_ONLY_FIRST_FAILURE and
|
|
|
failures > 0)
|
|
|
|
|
|
# Merge in the example's options.
|
|
|
self.optionflags = original_optionflags
|
|
|
if example.options:
|
|
|
for (optionflag, val) in example.options.items():
|
|
|
if val:
|
|
|
self.optionflags |= optionflag
|
|
|
else:
|
|
|
self.optionflags &= ~optionflag
|
|
|
|
|
|
# If 'SKIP' is set, then skip this example.
|
|
|
if self.optionflags & SKIP:
|
|
|
continue
|
|
|
|
|
|
# Record that we started this example.
|
|
|
tries += 1
|
|
|
if not quiet:
|
|
|
self.report_start(out, test, example)
|
|
|
|
|
|
# Run the example in the given context (globs), and record
|
|
|
# any exception that gets raised. (But don't intercept
|
|
|
# keyboard interrupts.)
|
|
|
try:
|
|
|
# Don't blink! This is where the user's code gets run.
|
|
|
got = ''
|
|
|
# The code is run in an external process
|
|
|
got = iprunner.run_source(example.source,get_output=True)
|
|
|
except KeyboardInterrupt:
|
|
|
raise
|
|
|
except:
|
|
|
self.debugger.set_continue() # ==== Example Finished ====
|
|
|
|
|
|
outcome = FAILURE # guilty until proved innocent or insane
|
|
|
|
|
|
if check(example.want, got, self.optionflags):
|
|
|
outcome = SUCCESS
|
|
|
|
|
|
# Report the outcome.
|
|
|
if outcome is SUCCESS:
|
|
|
if not quiet:
|
|
|
self.report_success(out, test, example, got)
|
|
|
elif outcome is FAILURE:
|
|
|
if not quiet:
|
|
|
self.report_failure(out, test, example, got)
|
|
|
failures += 1
|
|
|
elif outcome is BOOM:
|
|
|
if not quiet:
|
|
|
self.report_unexpected_exception(out, test, example,
|
|
|
exc_info)
|
|
|
failures += 1
|
|
|
else:
|
|
|
assert False, ("unknown outcome", outcome)
|
|
|
|
|
|
# Restore the option flags (in case they were modified)
|
|
|
self.optionflags = original_optionflags
|
|
|
|
|
|
# Record and return the number of failures and tries.
|
|
|
|
|
|
# Hack to access a parent private method by working around Python's
|
|
|
# name mangling (which is fortunately simple).
|
|
|
doctest.DocTestRunner._DocTestRunner__record_outcome(self,test,
|
|
|
failures, tries)
|
|
|
return failures, tries
|
|
|
|
|
|
def run(self, test, compileflags=None, out=None, clear_globs=True):
|
|
|
"""Run examples in `test`.
|
|
|
|
|
|
This method will defer to the parent for normal Python examples, but it
|
|
|
will run IPython ones via pexpect.
|
|
|
"""
|
|
|
if not test.examples:
|
|
|
return
|
|
|
|
|
|
if isinstance(test.examples[0],IPExternalExample):
|
|
|
self._run_iptest(test,out)
|
|
|
else:
|
|
|
DocTestRunner.run(self,test,compileflags,out,clear_globs)
|
|
|
|
|
|
|
|
|
class IPDebugRunner(IPDocTestRunner,doctest.DebugRunner):
|
|
|
"""IPython-modified DebugRunner, see the original class for details."""
|
|
|
|
|
|
def run(self, test, compileflags=None, out=None, clear_globs=True):
|
|
|
r = IPDocTestRunner.run(self, test, compileflags, out, False)
|
|
|
if clear_globs:
|
|
|
test.globs.clear()
|
|
|
return r
|
|
|
|
|
|
|
|
|
class IPDocTestLoader(unittest.TestLoader):
|
|
|
"""A test loader with IPython-enhanced doctest support.
|
|
|
|
|
|
Instances of this loader will automatically add doctests found in a module
|
|
|
to the test suite returned by the loadTestsFromModule method. In
|
|
|
addition, at initialization time a string of doctests can be given to the
|
|
|
loader, enabling it to add doctests to a module which didn't have them in
|
|
|
its docstring, coming from an external source."""
|
|
|
|
|
|
|
|
|
def __init__(self,dt_files=None,dt_modules=None,test_finder=None):
|
|
|
"""Initialize the test loader.
|
|
|
|
|
|
:Keywords:
|
|
|
|
|
|
dt_files : list (None)
|
|
|
List of names of files to be executed as doctests.
|
|
|
|
|
|
dt_modules : list (None)
|
|
|
List of module names to be scanned for doctests in their
|
|
|
docstrings.
|
|
|
|
|
|
test_finder : instance (None)
|
|
|
Instance of a testfinder (see doctest for details).
|
|
|
"""
|
|
|
|
|
|
if dt_files is None: dt_files = []
|
|
|
if dt_modules is None: dt_modules = []
|
|
|
self.dt_files = utils.list_strings(dt_files)
|
|
|
self.dt_modules = utils.list_strings(dt_modules)
|
|
|
if test_finder is None:
|
|
|
test_finder = doctest.DocTestFinder(parser=IPDocTestParser())
|
|
|
self.test_finder = test_finder
|
|
|
|
|
|
def loadTestsFromModule(self, module):
|
|
|
"""Return a suite of all tests cases contained in the given module.
|
|
|
|
|
|
If the loader was initialized with a doctests argument, then this
|
|
|
string is assigned as the module's docstring."""
|
|
|
|
|
|
# Start by loading any tests in the called module itself
|
|
|
suite = super(self.__class__,self).loadTestsFromModule(module)
|
|
|
|
|
|
# Now, load also tests referenced at construction time as companion
|
|
|
# doctests that reside in standalone files
|
|
|
for fname in self.dt_files:
|
|
|
#print 'mod:',module # dbg
|
|
|
#print 'fname:',fname # dbg
|
|
|
#suite.addTest(doctest.DocFileSuite(fname))
|
|
|
suite.addTest(doctest.DocFileSuite(fname,module_relative=False))
|
|
|
# Add docstring tests from module, if given at construction time
|
|
|
for mod in self.dt_modules:
|
|
|
suite.addTest(doctest.DocTestSuite(mod,
|
|
|
test_finder=self.test_finder))
|
|
|
|
|
|
#ipshell() # dbg
|
|
|
return suite
|
|
|
|
|
|
def my_import(name):
|
|
|
"""Module importer - taken from the python documentation.
|
|
|
|
|
|
This function allows importing names with dots in them."""
|
|
|
|
|
|
mod = __import__(name)
|
|
|
components = name.split('.')
|
|
|
for comp in components[1:]:
|
|
|
mod = getattr(mod, comp)
|
|
|
return mod
|
|
|
|
|
|
def makeTestSuite(module_name,dt_files=None,dt_modules=None,idt=True):
|
|
|
"""Make a TestSuite object for a given module, specified by name.
|
|
|
|
|
|
This extracts all the doctests associated with a module using a
|
|
|
DocTestLoader object.
|
|
|
|
|
|
:Parameters:
|
|
|
|
|
|
- module_name: a string containing the name of a module with unittests.
|
|
|
|
|
|
:Keywords:
|
|
|
|
|
|
dt_files : list of strings
|
|
|
List of names of plain text files to be treated as doctests.
|
|
|
|
|
|
dt_modules : list of strings
|
|
|
List of names of modules to be scanned for doctests in docstrings.
|
|
|
|
|
|
idt : bool (True)
|
|
|
If True, return integrated doctests. This means that each filename
|
|
|
listed in dt_files is turned into a *single* unittest, suitable for
|
|
|
running via unittest's runner or Twisted's Trial runner. If false, the
|
|
|
dt_files parameter is returned unmodified, so that other test runners
|
|
|
(such as oilrun) can run the doctests with finer granularity.
|
|
|
"""
|
|
|
|
|
|
mod = my_import(module_name)
|
|
|
if idt:
|
|
|
suite = IPDocTestLoader(dt_files,dt_modules).loadTestsFromModule(mod)
|
|
|
else:
|
|
|
suite = IPDocTestLoader(None,dt_modules).loadTestsFromModule(mod)
|
|
|
|
|
|
if idt:
|
|
|
return suite
|
|
|
else:
|
|
|
return suite,dt_files
|
|
|
|
|
|
# Copied from doctest in py2.5 and modified for our purposes (since they don't
|
|
|
# parametrize what we need)
|
|
|
|
|
|
# For backward compatibility, a global instance of a DocTestRunner
|
|
|
# class, updated by testmod.
|
|
|
master = None
|
|
|
|
|
|
def testmod(m=None, name=None, globs=None, verbose=None,
|
|
|
report=True, optionflags=0, extraglobs=None,
|
|
|
raise_on_error=False, exclude_empty=False):
|
|
|
"""m=None, name=None, globs=None, verbose=None, report=True,
|
|
|
optionflags=0, extraglobs=None, raise_on_error=False,
|
|
|
exclude_empty=False
|
|
|
|
|
|
Note: IPython-modified version which loads test finder and runners that
|
|
|
recognize IPython syntax in doctests.
|
|
|
|
|
|
Test examples in docstrings in functions and classes reachable
|
|
|
from module m (or the current module if m is not supplied), starting
|
|
|
with m.__doc__.
|
|
|
|
|
|
Also test examples reachable from dict m.__test__ if it exists and is
|
|
|
not None. m.__test__ maps names to functions, classes and strings;
|
|
|
function and class docstrings are tested even if the name is private;
|
|
|
strings are tested directly, as if they were docstrings.
|
|
|
|
|
|
Return (#failures, #tests).
|
|
|
|
|
|
See doctest.__doc__ for an overview.
|
|
|
|
|
|
Optional keyword arg "name" gives the name of the module; by default
|
|
|
use m.__name__.
|
|
|
|
|
|
Optional keyword arg "globs" gives a dict to be used as the globals
|
|
|
when executing examples; by default, use m.__dict__. A copy of this
|
|
|
dict is actually used for each docstring, so that each docstring's
|
|
|
examples start with a clean slate.
|
|
|
|
|
|
Optional keyword arg "extraglobs" gives a dictionary that should be
|
|
|
merged into the globals that are used to execute examples. By
|
|
|
default, no extra globals are used. This is new in 2.4.
|
|
|
|
|
|
Optional keyword arg "verbose" prints lots of stuff if true, prints
|
|
|
only failures if false; by default, it's true iff "-v" is in sys.argv.
|
|
|
|
|
|
Optional keyword arg "report" prints a summary at the end when true,
|
|
|
else prints nothing at the end. In verbose mode, the summary is
|
|
|
detailed, else very brief (in fact, empty if all tests passed).
|
|
|
|
|
|
Optional keyword arg "optionflags" or's together module constants,
|
|
|
and defaults to 0. This is new in 2.3. Possible values (see the
|
|
|
docs for details):
|
|
|
|
|
|
DONT_ACCEPT_TRUE_FOR_1
|
|
|
DONT_ACCEPT_BLANKLINE
|
|
|
NORMALIZE_WHITESPACE
|
|
|
ELLIPSIS
|
|
|
SKIP
|
|
|
IGNORE_EXCEPTION_DETAIL
|
|
|
REPORT_UDIFF
|
|
|
REPORT_CDIFF
|
|
|
REPORT_NDIFF
|
|
|
REPORT_ONLY_FIRST_FAILURE
|
|
|
|
|
|
Optional keyword arg "raise_on_error" raises an exception on the
|
|
|
first unexpected exception or failure. This allows failures to be
|
|
|
post-mortem debugged.
|
|
|
|
|
|
Advanced tomfoolery: testmod runs methods of a local instance of
|
|
|
class doctest.Tester, then merges the results into (or creates)
|
|
|
global Tester instance doctest.master. Methods of doctest.master
|
|
|
can be called directly too, if you want to do something unusual.
|
|
|
Passing report=0 to testmod is especially useful then, to delay
|
|
|
displaying a summary. Invoke doctest.master.summarize(verbose)
|
|
|
when you're done fiddling.
|
|
|
"""
|
|
|
global master
|
|
|
|
|
|
# If no module was given, then use __main__.
|
|
|
if m is None:
|
|
|
# DWA - m will still be None if this wasn't invoked from the command
|
|
|
# line, in which case the following TypeError is about as good an error
|
|
|
# as we should expect
|
|
|
m = sys.modules.get('__main__')
|
|
|
|
|
|
# Check that we were actually given a module.
|
|
|
if not inspect.ismodule(m):
|
|
|
raise TypeError("testmod: module required; %r" % (m,))
|
|
|
|
|
|
# If no name was given, then use the module's name.
|
|
|
if name is None:
|
|
|
name = m.__name__
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
# fperez - make IPython finder and runner:
|
|
|
# Find, parse, and run all tests in the given module.
|
|
|
finder = DocTestFinder(exclude_empty=exclude_empty,
|
|
|
parser=IPDocTestParser())
|
|
|
|
|
|
if raise_on_error:
|
|
|
runner = IPDebugRunner(verbose=verbose, optionflags=optionflags)
|
|
|
else:
|
|
|
runner = IPDocTestRunner(verbose=verbose, optionflags=optionflags,
|
|
|
#checker=IPOutputChecker() # dbg
|
|
|
)
|
|
|
|
|
|
# /fperez - end of ipython changes
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
|
for test in finder.find(m, name, globs=globs, extraglobs=extraglobs):
|
|
|
runner.run(test)
|
|
|
|
|
|
if report:
|
|
|
runner.summarize()
|
|
|
|
|
|
if master is None:
|
|
|
master = runner
|
|
|
else:
|
|
|
master.merge(runner)
|
|
|
|
|
|
return runner.failures, runner.tries
|
|
|
|
|
|
|
|
|
# Simple testing and example code
|
|
|
if __name__ == "__main__":
|
|
|
|
|
|
def ipfunc():
|
|
|
"""
|
|
|
Some ipython tests...
|
|
|
|
|
|
In [1]: import os
|
|
|
|
|
|
In [2]: cd /
|
|
|
/
|
|
|
|
|
|
In [3]: 2+3
|
|
|
Out[3]: 5
|
|
|
|
|
|
In [26]: for i in range(3):
|
|
|
....: print i,
|
|
|
....: print i+1,
|
|
|
....:
|
|
|
0 1 1 2 2 3
|
|
|
|
|
|
|
|
|
Examples that access the operating system work:
|
|
|
|
|
|
In [19]: cd /tmp
|
|
|
/tmp
|
|
|
|
|
|
In [20]: mkdir foo_ipython
|
|
|
|
|
|
In [21]: cd foo_ipython
|
|
|
/tmp/foo_ipython
|
|
|
|
|
|
In [23]: !touch bar baz
|
|
|
|
|
|
# We unfortunately can't just call 'ls' because its output is not
|
|
|
# seen by doctest, since it happens in a separate process
|
|
|
|
|
|
In [24]: os.listdir('.')
|
|
|
Out[24]: ['bar', 'baz']
|
|
|
|
|
|
In [25]: cd /tmp
|
|
|
/tmp
|
|
|
|
|
|
In [26]: rm -rf foo_ipython
|
|
|
|
|
|
|
|
|
It's OK to use '_' for the last result, but do NOT try to use IPython's
|
|
|
numbered history of _NN outputs, since those won't exist under the
|
|
|
doctest environment:
|
|
|
|
|
|
In [7]: 3+4
|
|
|
Out[7]: 7
|
|
|
|
|
|
In [8]: _+3
|
|
|
Out[8]: 10
|
|
|
"""
|
|
|
|
|
|
def ipfunc_external():
|
|
|
"""
|
|
|
Tests that must be run in an external process
|
|
|
|
|
|
|
|
|
# ipdoctest: EXTERNAL
|
|
|
|
|
|
In [11]: for i in range(10):
|
|
|
....: print i,
|
|
|
....: print i+1,
|
|
|
....:
|
|
|
0 1 1 2 2 3 3 4 4 5 5 6 6 7 7 8 8 9 9 10
|
|
|
|
|
|
|
|
|
In [1]: import os
|
|
|
|
|
|
In [1]: print "hello"
|
|
|
hello
|
|
|
|
|
|
In [19]: cd /tmp
|
|
|
/tmp
|
|
|
|
|
|
In [20]: mkdir foo_ipython2
|
|
|
|
|
|
In [21]: cd foo_ipython2
|
|
|
/tmp/foo_ipython2
|
|
|
|
|
|
In [23]: !touch bar baz
|
|
|
|
|
|
In [24]: ls
|
|
|
bar baz
|
|
|
|
|
|
In [24]: !ls
|
|
|
bar baz
|
|
|
|
|
|
In [25]: cd /tmp
|
|
|
/tmp
|
|
|
|
|
|
In [26]: rm -rf foo_ipython2
|
|
|
"""
|
|
|
|
|
|
def pyfunc():
|
|
|
"""
|
|
|
Some pure python tests...
|
|
|
|
|
|
>>> import os
|
|
|
|
|
|
>>> 2+3
|
|
|
5
|
|
|
|
|
|
>>> for i in range(3):
|
|
|
... print i,
|
|
|
... print i+1,
|
|
|
...
|
|
|
0 1 1 2 2 3
|
|
|
"""
|
|
|
|
|
|
# Call the global testmod() just like you would with normal doctest
|
|
|
testmod()
|
|
|
|