Автоматический запуск Python-отладчика при ошибке - PullRequest
176 голосов
/ 28 октября 2008

Это вопрос, который меня интересовал уже довольно давно, но я так и не нашел подходящего решения. Если я запускаю скрипт и сталкиваюсь, скажем, с IndexError, python печатает строку, местоположение и краткое описание ошибки и завершает работу. Можно ли автоматически запускать pdb при возникновении ошибки? Я не против того, чтобы в верхней части файла были дополнительные операторы импорта или несколько строк кода.

Ответы [ 13 ]

366 голосов
/ 13 марта 2010
python -m pdb -c continue myscript.py

Если вы не укажете флаг -c continue, вам нужно будет ввести 'c' (для продолжения), когда начнется выполнение. Затем он дойдет до точки ошибки и даст вам контроль там. Как упоминается eqzx , этот флаг является новым дополнением в Python 3.2, поэтому для более ранних версий Python требуется ввод 'c' (см. https://docs.python.org/3/library/pdb.html).

107 голосов
/ 28 октября 2008

Вы можете использовать traceback.print_exc для печати трассировки исключений. Затем используйте sys.exc_info для извлечения трассировки и, наконец, вызовите pdb.post_mortem с этой трассировкой

import pdb, traceback, sys

def bombs():
    a = []
    print a[0]

if __name__ == '__main__':
    try:
        bombs()
    except:
        extype, value, tb = sys.exc_info()
        traceback.print_exc()
        pdb.post_mortem(tb)

Если вы хотите запустить интерактивную командную строку с code.interact , используя локальные элементы фрейма, из которого возникло исключение, вы можете сделать

import traceback, sys, code

def bombs():
    a = []
    print a[0]

if __name__ == '__main__':
    try:
        bombs()
    except:
        type, value, tb = sys.exc_info()
        traceback.print_exc()
        last_frame = lambda tb=tb: last_frame(tb.tb_next) if tb.tb_next else tb
        frame = last_frame().tb_frame
        ns = dict(frame.f_globals)
        ns.update(frame.f_locals)
        code.interact(local=ns)
60 голосов
/ 28 октября 2008

Используйте следующий модуль:

import sys

def info(type, value, tb):
    if hasattr(sys, 'ps1') or not sys.stderr.isatty():
    # we are in interactive mode or we don't have a tty-like
    # device, so we call the default hook
        sys.__excepthook__(type, value, tb)
    else:
        import traceback, pdb
        # we are NOT in interactive mode, print the exception...
        traceback.print_exception(type, value, tb)
        print
        # ...then start the debugger in post-mortem mode.
        # pdb.pm() # deprecated
        pdb.post_mortem(tb) # more "modern"

sys.excepthook = info

Назовите его debug (или как вам нравится) и поместите его где-нибудь на вашем пути к питону.

Теперь, в начале вашего скрипта, просто добавьте import debug.

36 голосов
/ 22 октября 2012

Ipython имеет команду для переключения этого поведения: % pdb . Он делает именно то, что вы описали, может быть, даже немного больше (дает вам более информативные обратные следы с подсветкой синтаксиса и дополнением кода). Это определенно стоит попробовать!

30 голосов
/ 28 октября 2008

Это не отладчик, но, вероятно, так же полезен (?)

Я знаю, что слышал, как Гвидо упоминал об этом в своей речи.

Я только что проверил python -?, И если вы используете команду -i, вы можете взаимодействовать, когда ваш скрипт остановился.

Итак, с учетом этого сценария:

testlist = [1,2,3,4,5, 0]

prev_i = None
for i in testlist:
    if not prev_i:
        prev_i = i
    else:
        result = prev_i/i

Вы можете получить этот вывод!

PS D:\> python -i debugtest.py
Traceback (most recent call last):
  File "debugtest.py", line 10, in <module>
    result = prev_i/i
ZeroDivisionError: integer division or modulo by zero
>>>
>>>
>>> prev_i
1
>>> i
0
>>>

Если честно, я не использовал это, но я должен быть, кажется очень полезным.

17 голосов
/ 27 января 2016

IPython делает это простым в командной строке:

python myscript.py arg1 arg2

можно переписать на

ipython --pdb myscript.py -- arg1 arg2

Или, аналогично, при вызове модуля:

python -m mymodule arg1 arg2

можно переписать на

ipython --pdb -m mymodule -- arg1 arg2

Обратите внимание на --, чтобы IPython не мог читать аргументы скрипта как свои собственные.

Это также имеет то преимущество, что вместо pdb вызывается расширенный отладчик IPython (ipdb).

6 голосов
/ 15 января 2017

Если вы используете среду IPython, вы можете просто использовать% debug, и оболочка перенесет вас обратно в вызывающую проблему строку со средой ipdb для проверок и т. Д. Другой вариант, как указано выше, - это использовать iPython magic% pdb. который делает то же самое.

4 голосов
/ 26 июня 2018

Если вы используете ipython, после запуска введите %pdb

In [1]: %pdb
Automatic pdb calling has been turned ON
4 голосов
/ 02 февраля 2011

Вы можете поместить эту строку в свой код:

import pdb ; pdb.set_trace()

Подробнее: Запустить отладчик python в любой строке

3 голосов
/ 12 января 2016

Чтобы запустить его без необходимости вводить c в начале, используйте:

python -m pdb -c c <script name>

Pdb имеет свои собственные аргументы командной строки: -c c выполнит команду c (продолжение) в начале выполнения, и программа будет работать непрерывно до появления ошибки.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...