pdb: move/refactor initialization of PytestPdbWrapper

This commit is contained in:
Daniel Hahler 2019-05-28 14:31:35 +02:00
parent b10f28949d
commit 61dfd0a94f
2 changed files with 169 additions and 128 deletions

View File

@ -81,6 +81,7 @@ class pytestPDB(object):
_config = None _config = None
_saved = [] _saved = []
_recursive_debug = 0 _recursive_debug = 0
_wrapped_pdb_cls = None
@classmethod @classmethod
def _is_capturing(cls, capman): def _is_capturing(cls, capman):
@ -89,16 +90,18 @@ class pytestPDB(object):
return False return False
@classmethod @classmethod
def _import_pdb_cls(cls): def _import_pdb_cls(cls, capman):
if not cls._config: if not cls._config:
# Happens when using pytest.set_trace outside of a test. # Happens when using pytest.set_trace outside of a test.
return pdb.Pdb return pdb.Pdb
pdb_cls = cls._config.getvalue("usepdb_cls") usepdb_cls = cls._config.getvalue("usepdb_cls")
if not pdb_cls:
return pdb.Pdb
modname, classname = pdb_cls if cls._wrapped_pdb_cls and cls._wrapped_pdb_cls[0] == usepdb_cls:
return cls._wrapped_pdb_cls[1]
if usepdb_cls:
modname, classname = usepdb_cls
try: try:
__import__(modname) __import__(modname)
@ -109,43 +112,21 @@ class pytestPDB(object):
pdb_cls = getattr(mod, parts[0]) pdb_cls = getattr(mod, parts[0])
for part in parts[1:]: for part in parts[1:]:
pdb_cls = getattr(pdb_cls, part) pdb_cls = getattr(pdb_cls, part)
return pdb_cls
except Exception as exc: except Exception as exc:
value = ":".join((modname, classname)) value = ":".join((modname, classname))
raise UsageError("--pdbcls: could not import {!r}: {}".format(value, exc)) raise UsageError(
"--pdbcls: could not import {!r}: {}".format(value, exc)
@classmethod
def _init_pdb(cls, *args, **kwargs):
""" Initialize PDB debugging, dropping any IO capturing. """
import _pytest.config
if cls._pluginmanager is not None:
capman = cls._pluginmanager.getplugin("capturemanager")
if capman:
capman.suspend(in_=True)
tw = _pytest.config.create_terminal_writer(cls._config)
tw.line()
if cls._recursive_debug == 0:
# Handle header similar to pdb.set_trace in py37+.
header = kwargs.pop("header", None)
if header is not None:
tw.sep(">", header)
else:
capturing = cls._is_capturing(capman)
if capturing:
if capturing == "global":
tw.sep(">", "PDB set_trace (IO-capturing turned off)")
else:
tw.sep(
">",
"PDB set_trace (IO-capturing turned off for %s)"
% capturing,
) )
else: else:
tw.sep(">", "PDB set_trace") pdb_cls = pdb.Pdb
pdb_cls = cls._import_pdb_cls() wrapped_cls = cls._get_pdb_wrapper_class(pdb_cls, capman)
cls._wrapped_pdb_cls = (usepdb_cls, wrapped_cls)
return wrapped_cls
@classmethod
def _get_pdb_wrapper_class(cls, pdb_cls, capman):
import _pytest.config
class PytestPdbWrapper(pdb_cls, object): class PytestPdbWrapper(pdb_cls, object):
_pytest_capman = capman _pytest_capman = capman
@ -177,9 +158,7 @@ class pytestPDB(object):
capman.resume() capman.resume()
else: else:
tw.sep(">", "PDB continue") tw.sep(">", "PDB continue")
cls._pluginmanager.hook.pytest_leave_pdb( cls._pluginmanager.hook.pytest_leave_pdb(config=cls._config, pdb=self)
config=cls._config, pdb=self
)
self._continued = True self._continued = True
return ret return ret
@ -221,24 +200,57 @@ class pytestPDB(object):
if f is None: if f is None:
# Find last non-hidden frame. # Find last non-hidden frame.
i = max(0, len(stack) - 1) i = max(0, len(stack) - 1)
while i and stack[i][0].f_locals.get( while i and stack[i][0].f_locals.get("__tracebackhide__", False):
"__tracebackhide__", False
):
i -= 1 i -= 1
return stack, i return stack, i
_pdb = PytestPdbWrapper(**kwargs) return PytestPdbWrapper
cls._pluginmanager.hook.pytest_enter_pdb(config=cls._config, pdb=_pdb)
@classmethod
def _init_pdb(cls, method, *args, **kwargs):
""" Initialize PDB debugging, dropping any IO capturing. """
import _pytest.config
if cls._pluginmanager is not None:
capman = cls._pluginmanager.getplugin("capturemanager")
else: else:
pdb_cls = cls._import_pdb_cls() capman = None
_pdb = pdb_cls(**kwargs) if capman:
capman.suspend(in_=True)
if cls._config:
tw = _pytest.config.create_terminal_writer(cls._config)
tw.line()
if cls._recursive_debug == 0:
# Handle header similar to pdb.set_trace in py37+.
header = kwargs.pop("header", None)
if header is not None:
tw.sep(">", header)
else:
capturing = cls._is_capturing(capman)
if capturing == "global":
tw.sep(">", "PDB %s (IO-capturing turned off)" % (method,))
elif capturing:
tw.sep(
">",
"PDB %s (IO-capturing turned off for %s)"
% (method, capturing),
)
else:
tw.sep(">", "PDB %s" % (method,))
_pdb = cls._import_pdb_cls(capman)(**kwargs)
if cls._pluginmanager:
cls._pluginmanager.hook.pytest_enter_pdb(config=cls._config, pdb=_pdb)
return _pdb return _pdb
@classmethod @classmethod
def set_trace(cls, *args, **kwargs): def set_trace(cls, *args, **kwargs):
"""Invoke debugging via ``Pdb.set_trace``, dropping any IO capturing.""" """Invoke debugging via ``Pdb.set_trace``, dropping any IO capturing."""
frame = sys._getframe().f_back frame = sys._getframe().f_back
_pdb = cls._init_pdb(*args, **kwargs) _pdb = cls._init_pdb("set_trace", *args, **kwargs)
_pdb.set_trace(frame) _pdb.set_trace(frame)
@ -265,7 +277,7 @@ class PdbTrace(object):
def _test_pytest_function(pyfuncitem): def _test_pytest_function(pyfuncitem):
_pdb = pytestPDB._init_pdb() _pdb = pytestPDB._init_pdb("runcall")
testfunction = pyfuncitem.obj testfunction = pyfuncitem.obj
pyfuncitem.obj = _pdb.runcall pyfuncitem.obj = _pdb.runcall
if "func" in pyfuncitem._fixtureinfo.argnames: # pragma: no branch if "func" in pyfuncitem._fixtureinfo.argnames: # pragma: no branch
@ -315,7 +327,7 @@ def _postmortem_traceback(excinfo):
def post_mortem(t): def post_mortem(t):
p = pytestPDB._init_pdb() p = pytestPDB._init_pdb("post_mortem")
p.reset() p.reset()
p.interaction(None, t) p.interaction(None, t)
if p.quitting: if p.quitting:

View File

@ -638,36 +638,35 @@ class TestPDB(object):
class pytestPDBTest(_pytest.debugging.pytestPDB): class pytestPDBTest(_pytest.debugging.pytestPDB):
@classmethod @classmethod
def set_trace(cls, *args, **kwargs): def set_trace(cls, *args, **kwargs):
# Init _PdbWrapper to handle capturing. # Init PytestPdbWrapper to handle capturing.
_pdb = cls._init_pdb(*args, **kwargs) _pdb = cls._init_pdb("set_trace", *args, **kwargs)
# Mock out pdb.Pdb.do_continue. # Mock out pdb.Pdb.do_continue.
import pdb import pdb
pdb.Pdb.do_continue = lambda self, arg: None pdb.Pdb.do_continue = lambda self, arg: None
print("=== SET_TRACE ===") print("===" + " SET_TRACE ===")
assert input() == "debug set_trace()" assert input() == "debug set_trace()"
# Simulate _PdbWrapper.do_debug # Simulate PytestPdbWrapper.do_debug
cls._recursive_debug += 1 cls._recursive_debug += 1
print("ENTERING RECURSIVE DEBUGGER") print("ENTERING RECURSIVE DEBUGGER")
print("=== SET_TRACE_2 ===") print("===" + " SET_TRACE_2 ===")
assert input() == "c" assert input() == "c"
_pdb.do_continue("") _pdb.do_continue("")
print("=== SET_TRACE_3 ===") print("===" + " SET_TRACE_3 ===")
# Simulate _PdbWrapper.do_debug # Simulate PytestPdbWrapper.do_debug
print("LEAVING RECURSIVE DEBUGGER") print("LEAVING RECURSIVE DEBUGGER")
cls._recursive_debug -= 1 cls._recursive_debug -= 1
print("=== SET_TRACE_4 ===") print("===" + " SET_TRACE_4 ===")
assert input() == "c" assert input() == "c"
_pdb.do_continue("") _pdb.do_continue("")
def do_continue(self, arg): def do_continue(self, arg):
print("=== do_continue") print("=== do_continue")
# _PdbWrapper.do_continue("")
monkeypatch.setattr(_pytest.debugging, "pytestPDB", pytestPDBTest) monkeypatch.setattr(_pytest.debugging, "pytestPDB", pytestPDBTest)
@ -677,7 +676,7 @@ class TestPDB(object):
set_trace() set_trace()
""" """
) )
child = testdir.spawn_pytest("%s %s" % (p1, capture_arg)) child = testdir.spawn_pytest("--tb=short %s %s" % (p1, capture_arg))
child.expect("=== SET_TRACE ===") child.expect("=== SET_TRACE ===")
before = child.before.decode("utf8") before = child.before.decode("utf8")
if not capture_arg: if not capture_arg:
@ -1207,3 +1206,33 @@ def test_raises_bdbquit_with_eoferror(testdir):
result = testdir.runpytest(str(p1)) result = testdir.runpytest(str(p1))
result.stdout.fnmatch_lines(["E *BdbQuit", "*= 1 failed in*"]) result.stdout.fnmatch_lines(["E *BdbQuit", "*= 1 failed in*"])
assert result.ret == 1 assert result.ret == 1
def test_pdb_wrapper_class_is_reused(testdir):
p1 = testdir.makepyfile(
"""
def test():
__import__("pdb").set_trace()
__import__("pdb").set_trace()
import mypdb
instances = mypdb.instances
assert len(instances) == 2
assert instances[0].__class__ is instances[1].__class__
""",
mypdb="""
instances = []
class MyPdb:
def __init__(self, *args, **kwargs):
instances.append(self)
def set_trace(self, *args):
print("set_trace_called", args)
""",
)
result = testdir.runpytest(str(p1), "--pdbcls=mypdb:MyPdb", syspathinsert=True)
assert result.ret == 0
result.stdout.fnmatch_lines(
["*set_trace_called*", "*set_trace_called*", "* 1 passed in *"]
)