101 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			101 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Python
		
	
	
	
"""
 | 
						|
collect and execute doctests from modules and test files. 
 | 
						|
 | 
						|
Usage
 | 
						|
-------------
 | 
						|
 | 
						|
By default all files matching the ``test*.txt`` pattern will 
 | 
						|
be run through the python standard ``doctest`` module.  Issue::
 | 
						|
 | 
						|
    py.test --doctest-glob='*.rst'
 | 
						|
 | 
						|
to change the pattern.  Additionally you can trigger running of
 | 
						|
tests in all python modules (including regular python test modules)::
 | 
						|
 | 
						|
    py.test --doctest-modules
 | 
						|
 | 
						|
You can also make these changes permanent in your project by 
 | 
						|
putting them into a conftest.py file like this::
 | 
						|
 | 
						|
    # content of conftest.py 
 | 
						|
    option_doctestmodules = True
 | 
						|
    option_doctestglob = "*.rst"
 | 
						|
"""
 | 
						|
 | 
						|
import py
 | 
						|
from py.impl.code.code import TerminalRepr, ReprFileLocation
 | 
						|
import doctest
 | 
						|
 | 
						|
def pytest_addoption(parser):
 | 
						|
    group = parser.getgroup("collect")
 | 
						|
    group.addoption("--doctest-modules", 
 | 
						|
        action="store_true", default=False, 
 | 
						|
        help="run doctests in all .py modules",
 | 
						|
        dest="doctestmodules")
 | 
						|
    group.addoption("--doctest-glob",
 | 
						|
        action="store", default="test*.txt", metavar="pat",
 | 
						|
        help="doctests file matching pattern, default: test*.txt",
 | 
						|
        dest="doctestglob")
 | 
						|
 | 
						|
def pytest_collect_file(path, parent):
 | 
						|
    config = parent.config
 | 
						|
    if path.ext == ".py":
 | 
						|
        if config.getvalue("doctestmodules"):
 | 
						|
            return DoctestModule(path, parent)
 | 
						|
    elif path.check(fnmatch=config.getvalue("doctestglob")):
 | 
						|
        return DoctestTextfile(path, parent)
 | 
						|
 | 
						|
class ReprFailDoctest(TerminalRepr):
 | 
						|
    def __init__(self, reprlocation, lines):
 | 
						|
        self.reprlocation = reprlocation
 | 
						|
        self.lines = lines
 | 
						|
    def toterminal(self, tw):
 | 
						|
        for line in self.lines:
 | 
						|
            tw.line(line)
 | 
						|
        self.reprlocation.toterminal(tw)
 | 
						|
             
 | 
						|
class DoctestItem(py.test.collect.Item):
 | 
						|
    def __init__(self, path, parent):
 | 
						|
        name = self.__class__.__name__ + ":" + path.basename
 | 
						|
        super(DoctestItem, self).__init__(name=name, parent=parent)
 | 
						|
        self.fspath = path 
 | 
						|
 | 
						|
    def repr_failure(self, excinfo):
 | 
						|
        if excinfo.errisinstance(doctest.DocTestFailure):
 | 
						|
            doctestfailure = excinfo.value
 | 
						|
            example = doctestfailure.example
 | 
						|
            test = doctestfailure.test
 | 
						|
            filename = test.filename 
 | 
						|
            lineno = test.lineno + example.lineno + 1
 | 
						|
            message = excinfo.type.__name__
 | 
						|
            reprlocation = ReprFileLocation(filename, lineno, message)
 | 
						|
            checker = doctest.OutputChecker() 
 | 
						|
            REPORT_UDIFF = doctest.REPORT_UDIFF
 | 
						|
            filelines = py.path.local(filename).readlines(cr=0)
 | 
						|
            i = max(test.lineno, max(0, lineno - 10)) # XXX? 
 | 
						|
            lines = []
 | 
						|
            for line in filelines[i:lineno]:
 | 
						|
                lines.append("%03d %s" % (i+1, line))
 | 
						|
                i += 1
 | 
						|
            lines += checker.output_difference(example, 
 | 
						|
                    doctestfailure.got, REPORT_UDIFF).split("\n")
 | 
						|
            return ReprFailDoctest(reprlocation, lines)
 | 
						|
        elif excinfo.errisinstance(doctest.UnexpectedException):
 | 
						|
            excinfo = py.code.ExceptionInfo(excinfo.value.exc_info)
 | 
						|
            return super(DoctestItem, self).repr_failure(excinfo)
 | 
						|
        else: 
 | 
						|
            return super(DoctestItem, self).repr_failure(excinfo)
 | 
						|
 | 
						|
class DoctestTextfile(DoctestItem):
 | 
						|
    def runtest(self):
 | 
						|
        if not self._deprecated_testexecution():
 | 
						|
            failed, tot = doctest.testfile(
 | 
						|
                str(self.fspath), module_relative=False, 
 | 
						|
                raise_on_error=True, verbose=0)
 | 
						|
 | 
						|
class DoctestModule(DoctestItem):
 | 
						|
    def runtest(self):
 | 
						|
        module = self.fspath.pyimport()
 | 
						|
        failed, tot = doctest.testmod(
 | 
						|
            module, raise_on_error=True, verbose=0)
 |