Skip to content

Commit

Permalink
Copy docstrings to wrapped pdb methods
Browse files Browse the repository at this point in the history
Co-authored-by: Bruno Oliveira <[email protected]>
Co-authored-by: 🇺🇦 Sviatoslav Sydorenko (Святослав Сидоренко) <[email protected]>
  • Loading branch information
3 people committed Dec 27, 2024
1 parent 24e84f0 commit ecc28fe
Show file tree
Hide file tree
Showing 3 changed files with 80 additions and 6 deletions.
1 change: 1 addition & 0 deletions changelog/12946.bugfix.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Fixed missing help for :mod:`pdb` commands wrapped by pytest -- by :user:`adamchainz`.
17 changes: 11 additions & 6 deletions src/_pytest/debugging.py
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,8 @@ def do_debug(self, arg):
cls._recursive_debug -= 1
return ret

do_debug.__doc__ = pdb_cls.do_debug.__doc__

def do_continue(self, arg):
ret = super().do_continue(arg)
if cls._recursive_debug == 0:
Expand All @@ -185,22 +187,25 @@ def do_continue(self, arg):
self._continued = True
return ret

do_continue.__doc__ = pdb_cls.do_continue.__doc__

do_c = do_cont = do_continue

def do_quit(self, arg):
"""Raise Exit outcome when quit command is used in pdb.
This is a bit of a hack - it would be better if BdbQuit
could be handled, but this would require to wrap the
whole pytest run, and adjust the report etc.
"""
# Raise Exit outcome when quit command is used in pdb.
#
# This is a bit of a hack - it would be better if BdbQuit
# could be handled, but this would require to wrap the
# whole pytest run, and adjust the report etc.
ret = super().do_quit(arg)

if cls._recursive_debug == 0:
outcomes.exit("Quitting debugger")

return ret

do_quit.__doc__ = pdb_cls.do_quit.__doc__

do_q = do_quit
do_exit = do_quit

Expand Down
68 changes: 68 additions & 0 deletions testing/test_debugging.py
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,16 @@ def reset(self):
def interaction(self, *args):
called.append("interaction")

# Methods which we copy the docstring over.
def do_debug(self, *args):
pass

Check warning on line 57 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L57

Added line #L57 was not covered by tests

def do_continue(self, *args):
pass

Check warning on line 60 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L60

Added line #L60 was not covered by tests

def do_quit(self, *args):
pass

Check warning on line 63 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L63

Added line #L63 was not covered by tests

_pytest._CustomPdb = _CustomPdb # type: ignore
return called

Expand All @@ -75,6 +85,16 @@ def set_trace(self, frame):
print("**CustomDebugger**")
called.append("set_trace")

# Methods which we copy the docstring over.
def do_debug(self, *args):
pass

Check warning on line 90 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L90

Added line #L90 was not covered by tests

def do_continue(self, *args):
pass

Check warning on line 93 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L93

Added line #L93 was not covered by tests

def do_quit(self, *args):
pass

Check warning on line 96 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L96

Added line #L96 was not covered by tests

_pytest._CustomDebugger = _CustomDebugger # type: ignore
yield called
del _pytest._CustomDebugger # type: ignore
Expand Down Expand Up @@ -965,6 +985,34 @@ def test_1():
child.sendeof()
self.flush(child)

def test_pdb_wrapped_commands_docstrings(self, pytester: Pytester) -> None:
p1 = pytester.makepyfile(
"""
def test_1():
assert False
"""
)

child = pytester.spawn_pytest(f"--pdb {p1}")
child.expect("Pdb")

Check warning on line 997 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L997

Added line #L997 was not covered by tests

# Verify no undocumented commands
child.sendline("help")
child.expect("Documented commands")
assert "Undocumented commands" not in child.before.decode()

Check warning on line 1002 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L1000-L1002

Added lines #L1000 - L1002 were not covered by tests

child.sendline("help continue")
child.expect("Continue execution")
child.expect("Pdb")

Check warning on line 1006 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L1004-L1006

Added lines #L1004 - L1006 were not covered by tests

child.sendline("help debug")
child.expect("Enter a recursive debugger")
child.expect("Pdb")

Check warning on line 1010 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L1008-L1010

Added lines #L1008 - L1010 were not covered by tests

child.sendline("c")
child.sendeof()
self.flush(child)

Check warning on line 1014 in testing/test_debugging.py

View check run for this annotation

Codecov / codecov/patch

testing/test_debugging.py#L1012-L1014

Added lines #L1012 - L1014 were not covered by tests


class TestDebuggingBreakpoints:
@pytest.mark.parametrize("arg", ["--pdb", ""])
Expand Down Expand Up @@ -1288,6 +1336,16 @@ def set_trace(self, *args):
def runcall(self, *args, **kwds):
print("runcall_called", args, kwds)
# Methods which we copy the docstring over.
def do_debug(self, *args):
pass
def do_continue(self, *args):
pass
def do_quit(self, *args):
pass
""",
)
result = pytester.runpytest(
Expand Down Expand Up @@ -1354,6 +1412,16 @@ def __init__(self, *args, **kwargs):
def set_trace(self, *args):
print("set_trace_called", args)
# Methods which we copy the docstring over.
def do_debug(self, *args):
pass
def do_continue(self, *args):
pass
def do_quit(self, *args):
pass
""",
)
result = pytester.runpytest(str(p1), "--pdbcls=mypdb:MyPdb", syspathinsert=True)
Expand Down

0 comments on commit ecc28fe

Please sign in to comment.