Skip to content
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion Doc/library/pdb.rst
Original file line number Diff line number Diff line change
Expand Up @@ -198,7 +198,7 @@ The ``run*`` functions and :func:`set_trace` are aliases for instantiating the
access further features, you have to do this yourself:

.. class:: Pdb(completekey='tab', stdin=None, stdout=None, skip=None, \
nosigint=False, readrc=True)
nosigint=False, readrc=True, mode=None)

:class:`Pdb` is the debugger class.

Expand All @@ -217,6 +217,13 @@ access further features, you have to do this yourself:
The *readrc* argument defaults to true and controls whether Pdb will load
.pdbrc files from the filesystem.

The *mode* argument specifies how the debugger was invoked.
It impacts the workings of some debugger commands.
Valid values are ``'inline'`` (when called from the breakpoint() builtin),
``'cli'`` (when called from a command line invocation)
or ``None`` (for backwards compatible behaviour, as before the *mode*
argument was added).

Example call to enable tracing with *skip*::

import pdb; pdb.Pdb(skip=['django.*']).set_trace()
Expand All @@ -233,6 +240,9 @@ access further features, you have to do this yourself:
.. versionchanged:: 3.6
The *readrc* argument.

.. versionadded:: 3.14
Added the *mode* argument.

.. method:: run(statement, globals=None, locals=None)
runeval(expression, globals=None, locals=None)
runcall(function, *args, **kwds)
Expand Down Expand Up @@ -675,6 +685,10 @@ can be overridden by the local file.
History, breakpoints, actions and debugger options are preserved.
:pdbcmd:`restart` is an alias for :pdbcmd:`run`.

.. versionchanged:: 3.14
:pdbcmd:`run` and :pdbcmd:`restart` commands are not allowed when the
debugger is invoked from an inline breakpoint.

.. pdbcommand:: q(uit)

Quit from the debugger. The program being executed is aborted.
Expand Down
12 changes: 9 additions & 3 deletions Lib/pdb.py
Original file line number Diff line number Diff line change
Expand Up @@ -309,7 +309,7 @@ class Pdb(bdb.Bdb, cmd.Cmd):
_last_pdb_instance = None

def __init__(self, completekey='tab', stdin=None, stdout=None, skip=None,
nosigint=False, readrc=True):
nosigint=False, readrc=True, mode=None):
bdb.Bdb.__init__(self, skip=skip)
cmd.Cmd.__init__(self, completekey, stdin, stdout)
sys.audit("pdb.Pdb")
Expand All @@ -321,6 +321,7 @@ def __init__(self, completekey='tab', stdin=None, stdout=None, skip=None,
self.mainpyfile = ''
self._wait_for_mainpyfile = False
self.tb_lineno = {}
self.mode = mode
# Try to load readline if it exists
try:
import readline
Expand Down Expand Up @@ -1611,6 +1612,11 @@ def do_run(self, arg):
sys.argv. History, breakpoints, actions and debugger options
are preserved. "restart" is an alias for "run".
"""
if self.mode == 'inline':
self.error('run/restart command is disabled in inline breakpoints.\n'
'Use the command line interface to enable restarting your program\n'
'e.g. "python -m pdb myscript.py"')
return
if arg:
import shlex
argv0 = sys.argv[0:1]
Expand Down Expand Up @@ -2366,7 +2372,7 @@ def set_trace(*, header=None, commands=None):
if Pdb._last_pdb_instance is not None:
pdb = Pdb._last_pdb_instance
else:
pdb = Pdb()
pdb = Pdb(mode='inline')
if header is not None:
pdb.message(header)
pdb.set_trace(sys._getframe().f_back, commands=commands)
Expand Down Expand Up @@ -2481,7 +2487,7 @@ def main():
# modified by the script being debugged. It's a bad idea when it was
# changed by the user from the command line. There is a "restart" command
# which allows explicit specification of command line arguments.
pdb = Pdb()
pdb = Pdb(mode='cli')
pdb.rcLines.extend(opts.commands)
while True:
try:
Expand Down
21 changes: 21 additions & 0 deletions Lib/test/test_pdb.py
Original file line number Diff line number Diff line change
Expand Up @@ -901,6 +901,27 @@ def test_pdb_where_command():
(Pdb) continue
"""

def test_pdb_restart_command():
"""Test restart command

>>> def test_function():
... import pdb; pdb.Pdb(nosigint=True, readrc=False, mode='inline').set_trace()
... x = 1

>>> with PdbTestInput([ # doctest: +ELLIPSIS
... 'restart',
... 'continue',
... ]):
... test_function()
> <doctest test.test_pdb.test_pdb_restart_command[0]>(2)test_function()
-> import pdb; pdb.Pdb(nosigint=True, readrc=False, mode='inline').set_trace()
(Pdb) restart
*** run/restart command is disabled in inline breakpoints.
Use the command line interface to enable restarting your program
e.g. "python -m pdb myscript.py"
(Pdb) continue
"""

def test_pdb_commands_with_set_trace():
"""Test that commands can be passed to Pdb.set_trace()

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Added a new argument ``mode`` to :class:`pdb.Pdb`. Only allow :mod:`pdb` from command line to use ``restart`` command.
Loading