1187 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			1187 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Python
		
	
	
	
| """(disabled by default) support for testing pytest and pytest plugins."""
 | |
| from __future__ import absolute_import, division, print_function
 | |
| 
 | |
| import codecs
 | |
| import gc
 | |
| import os
 | |
| import platform
 | |
| import re
 | |
| import subprocess
 | |
| import six
 | |
| import sys
 | |
| import time
 | |
| import traceback
 | |
| from fnmatch import fnmatch
 | |
| 
 | |
| from weakref import WeakKeyDictionary
 | |
| 
 | |
| from _pytest.capture import MultiCapture, SysCapture
 | |
| from _pytest._code import Source
 | |
| import py
 | |
| import pytest
 | |
| from _pytest.main import Session, EXIT_OK
 | |
| from _pytest.assertion.rewrite import AssertionRewritingHook
 | |
| 
 | |
| 
 | |
| PYTEST_FULLPATH = os.path.abspath(pytest.__file__.rstrip("oc")).replace("$py.class", ".py")
 | |
| 
 | |
| 
 | |
| IGNORE_PAM = [  # filenames added when obtaining details about the current user
 | |
|      u'/var/lib/sss/mc/passwd'
 | |
| ]
 | |
| 
 | |
| 
 | |
| def pytest_addoption(parser):
 | |
|     parser.addoption('--lsof',
 | |
|                      action="store_true", dest="lsof", default=False,
 | |
|                      help=("run FD checks if lsof is available"))
 | |
| 
 | |
|     parser.addoption('--runpytest', default="inprocess", dest="runpytest",
 | |
|                      choices=("inprocess", "subprocess"),
 | |
|                      help=("run pytest sub runs in tests using an 'inprocess' "
 | |
|                            "or 'subprocess' (python -m main) method"))
 | |
| 
 | |
| 
 | |
| def pytest_configure(config):
 | |
|     if config.getvalue("lsof"):
 | |
|         checker = LsofFdLeakChecker()
 | |
|         if checker.matching_platform():
 | |
|             config.pluginmanager.register(checker)
 | |
| 
 | |
| 
 | |
| class LsofFdLeakChecker(object):
 | |
|     def get_open_files(self):
 | |
|         out = self._exec_lsof()
 | |
|         open_files = self._parse_lsof_output(out)
 | |
|         return open_files
 | |
| 
 | |
|     def _exec_lsof(self):
 | |
|         pid = os.getpid()
 | |
|         return py.process.cmdexec("lsof -Ffn0 -p %d" % pid)
 | |
| 
 | |
|     def _parse_lsof_output(self, out):
 | |
|         def isopen(line):
 | |
|             return line.startswith('f') and ("deleted" not in line and
 | |
|                                              'mem' not in line and "txt" not in line and 'cwd' not in line)
 | |
| 
 | |
|         open_files = []
 | |
| 
 | |
|         for line in out.split("\n"):
 | |
|             if isopen(line):
 | |
|                 fields = line.split('\0')
 | |
|                 fd = fields[0][1:]
 | |
|                 filename = fields[1][1:]
 | |
|                 if filename in IGNORE_PAM:
 | |
|                     continue
 | |
|                 if filename.startswith('/'):
 | |
|                     open_files.append((fd, filename))
 | |
| 
 | |
|         return open_files
 | |
| 
 | |
|     def matching_platform(self):
 | |
|         try:
 | |
|             py.process.cmdexec("lsof -v")
 | |
|         except (py.process.cmdexec.Error, UnicodeDecodeError):
 | |
|             # cmdexec may raise UnicodeDecodeError on Windows systems with
 | |
|             # locale other than English:
 | |
|             # https://bitbucket.org/pytest-dev/py/issues/66
 | |
|             return False
 | |
|         else:
 | |
|             return True
 | |
| 
 | |
|     @pytest.hookimpl(hookwrapper=True, tryfirst=True)
 | |
|     def pytest_runtest_protocol(self, item):
 | |
|         lines1 = self.get_open_files()
 | |
|         yield
 | |
|         if hasattr(sys, "pypy_version_info"):
 | |
|             gc.collect()
 | |
|         lines2 = self.get_open_files()
 | |
| 
 | |
|         new_fds = set([t[0] for t in lines2]) - set([t[0] for t in lines1])
 | |
|         leaked_files = [t for t in lines2 if t[0] in new_fds]
 | |
|         if leaked_files:
 | |
|             error = []
 | |
|             error.append("***** %s FD leakage detected" % len(leaked_files))
 | |
|             error.extend([str(f) for f in leaked_files])
 | |
|             error.append("*** Before:")
 | |
|             error.extend([str(f) for f in lines1])
 | |
|             error.append("*** After:")
 | |
|             error.extend([str(f) for f in lines2])
 | |
|             error.append(error[0])
 | |
|             error.append("*** function %s:%s: %s " % item.location)
 | |
|             error.append("See issue #2366")
 | |
|             item.warn('', "\n".join(error))
 | |
| 
 | |
| 
 | |
| # XXX copied from execnet's conftest.py - needs to be merged
 | |
| winpymap = {
 | |
|     'python2.7': r'C:\Python27\python.exe',
 | |
|     'python3.4': r'C:\Python34\python.exe',
 | |
|     'python3.5': r'C:\Python35\python.exe',
 | |
|     'python3.6': r'C:\Python36\python.exe',
 | |
| }
 | |
| 
 | |
| 
 | |
| def getexecutable(name, cache={}):
 | |
|     try:
 | |
|         return cache[name]
 | |
|     except KeyError:
 | |
|         executable = py.path.local.sysfind(name)
 | |
|         if executable:
 | |
|             import subprocess
 | |
|             popen = subprocess.Popen([str(executable), "--version"],
 | |
|                                      universal_newlines=True, stderr=subprocess.PIPE)
 | |
|             out, err = popen.communicate()
 | |
|             if name == "jython":
 | |
|                 if not err or "2.5" not in err:
 | |
|                     executable = None
 | |
|                 if "2.5.2" in err:
 | |
|                     executable = None  # http://bugs.jython.org/issue1790
 | |
|             elif popen.returncode != 0:
 | |
|                 # handle pyenv's 127
 | |
|                 executable = None
 | |
|         cache[name] = executable
 | |
|         return executable
 | |
| 
 | |
| 
 | |
| @pytest.fixture(params=['python2.7', 'python3.4', 'pypy', 'pypy3'])
 | |
| def anypython(request):
 | |
|     name = request.param
 | |
|     executable = getexecutable(name)
 | |
|     if executable is None:
 | |
|         if sys.platform == "win32":
 | |
|             executable = winpymap.get(name, None)
 | |
|             if executable:
 | |
|                 executable = py.path.local(executable)
 | |
|                 if executable.check():
 | |
|                     return executable
 | |
|         pytest.skip("no suitable %s found" % (name,))
 | |
|     return executable
 | |
| 
 | |
| # used at least by pytest-xdist plugin
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def _pytest(request):
 | |
|     """Return a helper which offers a gethookrecorder(hook) method which
 | |
|     returns a HookRecorder instance which helps to make assertions about called
 | |
|     hooks.
 | |
| 
 | |
|     """
 | |
|     return PytestArg(request)
 | |
| 
 | |
| 
 | |
| class PytestArg:
 | |
|     def __init__(self, request):
 | |
|         self.request = request
 | |
| 
 | |
|     def gethookrecorder(self, hook):
 | |
|         hookrecorder = HookRecorder(hook._pm)
 | |
|         self.request.addfinalizer(hookrecorder.finish_recording)
 | |
|         return hookrecorder
 | |
| 
 | |
| 
 | |
| def get_public_names(values):
 | |
|     """Only return names from iterator values without a leading underscore."""
 | |
|     return [x for x in values if x[0] != "_"]
 | |
| 
 | |
| 
 | |
| class ParsedCall:
 | |
|     def __init__(self, name, kwargs):
 | |
|         self.__dict__.update(kwargs)
 | |
|         self._name = name
 | |
| 
 | |
|     def __repr__(self):
 | |
|         d = self.__dict__.copy()
 | |
|         del d['_name']
 | |
|         return "<ParsedCall %r(**%r)>" % (self._name, d)
 | |
| 
 | |
| 
 | |
| class HookRecorder:
 | |
|     """Record all hooks called in a plugin manager.
 | |
| 
 | |
|     This wraps all the hook calls in the plugin manager, recording each call
 | |
|     before propagating the normal calls.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     def __init__(self, pluginmanager):
 | |
|         self._pluginmanager = pluginmanager
 | |
|         self.calls = []
 | |
| 
 | |
|         def before(hook_name, hook_impls, kwargs):
 | |
|             self.calls.append(ParsedCall(hook_name, kwargs))
 | |
| 
 | |
|         def after(outcome, hook_name, hook_impls, kwargs):
 | |
|             pass
 | |
| 
 | |
|         self._undo_wrapping = pluginmanager.add_hookcall_monitoring(before, after)
 | |
| 
 | |
|     def finish_recording(self):
 | |
|         self._undo_wrapping()
 | |
| 
 | |
|     def getcalls(self, names):
 | |
|         if isinstance(names, str):
 | |
|             names = names.split()
 | |
|         return [call for call in self.calls if call._name in names]
 | |
| 
 | |
|     def assert_contains(self, entries):
 | |
|         __tracebackhide__ = True
 | |
|         i = 0
 | |
|         entries = list(entries)
 | |
|         backlocals = sys._getframe(1).f_locals
 | |
|         while entries:
 | |
|             name, check = entries.pop(0)
 | |
|             for ind, call in enumerate(self.calls[i:]):
 | |
|                 if call._name == name:
 | |
|                     print("NAMEMATCH", name, call)
 | |
|                     if eval(check, backlocals, call.__dict__):
 | |
|                         print("CHECKERMATCH", repr(check), "->", call)
 | |
|                     else:
 | |
|                         print("NOCHECKERMATCH", repr(check), "-", call)
 | |
|                         continue
 | |
|                     i += ind + 1
 | |
|                     break
 | |
|                 print("NONAMEMATCH", name, "with", call)
 | |
|             else:
 | |
|                 pytest.fail("could not find %r check %r" % (name, check))
 | |
| 
 | |
|     def popcall(self, name):
 | |
|         __tracebackhide__ = True
 | |
|         for i, call in enumerate(self.calls):
 | |
|             if call._name == name:
 | |
|                 del self.calls[i]
 | |
|                 return call
 | |
|         lines = ["could not find call %r, in:" % (name,)]
 | |
|         lines.extend(["  %s" % str(x) for x in self.calls])
 | |
|         pytest.fail("\n".join(lines))
 | |
| 
 | |
|     def getcall(self, name):
 | |
|         values = self.getcalls(name)
 | |
|         assert len(values) == 1, (name, values)
 | |
|         return values[0]
 | |
| 
 | |
|     # functionality for test reports
 | |
| 
 | |
|     def getreports(self,
 | |
|                    names="pytest_runtest_logreport pytest_collectreport"):
 | |
|         return [x.report for x in self.getcalls(names)]
 | |
| 
 | |
|     def matchreport(self, inamepart="",
 | |
|                     names="pytest_runtest_logreport pytest_collectreport", when=None):
 | |
|         """return a testreport whose dotted import path matches"""
 | |
|         values = []
 | |
|         for rep in self.getreports(names=names):
 | |
|             try:
 | |
|                 if not when and rep.when != "call" and rep.passed:
 | |
|                     # setup/teardown passing reports - let's ignore those
 | |
|                     continue
 | |
|             except AttributeError:
 | |
|                 pass
 | |
|             if when and getattr(rep, 'when', None) != when:
 | |
|                 continue
 | |
|             if not inamepart or inamepart in rep.nodeid.split("::"):
 | |
|                 values.append(rep)
 | |
|         if not values:
 | |
|             raise ValueError("could not find test report matching %r: "
 | |
|                              "no test reports at all!" % (inamepart,))
 | |
|         if len(values) > 1:
 | |
|             raise ValueError(
 | |
|                 "found 2 or more testreports matching %r: %s" % (inamepart, values))
 | |
|         return values[0]
 | |
| 
 | |
|     def getfailures(self,
 | |
|                     names='pytest_runtest_logreport pytest_collectreport'):
 | |
|         return [rep for rep in self.getreports(names) if rep.failed]
 | |
| 
 | |
|     def getfailedcollections(self):
 | |
|         return self.getfailures('pytest_collectreport')
 | |
| 
 | |
|     def listoutcomes(self):
 | |
|         passed = []
 | |
|         skipped = []
 | |
|         failed = []
 | |
|         for rep in self.getreports(
 | |
|                 "pytest_collectreport pytest_runtest_logreport"):
 | |
|             if rep.passed:
 | |
|                 if getattr(rep, "when", None) == "call":
 | |
|                     passed.append(rep)
 | |
|             elif rep.skipped:
 | |
|                 skipped.append(rep)
 | |
|             elif rep.failed:
 | |
|                 failed.append(rep)
 | |
|         return passed, skipped, failed
 | |
| 
 | |
|     def countoutcomes(self):
 | |
|         return [len(x) for x in self.listoutcomes()]
 | |
| 
 | |
|     def assertoutcome(self, passed=0, skipped=0, failed=0):
 | |
|         realpassed, realskipped, realfailed = self.listoutcomes()
 | |
|         assert passed == len(realpassed)
 | |
|         assert skipped == len(realskipped)
 | |
|         assert failed == len(realfailed)
 | |
| 
 | |
|     def clear(self):
 | |
|         self.calls[:] = []
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def linecomp(request):
 | |
|     return LineComp()
 | |
| 
 | |
| 
 | |
| @pytest.fixture(name='LineMatcher')
 | |
| def LineMatcher_fixture(request):
 | |
|     return LineMatcher
 | |
| 
 | |
| 
 | |
| @pytest.fixture
 | |
| def testdir(request, tmpdir_factory):
 | |
|     return Testdir(request, tmpdir_factory)
 | |
| 
 | |
| 
 | |
| rex_outcome = re.compile(r"(\d+) ([\w-]+)")
 | |
| 
 | |
| 
 | |
| class RunResult:
 | |
|     """The result of running a command.
 | |
| 
 | |
|     Attributes:
 | |
| 
 | |
|     :ret: the return value
 | |
|     :outlines: list of lines captured from stdout
 | |
|     :errlines: list of lines captures from stderr
 | |
|     :stdout: :py:class:`LineMatcher` of stdout, use ``stdout.str()`` to
 | |
|        reconstruct stdout or the commonly used ``stdout.fnmatch_lines()``
 | |
|        method
 | |
|     :stderr: :py:class:`LineMatcher` of stderr
 | |
|     :duration: duration in seconds
 | |
| 
 | |
|     """
 | |
| 
 | |
|     def __init__(self, ret, outlines, errlines, duration):
 | |
|         self.ret = ret
 | |
|         self.outlines = outlines
 | |
|         self.errlines = errlines
 | |
|         self.stdout = LineMatcher(outlines)
 | |
|         self.stderr = LineMatcher(errlines)
 | |
|         self.duration = duration
 | |
| 
 | |
|     def parseoutcomes(self):
 | |
|         """Return a dictionary of outcomestring->num from parsing the terminal
 | |
|         output that the test process produced.
 | |
| 
 | |
|         """
 | |
|         for line in reversed(self.outlines):
 | |
|             if 'seconds' in line:
 | |
|                 outcomes = rex_outcome.findall(line)
 | |
|                 if outcomes:
 | |
|                     d = {}
 | |
|                     for num, cat in outcomes:
 | |
|                         d[cat] = int(num)
 | |
|                     return d
 | |
|         raise ValueError("Pytest terminal report not found")
 | |
| 
 | |
|     def assert_outcomes(self, passed=0, skipped=0, failed=0, error=0):
 | |
|         """Assert that the specified outcomes appear with the respective
 | |
|         numbers (0 means it didn't occur) in the text output from a test run.
 | |
| 
 | |
|         """
 | |
|         d = self.parseoutcomes()
 | |
|         obtained = {
 | |
|             'passed': d.get('passed', 0),
 | |
|             'skipped': d.get('skipped', 0),
 | |
|             'failed': d.get('failed', 0),
 | |
|             'error': d.get('error', 0),
 | |
|         }
 | |
|         assert obtained == dict(passed=passed, skipped=skipped, failed=failed, error=error)
 | |
| 
 | |
| 
 | |
| class Testdir:
 | |
|     """Temporary test directory with tools to test/run pytest itself.
 | |
| 
 | |
|     This is based on the ``tmpdir`` fixture but provides a number of methods
 | |
|     which aid with testing pytest itself.  Unless :py:meth:`chdir` is used all
 | |
|     methods will use :py:attr:`tmpdir` as their current working directory.
 | |
| 
 | |
|     Attributes:
 | |
| 
 | |
|     :tmpdir: The :py:class:`py.path.local` instance of the temporary directory.
 | |
| 
 | |
|     :plugins: A list of plugins to use with :py:meth:`parseconfig` and
 | |
|        :py:meth:`runpytest`.  Initially this is an empty list but plugins can
 | |
|        be added to the list.  The type of items to add to the list depends on
 | |
|        the method using them so refer to them for details.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     def __init__(self, request, tmpdir_factory):
 | |
|         self.request = request
 | |
|         self._mod_collections = WeakKeyDictionary()
 | |
|         name = request.function.__name__
 | |
|         self.tmpdir = tmpdir_factory.mktemp(name, numbered=True)
 | |
|         self.plugins = []
 | |
|         self._savesyspath = (list(sys.path), list(sys.meta_path))
 | |
|         self._savemodulekeys = set(sys.modules)
 | |
|         self.chdir()  # always chdir
 | |
|         self.request.addfinalizer(self.finalize)
 | |
|         method = self.request.config.getoption("--runpytest")
 | |
|         if method == "inprocess":
 | |
|             self._runpytest_method = self.runpytest_inprocess
 | |
|         elif method == "subprocess":
 | |
|             self._runpytest_method = self.runpytest_subprocess
 | |
| 
 | |
|     def __repr__(self):
 | |
|         return "<Testdir %r>" % (self.tmpdir,)
 | |
| 
 | |
|     def finalize(self):
 | |
|         """Clean up global state artifacts.
 | |
| 
 | |
|         Some methods modify the global interpreter state and this tries to
 | |
|         clean this up.  It does not remove the temporary directory however so
 | |
|         it can be looked at after the test run has finished.
 | |
| 
 | |
|         """
 | |
|         sys.path[:], sys.meta_path[:] = self._savesyspath
 | |
|         if hasattr(self, '_olddir'):
 | |
|             self._olddir.chdir()
 | |
|         self.delete_loaded_modules()
 | |
| 
 | |
|     def delete_loaded_modules(self):
 | |
|         """Delete modules that have been loaded during a test.
 | |
| 
 | |
|         This allows the interpreter to catch module changes in case
 | |
|         the module is re-imported.
 | |
|         """
 | |
|         for name in set(sys.modules).difference(self._savemodulekeys):
 | |
|             # some zope modules used by twisted-related tests keeps internal
 | |
|             # state and can't be deleted; we had some trouble in the past
 | |
|             # with zope.interface for example
 | |
|             if not name.startswith("zope"):
 | |
|                 del sys.modules[name]
 | |
| 
 | |
|     def make_hook_recorder(self, pluginmanager):
 | |
|         """Create a new :py:class:`HookRecorder` for a PluginManager."""
 | |
|         assert not hasattr(pluginmanager, "reprec")
 | |
|         pluginmanager.reprec = reprec = HookRecorder(pluginmanager)
 | |
|         self.request.addfinalizer(reprec.finish_recording)
 | |
|         return reprec
 | |
| 
 | |
|     def chdir(self):
 | |
|         """Cd into the temporary directory.
 | |
| 
 | |
|         This is done automatically upon instantiation.
 | |
| 
 | |
|         """
 | |
|         old = self.tmpdir.chdir()
 | |
|         if not hasattr(self, '_olddir'):
 | |
|             self._olddir = old
 | |
| 
 | |
|     def _makefile(self, ext, args, kwargs, encoding='utf-8'):
 | |
|         items = list(kwargs.items())
 | |
| 
 | |
|         def to_text(s):
 | |
|             return s.decode(encoding) if isinstance(s, bytes) else six.text_type(s)
 | |
| 
 | |
|         if args:
 | |
|             source = u"\n".join(to_text(x) for x in args)
 | |
|             basename = self.request.function.__name__
 | |
|             items.insert(0, (basename, source))
 | |
| 
 | |
|         ret = None
 | |
|         for basename, value in items:
 | |
|             p = self.tmpdir.join(basename).new(ext=ext)
 | |
|             p.dirpath().ensure_dir()
 | |
|             source = Source(value)
 | |
|             source = u"\n".join(to_text(line) for line in source.lines)
 | |
|             p.write(source.strip().encode(encoding), "wb")
 | |
|             if ret is None:
 | |
|                 ret = p
 | |
|         return ret
 | |
| 
 | |
|     def makefile(self, ext, *args, **kwargs):
 | |
|         """Create a new file in the testdir.
 | |
| 
 | |
|         ext: The extension the file should use, including the dot, e.g. `.py`.
 | |
| 
 | |
|         args: All args will be treated as strings and joined using newlines.
 | |
|            The result will be written as contents to the file.  The name of the
 | |
|            file will be based on the test function requesting this fixture.
 | |
|            E.g. "testdir.makefile('.txt', 'line1', 'line2')"
 | |
| 
 | |
|         kwargs: Each keyword is the name of a file, while the value of it will
 | |
|            be written as contents of the file.
 | |
|            E.g. "testdir.makefile('.ini', pytest='[pytest]\naddopts=-rs\n')"
 | |
| 
 | |
|         """
 | |
|         return self._makefile(ext, args, kwargs)
 | |
| 
 | |
|     def makeconftest(self, source):
 | |
|         """Write a contest.py file with 'source' as contents."""
 | |
|         return self.makepyfile(conftest=source)
 | |
| 
 | |
|     def makeini(self, source):
 | |
|         """Write a tox.ini file with 'source' as contents."""
 | |
|         return self.makefile('.ini', tox=source)
 | |
| 
 | |
|     def getinicfg(self, source):
 | |
|         """Return the pytest section from the tox.ini config file."""
 | |
|         p = self.makeini(source)
 | |
|         return py.iniconfig.IniConfig(p)['pytest']
 | |
| 
 | |
|     def makepyfile(self, *args, **kwargs):
 | |
|         """Shortcut for .makefile() with a .py extension."""
 | |
|         return self._makefile('.py', args, kwargs)
 | |
| 
 | |
|     def maketxtfile(self, *args, **kwargs):
 | |
|         """Shortcut for .makefile() with a .txt extension."""
 | |
|         return self._makefile('.txt', args, kwargs)
 | |
| 
 | |
|     def syspathinsert(self, path=None):
 | |
|         """Prepend a directory to sys.path, defaults to :py:attr:`tmpdir`.
 | |
| 
 | |
|         This is undone automatically when this object dies at the end of each
 | |
|         test.
 | |
| 
 | |
|         """
 | |
|         if path is None:
 | |
|             path = self.tmpdir
 | |
|         sys.path.insert(0, str(path))
 | |
|         # a call to syspathinsert() usually means that the caller wants to
 | |
|         # import some dynamically created files, thus with python3 we
 | |
|         # invalidate its import caches
 | |
|         self._possibly_invalidate_import_caches()
 | |
| 
 | |
|     def _possibly_invalidate_import_caches(self):
 | |
|         # invalidate caches if we can (py33 and above)
 | |
|         try:
 | |
|             import importlib
 | |
|         except ImportError:
 | |
|             pass
 | |
|         else:
 | |
|             if hasattr(importlib, "invalidate_caches"):
 | |
|                 importlib.invalidate_caches()
 | |
| 
 | |
|     def mkdir(self, name):
 | |
|         """Create a new (sub)directory."""
 | |
|         return self.tmpdir.mkdir(name)
 | |
| 
 | |
|     def mkpydir(self, name):
 | |
|         """Create a new python package.
 | |
| 
 | |
|         This creates a (sub)directory with an empty ``__init__.py`` file so it
 | |
|         gets recognised as a python package.
 | |
| 
 | |
|         """
 | |
|         p = self.mkdir(name)
 | |
|         p.ensure("__init__.py")
 | |
|         return p
 | |
| 
 | |
|     Session = Session
 | |
| 
 | |
|     def getnode(self, config, arg):
 | |
|         """Return the collection node of a file.
 | |
| 
 | |
|         :param config: :py:class:`_pytest.config.Config` instance, see
 | |
|            :py:meth:`parseconfig` and :py:meth:`parseconfigure` to create the
 | |
|            configuration
 | |
| 
 | |
|         :param arg: a :py:class:`py.path.local` instance of the file
 | |
| 
 | |
|         """
 | |
|         session = Session(config)
 | |
|         assert '::' not in str(arg)
 | |
|         p = py.path.local(arg)
 | |
|         config.hook.pytest_sessionstart(session=session)
 | |
|         res = session.perform_collect([str(p)], genitems=False)[0]
 | |
|         config.hook.pytest_sessionfinish(session=session, exitstatus=EXIT_OK)
 | |
|         return res
 | |
| 
 | |
|     def getpathnode(self, path):
 | |
|         """Return the collection node of a file.
 | |
| 
 | |
|         This is like :py:meth:`getnode` but uses :py:meth:`parseconfigure` to
 | |
|         create the (configured) pytest Config instance.
 | |
| 
 | |
|         :param path: a :py:class:`py.path.local` instance of the file
 | |
| 
 | |
|         """
 | |
|         config = self.parseconfigure(path)
 | |
|         session = Session(config)
 | |
|         x = session.fspath.bestrelpath(path)
 | |
|         config.hook.pytest_sessionstart(session=session)
 | |
|         res = session.perform_collect([x], genitems=False)[0]
 | |
|         config.hook.pytest_sessionfinish(session=session, exitstatus=EXIT_OK)
 | |
|         return res
 | |
| 
 | |
|     def genitems(self, colitems):
 | |
|         """Generate all test items from a collection node.
 | |
| 
 | |
|         This recurses into the collection node and returns a list of all the
 | |
|         test items contained within.
 | |
| 
 | |
|         """
 | |
|         session = colitems[0].session
 | |
|         result = []
 | |
|         for colitem in colitems:
 | |
|             result.extend(session.genitems(colitem))
 | |
|         return result
 | |
| 
 | |
|     def runitem(self, source):
 | |
|         """Run the "test_func" Item.
 | |
| 
 | |
|         The calling test instance (class containing the test method) must
 | |
|         provide a ``.getrunner()`` method which should return a runner which
 | |
|         can run the test protocol for a single item, e.g.
 | |
|         :py:func:`_pytest.runner.runtestprotocol`.
 | |
| 
 | |
|         """
 | |
|         # used from runner functional tests
 | |
|         item = self.getitem(source)
 | |
|         # the test class where we are called from wants to provide the runner
 | |
|         testclassinstance = self.request.instance
 | |
|         runner = testclassinstance.getrunner()
 | |
|         return runner(item)
 | |
| 
 | |
|     def inline_runsource(self, source, *cmdlineargs):
 | |
|         """Run a test module in process using ``pytest.main()``.
 | |
| 
 | |
|         This run writes "source" into a temporary file and runs
 | |
|         ``pytest.main()`` on it, returning a :py:class:`HookRecorder` instance
 | |
|         for the result.
 | |
| 
 | |
|         :param source: the source code of the test module
 | |
| 
 | |
|         :param cmdlineargs: any extra command line arguments to use
 | |
| 
 | |
|         :return: :py:class:`HookRecorder` instance of the result
 | |
| 
 | |
|         """
 | |
|         p = self.makepyfile(source)
 | |
|         values = list(cmdlineargs) + [p]
 | |
|         return self.inline_run(*values)
 | |
| 
 | |
|     def inline_genitems(self, *args):
 | |
|         """Run ``pytest.main(['--collectonly'])`` in-process.
 | |
| 
 | |
|         Runs the :py:func:`pytest.main` function to run all of pytest inside
 | |
|         the test process itself like :py:meth:`inline_run`, but returns a
 | |
|         tuple of the collected items and a :py:class:`HookRecorder` instance.
 | |
| 
 | |
|         """
 | |
|         rec = self.inline_run("--collect-only", *args)
 | |
|         items = [x.item for x in rec.getcalls("pytest_itemcollected")]
 | |
|         return items, rec
 | |
| 
 | |
|     def inline_run(self, *args, **kwargs):
 | |
|         """Run ``pytest.main()`` in-process, returning a HookRecorder.
 | |
| 
 | |
|         Runs the :py:func:`pytest.main` function to run all of pytest inside
 | |
|         the test process itself.  This means it can return a
 | |
|         :py:class:`HookRecorder` instance which gives more detailed results
 | |
|         from that run than can be done by matching stdout/stderr from
 | |
|         :py:meth:`runpytest`.
 | |
| 
 | |
|         :param args: command line arguments to pass to :py:func:`pytest.main`
 | |
| 
 | |
|         :param plugin: (keyword-only) extra plugin instances the
 | |
|            ``pytest.main()`` instance should use
 | |
| 
 | |
|         :return: a :py:class:`HookRecorder` instance
 | |
| 
 | |
|         """
 | |
|         # When running py.test inline any plugins active in the main test
 | |
|         # process are already imported.  So this disables the warning which
 | |
|         # will trigger to say they can no longer be rewritten, which is fine as
 | |
|         # they have already been rewritten.
 | |
|         orig_warn = AssertionRewritingHook._warn_already_imported
 | |
| 
 | |
|         def revert():
 | |
|             AssertionRewritingHook._warn_already_imported = orig_warn
 | |
| 
 | |
|         self.request.addfinalizer(revert)
 | |
|         AssertionRewritingHook._warn_already_imported = lambda *a: None
 | |
| 
 | |
|         rec = []
 | |
| 
 | |
|         class Collect:
 | |
|             def pytest_configure(x, config):
 | |
|                 rec.append(self.make_hook_recorder(config.pluginmanager))
 | |
| 
 | |
|         plugins = kwargs.get("plugins") or []
 | |
|         plugins.append(Collect())
 | |
|         ret = pytest.main(list(args), plugins=plugins)
 | |
|         self.delete_loaded_modules()
 | |
|         if len(rec) == 1:
 | |
|             reprec = rec.pop()
 | |
|         else:
 | |
|             class reprec:
 | |
|                 pass
 | |
|         reprec.ret = ret
 | |
| 
 | |
|         # typically we reraise keyboard interrupts from the child run because
 | |
|         # it's our user requesting interruption of the testing
 | |
|         if ret == 2 and not kwargs.get("no_reraise_ctrlc"):
 | |
|             calls = reprec.getcalls("pytest_keyboard_interrupt")
 | |
|             if calls and calls[-1].excinfo.type == KeyboardInterrupt:
 | |
|                 raise KeyboardInterrupt()
 | |
|         return reprec
 | |
| 
 | |
|     def runpytest_inprocess(self, *args, **kwargs):
 | |
|         """Return result of running pytest in-process, providing a similar
 | |
|         interface to what self.runpytest() provides.
 | |
| 
 | |
|         """
 | |
|         if kwargs.get("syspathinsert"):
 | |
|             self.syspathinsert()
 | |
|         now = time.time()
 | |
|         capture = MultiCapture(Capture=SysCapture)
 | |
|         capture.start_capturing()
 | |
|         try:
 | |
|             try:
 | |
|                 reprec = self.inline_run(*args, **kwargs)
 | |
|             except SystemExit as e:
 | |
| 
 | |
|                 class reprec:
 | |
|                     ret = e.args[0]
 | |
| 
 | |
|             except Exception:
 | |
|                 traceback.print_exc()
 | |
| 
 | |
|                 class reprec:
 | |
|                     ret = 3
 | |
|         finally:
 | |
|             out, err = capture.readouterr()
 | |
|             capture.stop_capturing()
 | |
|             sys.stdout.write(out)
 | |
|             sys.stderr.write(err)
 | |
| 
 | |
|         res = RunResult(reprec.ret,
 | |
|                         out.split("\n"), err.split("\n"),
 | |
|                         time.time() - now)
 | |
|         res.reprec = reprec
 | |
|         return res
 | |
| 
 | |
|     def runpytest(self, *args, **kwargs):
 | |
|         """Run pytest inline or in a subprocess, depending on the command line
 | |
|         option "--runpytest" and return a :py:class:`RunResult`.
 | |
| 
 | |
|         """
 | |
|         args = self._ensure_basetemp(args)
 | |
|         return self._runpytest_method(*args, **kwargs)
 | |
| 
 | |
|     def _ensure_basetemp(self, args):
 | |
|         args = [str(x) for x in args]
 | |
|         for x in args:
 | |
|             if str(x).startswith('--basetemp'):
 | |
|                 # print("basedtemp exists: %s" %(args,))
 | |
|                 break
 | |
|         else:
 | |
|             args.append("--basetemp=%s" % self.tmpdir.dirpath('basetemp'))
 | |
|             # print("added basetemp: %s" %(args,))
 | |
|         return args
 | |
| 
 | |
|     def parseconfig(self, *args):
 | |
|         """Return a new pytest Config instance from given commandline args.
 | |
| 
 | |
|         This invokes the pytest bootstrapping code in _pytest.config to create
 | |
|         a new :py:class:`_pytest.core.PluginManager` and call the
 | |
|         pytest_cmdline_parse hook to create a new
 | |
|         :py:class:`_pytest.config.Config` instance.
 | |
| 
 | |
|         If :py:attr:`plugins` has been populated they should be plugin modules
 | |
|         to be registered with the PluginManager.
 | |
| 
 | |
|         """
 | |
|         args = self._ensure_basetemp(args)
 | |
| 
 | |
|         import _pytest.config
 | |
|         config = _pytest.config._prepareconfig(args, self.plugins)
 | |
|         # we don't know what the test will do with this half-setup config
 | |
|         # object and thus we make sure it gets unconfigured properly in any
 | |
|         # case (otherwise capturing could still be active, for example)
 | |
|         self.request.addfinalizer(config._ensure_unconfigure)
 | |
|         return config
 | |
| 
 | |
|     def parseconfigure(self, *args):
 | |
|         """Return a new pytest configured Config instance.
 | |
| 
 | |
|         This returns a new :py:class:`_pytest.config.Config` instance like
 | |
|         :py:meth:`parseconfig`, but also calls the pytest_configure hook.
 | |
| 
 | |
|         """
 | |
|         config = self.parseconfig(*args)
 | |
|         config._do_configure()
 | |
|         self.request.addfinalizer(config._ensure_unconfigure)
 | |
|         return config
 | |
| 
 | |
|     def getitem(self, source, funcname="test_func"):
 | |
|         """Return the test item for a test function.
 | |
| 
 | |
|         This writes the source to a python file and runs pytest's collection on
 | |
|         the resulting module, returning the test item for the requested
 | |
|         function name.
 | |
| 
 | |
|         :param source: the module source
 | |
| 
 | |
|         :param funcname: the name of the test function for which to return a
 | |
|             test item
 | |
| 
 | |
|         """
 | |
|         items = self.getitems(source)
 | |
|         for item in items:
 | |
|             if item.name == funcname:
 | |
|                 return item
 | |
|         assert 0, "%r item not found in module:\n%s\nitems: %s" % (
 | |
|                   funcname, source, items)
 | |
| 
 | |
|     def getitems(self, source):
 | |
|         """Return all test items collected from the module.
 | |
| 
 | |
|         This writes the source to a python file and runs pytest's collection on
 | |
|         the resulting module, returning all test items contained within.
 | |
| 
 | |
|         """
 | |
|         modcol = self.getmodulecol(source)
 | |
|         return self.genitems([modcol])
 | |
| 
 | |
|     def getmodulecol(self, source, configargs=(), withinit=False):
 | |
|         """Return the module collection node for ``source``.
 | |
| 
 | |
|         This writes ``source`` to a file using :py:meth:`makepyfile` and then
 | |
|         runs the pytest collection on it, returning the collection node for the
 | |
|         test module.
 | |
| 
 | |
|         :param source: the source code of the module to collect
 | |
| 
 | |
|         :param configargs: any extra arguments to pass to
 | |
|             :py:meth:`parseconfigure`
 | |
| 
 | |
|         :param withinit: whether to also write an ``__init__.py`` file to the
 | |
|             same directory to ensure it is a package
 | |
| 
 | |
|         """
 | |
|         kw = {self.request.function.__name__: Source(source).strip()}
 | |
|         path = self.makepyfile(**kw)
 | |
|         if withinit:
 | |
|             self.makepyfile(__init__="#")
 | |
|         self.config = config = self.parseconfigure(path, *configargs)
 | |
|         node = self.getnode(config, path)
 | |
| 
 | |
|         return node
 | |
| 
 | |
|     def collect_by_name(self, modcol, name):
 | |
|         """Return the collection node for name from the module collection.
 | |
| 
 | |
|         This will search a module collection node for a collection node
 | |
|         matching the given name.
 | |
| 
 | |
|         :param modcol: a module collection node; see :py:meth:`getmodulecol`
 | |
| 
 | |
|         :param name: the name of the node to return
 | |
| 
 | |
|         """
 | |
|         if modcol not in self._mod_collections:
 | |
|             self._mod_collections[modcol] = list(modcol.collect())
 | |
|         for colitem in self._mod_collections[modcol]:
 | |
|             if colitem.name == name:
 | |
|                 return colitem
 | |
| 
 | |
|     def popen(self, cmdargs, stdout, stderr, **kw):
 | |
|         """Invoke subprocess.Popen.
 | |
| 
 | |
|         This calls subprocess.Popen making sure the current working directory
 | |
|         is in the PYTHONPATH.
 | |
| 
 | |
|         You probably want to use :py:meth:`run` instead.
 | |
| 
 | |
|         """
 | |
|         env = os.environ.copy()
 | |
|         env['PYTHONPATH'] = os.pathsep.join(filter(None, [
 | |
|             str(os.getcwd()), env.get('PYTHONPATH', '')]))
 | |
|         kw['env'] = env
 | |
| 
 | |
|         popen = subprocess.Popen(cmdargs, stdin=subprocess.PIPE, stdout=stdout, stderr=stderr, **kw)
 | |
|         popen.stdin.close()
 | |
| 
 | |
|         return popen
 | |
| 
 | |
|     def run(self, *cmdargs):
 | |
|         """Run a command with arguments.
 | |
| 
 | |
|         Run a process using subprocess.Popen saving the stdout and stderr.
 | |
| 
 | |
|         Returns a :py:class:`RunResult`.
 | |
| 
 | |
|         """
 | |
|         return self._run(*cmdargs)
 | |
| 
 | |
|     def _run(self, *cmdargs):
 | |
|         cmdargs = [str(x) for x in cmdargs]
 | |
|         p1 = self.tmpdir.join("stdout")
 | |
|         p2 = self.tmpdir.join("stderr")
 | |
|         print("running:", ' '.join(cmdargs))
 | |
|         print("     in:", str(py.path.local()))
 | |
|         f1 = codecs.open(str(p1), "w", encoding="utf8")
 | |
|         f2 = codecs.open(str(p2), "w", encoding="utf8")
 | |
|         try:
 | |
|             now = time.time()
 | |
|             popen = self.popen(cmdargs, stdout=f1, stderr=f2,
 | |
|                                close_fds=(sys.platform != "win32"))
 | |
|             ret = popen.wait()
 | |
|         finally:
 | |
|             f1.close()
 | |
|             f2.close()
 | |
|         f1 = codecs.open(str(p1), "r", encoding="utf8")
 | |
|         f2 = codecs.open(str(p2), "r", encoding="utf8")
 | |
|         try:
 | |
|             out = f1.read().splitlines()
 | |
|             err = f2.read().splitlines()
 | |
|         finally:
 | |
|             f1.close()
 | |
|             f2.close()
 | |
|         self._dump_lines(out, sys.stdout)
 | |
|         self._dump_lines(err, sys.stderr)
 | |
|         return RunResult(ret, out, err, time.time() - now)
 | |
| 
 | |
|     def _dump_lines(self, lines, fp):
 | |
|         try:
 | |
|             for line in lines:
 | |
|                 print(line, file=fp)
 | |
|         except UnicodeEncodeError:
 | |
|             print("couldn't print to %s because of encoding" % (fp,))
 | |
| 
 | |
|     def _getpytestargs(self):
 | |
|         # we cannot use `(sys.executable, script)` because on Windows the
 | |
|         # script is e.g. `pytest.exe`
 | |
|         return (sys.executable, PYTEST_FULLPATH) # noqa
 | |
| 
 | |
|     def runpython(self, script):
 | |
|         """Run a python script using sys.executable as interpreter.
 | |
| 
 | |
|         Returns a :py:class:`RunResult`.
 | |
| 
 | |
|         """
 | |
|         return self.run(sys.executable, script)
 | |
| 
 | |
|     def runpython_c(self, command):
 | |
|         """Run python -c "command", return a :py:class:`RunResult`."""
 | |
|         return self.run(sys.executable, "-c", command)
 | |
| 
 | |
|     def runpytest_subprocess(self, *args, **kwargs):
 | |
|         """Run pytest as a subprocess with given arguments.
 | |
| 
 | |
|         Any plugins added to the :py:attr:`plugins` list will added using the
 | |
|         ``-p`` command line option.  Additionally ``--basetemp`` is used put
 | |
|         any temporary files and directories in a numbered directory prefixed
 | |
|         with "runpytest-" so they do not conflict with the normal numbered
 | |
|         pytest location for temporary files and directories.
 | |
| 
 | |
|         Returns a :py:class:`RunResult`.
 | |
| 
 | |
|         """
 | |
|         p = py.path.local.make_numbered_dir(prefix="runpytest-",
 | |
|                                             keep=None, rootdir=self.tmpdir)
 | |
|         args = ('--basetemp=%s' % p,) + args
 | |
|         plugins = [x for x in self.plugins if isinstance(x, str)]
 | |
|         if plugins:
 | |
|             args = ('-p', plugins[0]) + args
 | |
|         args = self._getpytestargs() + args
 | |
|         return self.run(*args)
 | |
| 
 | |
|     def spawn_pytest(self, string, expect_timeout=10.0):
 | |
|         """Run pytest using pexpect.
 | |
| 
 | |
|         This makes sure to use the right pytest and sets up the temporary
 | |
|         directory locations.
 | |
| 
 | |
|         The pexpect child is returned.
 | |
| 
 | |
|         """
 | |
|         basetemp = self.tmpdir.mkdir("temp-pexpect")
 | |
|         invoke = " ".join(map(str, self._getpytestargs()))
 | |
|         cmd = "%s --basetemp=%s %s" % (invoke, basetemp, string)
 | |
|         return self.spawn(cmd, expect_timeout=expect_timeout)
 | |
| 
 | |
|     def spawn(self, cmd, expect_timeout=10.0):
 | |
|         """Run a command using pexpect.
 | |
| 
 | |
|         The pexpect child is returned.
 | |
| 
 | |
|         """
 | |
|         pexpect = pytest.importorskip("pexpect", "3.0")
 | |
|         if hasattr(sys, 'pypy_version_info') and '64' in platform.machine():
 | |
|             pytest.skip("pypy-64 bit not supported")
 | |
|         if sys.platform.startswith("freebsd"):
 | |
|             pytest.xfail("pexpect does not work reliably on freebsd")
 | |
|         logfile = self.tmpdir.join("spawn.out").open("wb")
 | |
|         child = pexpect.spawn(cmd, logfile=logfile)
 | |
|         self.request.addfinalizer(logfile.close)
 | |
|         child.timeout = expect_timeout
 | |
|         return child
 | |
| 
 | |
| 
 | |
| def getdecoded(out):
 | |
|     try:
 | |
|         return out.decode("utf-8")
 | |
|     except UnicodeDecodeError:
 | |
|         return "INTERNAL not-utf8-decodeable, truncated string:\n%s" % (
 | |
|             py.io.saferepr(out),)
 | |
| 
 | |
| 
 | |
| class LineComp:
 | |
|     def __init__(self):
 | |
|         self.stringio = py.io.TextIO()
 | |
| 
 | |
|     def assert_contains_lines(self, lines2):
 | |
|         """Assert that lines2 are contained (linearly) in lines1.
 | |
| 
 | |
|         Return a list of extralines found.
 | |
| 
 | |
|         """
 | |
|         __tracebackhide__ = True
 | |
|         val = self.stringio.getvalue()
 | |
|         self.stringio.truncate(0)
 | |
|         self.stringio.seek(0)
 | |
|         lines1 = val.split("\n")
 | |
|         return LineMatcher(lines1).fnmatch_lines(lines2)
 | |
| 
 | |
| 
 | |
| class LineMatcher:
 | |
|     """Flexible matching of text.
 | |
| 
 | |
|     This is a convenience class to test large texts like the output of
 | |
|     commands.
 | |
| 
 | |
|     The constructor takes a list of lines without their trailing newlines, i.e.
 | |
|     ``text.splitlines()``.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     def __init__(self, lines):
 | |
|         self.lines = lines
 | |
|         self._log_output = []
 | |
| 
 | |
|     def str(self):
 | |
|         """Return the entire original text."""
 | |
|         return "\n".join(self.lines)
 | |
| 
 | |
|     def _getlines(self, lines2):
 | |
|         if isinstance(lines2, str):
 | |
|             lines2 = Source(lines2)
 | |
|         if isinstance(lines2, Source):
 | |
|             lines2 = lines2.strip().lines
 | |
|         return lines2
 | |
| 
 | |
|     def fnmatch_lines_random(self, lines2):
 | |
|         """Check lines exist in the output using in any order.
 | |
| 
 | |
|         Lines are checked using ``fnmatch.fnmatch``. The argument is a list of
 | |
|         lines which have to occur in the output, in any order.
 | |
| 
 | |
|         """
 | |
|         self._match_lines_random(lines2, fnmatch)
 | |
| 
 | |
|     def re_match_lines_random(self, lines2):
 | |
|         """Check lines exist in the output using ``re.match``, in any order.
 | |
| 
 | |
|         The argument is a list of lines which have to occur in the output, in
 | |
|         any order.
 | |
| 
 | |
|         """
 | |
|         self._match_lines_random(lines2, lambda name, pat: re.match(pat, name))
 | |
| 
 | |
|     def _match_lines_random(self, lines2, match_func):
 | |
|         """Check lines exist in the output.
 | |
| 
 | |
|         The argument is a list of lines which have to occur in the output, in
 | |
|         any order.  Each line can contain glob whildcards.
 | |
| 
 | |
|         """
 | |
|         lines2 = self._getlines(lines2)
 | |
|         for line in lines2:
 | |
|             for x in self.lines:
 | |
|                 if line == x or match_func(x, line):
 | |
|                     self._log("matched: ", repr(line))
 | |
|                     break
 | |
|             else:
 | |
|                 self._log("line %r not found in output" % line)
 | |
|                 raise ValueError(self._log_text)
 | |
| 
 | |
|     def get_lines_after(self, fnline):
 | |
|         """Return all lines following the given line in the text.
 | |
| 
 | |
|         The given line can contain glob wildcards.
 | |
| 
 | |
|         """
 | |
|         for i, line in enumerate(self.lines):
 | |
|             if fnline == line or fnmatch(line, fnline):
 | |
|                 return self.lines[i + 1:]
 | |
|         raise ValueError("line %r not found in output" % fnline)
 | |
| 
 | |
|     def _log(self, *args):
 | |
|         self._log_output.append(' '.join((str(x) for x in args)))
 | |
| 
 | |
|     @property
 | |
|     def _log_text(self):
 | |
|         return '\n'.join(self._log_output)
 | |
| 
 | |
|     def fnmatch_lines(self, lines2):
 | |
|         """Search captured text for matching lines using ``fnmatch.fnmatch``.
 | |
| 
 | |
|         The argument is a list of lines which have to match and can use glob
 | |
|         wildcards.  If they do not match a pytest.fail() is called.  The
 | |
|         matches and non-matches are also printed on stdout.
 | |
| 
 | |
|         """
 | |
|         self._match_lines(lines2, fnmatch, 'fnmatch')
 | |
| 
 | |
|     def re_match_lines(self, lines2):
 | |
|         """Search captured text for matching lines using ``re.match``.
 | |
| 
 | |
|         The argument is a list of lines which have to match using ``re.match``.
 | |
|         If they do not match a pytest.fail() is called.
 | |
| 
 | |
|         The matches and non-matches are also printed on stdout.
 | |
| 
 | |
|         """
 | |
|         self._match_lines(lines2, lambda name, pat: re.match(pat, name), 're.match')
 | |
| 
 | |
|     def _match_lines(self, lines2, match_func, match_nickname):
 | |
|         """Underlying implementation of ``fnmatch_lines`` and ``re_match_lines``.
 | |
| 
 | |
|         :param list[str] lines2: list of string patterns to match. The actual
 | |
|             format depends on ``match_func``
 | |
|         :param match_func: a callable ``match_func(line, pattern)`` where line
 | |
|             is the captured line from stdout/stderr and pattern is the matching
 | |
|             pattern
 | |
|         :param str match_nickname: the nickname for the match function that
 | |
|             will be logged to stdout when a match occurs
 | |
| 
 | |
|         """
 | |
|         lines2 = self._getlines(lines2)
 | |
|         lines1 = self.lines[:]
 | |
|         nextline = None
 | |
|         extralines = []
 | |
|         __tracebackhide__ = True
 | |
|         for line in lines2:
 | |
|             nomatchprinted = False
 | |
|             while lines1:
 | |
|                 nextline = lines1.pop(0)
 | |
|                 if line == nextline:
 | |
|                     self._log("exact match:", repr(line))
 | |
|                     break
 | |
|                 elif match_func(nextline, line):
 | |
|                     self._log("%s:" % match_nickname, repr(line))
 | |
|                     self._log("   with:", repr(nextline))
 | |
|                     break
 | |
|                 else:
 | |
|                     if not nomatchprinted:
 | |
|                         self._log("nomatch:", repr(line))
 | |
|                         nomatchprinted = True
 | |
|                     self._log("    and:", repr(nextline))
 | |
|                 extralines.append(nextline)
 | |
|             else:
 | |
|                 self._log("remains unmatched: %r" % (line,))
 | |
|                 pytest.fail(self._log_text)
 |