96 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			96 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Python
		
	
	
	
| """ interactive debugging with PDB, the Python Debugger. """
 | |
| 
 | |
| import pytest, py
 | |
| import sys
 | |
| 
 | |
| def pytest_addoption(parser):
 | |
|     group = parser.getgroup("general")
 | |
|     group._addoption('--pdb',
 | |
|                action="store_true", dest="usepdb", default=False,
 | |
|                help="start the interactive Python debugger on errors.")
 | |
| 
 | |
| def pytest_namespace():
 | |
|     return {'set_trace': pytestPDB().set_trace}
 | |
| 
 | |
| def pytest_configure(config):
 | |
|     if config.getvalue("usepdb"):
 | |
|         config.pluginmanager.register(PdbInvoke(), 'pdbinvoke')
 | |
| 
 | |
| class pytestPDB:
 | |
|     """ Pseudo PDB that defers to the real pdb. """
 | |
|     item = None
 | |
|     collector = None
 | |
| 
 | |
|     def set_trace(self):
 | |
|         """ invoke PDB set_trace debugging, dropping any IO capturing. """
 | |
|         frame = sys._getframe().f_back
 | |
|         item = self.item or self.collector
 | |
| 
 | |
|         if item is not None:
 | |
|             capman = item.config.pluginmanager.getplugin("capturemanager")
 | |
|             out, err = capman.suspendcapture()
 | |
|             if hasattr(item, 'outerr'):
 | |
|                 item.outerr = (item.outerr[0] + out, item.outerr[1] + err)
 | |
|             tw = py.io.TerminalWriter()
 | |
|             tw.line()
 | |
|             tw.sep(">", "PDB set_trace (IO-capturing turned off)")
 | |
|         py.std.pdb.Pdb().set_trace(frame)
 | |
| 
 | |
| def pdbitem(item):
 | |
|     pytestPDB.item = item
 | |
| pytest_runtest_setup = pytest_runtest_call = pytest_runtest_teardown = pdbitem
 | |
| 
 | |
| @pytest.mark.tryfirst
 | |
| def pytest_make_collect_report(__multicall__, collector):
 | |
|     try:
 | |
|         pytestPDB.collector = collector
 | |
|         return __multicall__.execute()
 | |
|     finally:
 | |
|         pytestPDB.collector = None
 | |
| 
 | |
| def pytest_runtest_makereport():
 | |
|     pytestPDB.item = None
 | |
|     
 | |
| class PdbInvoke:
 | |
|     @pytest.mark.tryfirst
 | |
|     def pytest_runtest_makereport(self, item, call, __multicall__):
 | |
|         rep = __multicall__.execute()
 | |
|         if not call.excinfo or \
 | |
|             call.excinfo.errisinstance(pytest.skip.Exception) or \
 | |
|             call.excinfo.errisinstance(py.std.bdb.BdbQuit):
 | |
|             return rep
 | |
|         if "xfail" in rep.keywords:
 | |
|             return rep
 | |
|         # we assume that the above execute() suspended capturing
 | |
|         # XXX we re-use the TerminalReporter's terminalwriter
 | |
|         # because this seems to avoid some encoding related troubles
 | |
|         # for not completely clear reasons.
 | |
|         tw = item.config.pluginmanager.getplugin("terminalreporter")._tw
 | |
|         tw.line()
 | |
|         tw.sep(">", "traceback")
 | |
|         rep.toterminal(tw)
 | |
|         tw.sep(">", "entering PDB")
 | |
|         # A doctest.UnexpectedException is not useful for post_mortem.
 | |
|         # Use the underlying exception instead:
 | |
|         if isinstance(call.excinfo.value, py.std.doctest.UnexpectedException):
 | |
|             tb = call.excinfo.value.exc_info[2]
 | |
|         else:
 | |
|             tb = call.excinfo._excinfo[2]
 | |
|         post_mortem(tb)
 | |
|         rep._pdbshown = True
 | |
|         return rep
 | |
| 
 | |
| def post_mortem(t):
 | |
|     pdb = py.std.pdb
 | |
|     class Pdb(pdb.Pdb):
 | |
|         def get_stack(self, f, t):
 | |
|             stack, i = pdb.Pdb.get_stack(self, f, t)
 | |
|             if f is None:
 | |
|                 i = max(0, len(stack) - 1)
 | |
|                 while i and stack[i][0].f_locals.get("__tracebackhide__", False):
 | |
|                     i-=1
 | |
|             return stack, i
 | |
|     p = Pdb()
 | |
|     p.reset()
 | |
|     p.interaction(None, t)
 |