84

我正在使用 py.test 对我的 python 程序进行单元测试。我希望以正常方式使用 python 调试器调试我的测试代码(我的意思是代码中的 pdb.set_trace() ),但我无法让它工作。

将 pdb.set_trace() 放入代码中不起作用(引发 IOError:在捕获输出时从标准输入读取)。我也尝试过使用 --pdb 选项运行 py.test ,但如果我想探索断言之前发生的事情,这似乎并不能解决问题。当断言失败时它会中断,并且从该行继续意味着终止程序。

有谁知道调试的方法,或者调试和 py.test 只是不打算在一起?

4

7 回答 7

128

这真的很简单:assert 0在你的代码中放置一个你想开始调试的地方并运行你的测试:

py.test --pdb 

完毕 :)

或者,如果您使用的是 pytest-2.0.1 或更高版本,pytest.set_trace()您还可以将助手放在测试代码的任何位置。这是文档。在将您发送到 pdb 调试器命令行之前,它会注意在内部禁用捕获。

于 2010-07-15T09:09:50.980 回答
43

我发现我可以在禁用捕获的情况下运行 py.test,然后像往常一样使用 pdb.set_trace()。

> py.test --capture=no
============================= test session starts ==============================
platform linux2 -- Python 2.5.2 -- pytest-1.3.3
test path 1: project/lib/test/test_facet.py

project/lib/test/test_facet.py ...> /home/jaraco/projects/project/lib/functions.py(158)do_something()
-> code_about_to_run('')
(Pdb)
于 2010-08-05T19:49:50.780 回答
25

最简单的方法是使用 py.test 机制来创建断点

http://pytest.org/latest/usage.html#setting-a-breakpoint-aka-set-trace

import pytest
def test_function():
    ...
    pytest.set_trace()    # invoke PDB debugger and tracing

或者,如果您想将pytest's 调试器用作单行程序,请将您的更改import pdb; pdb.set_trace()import pytest; pytest.set_trace()

于 2013-07-10T23:08:49.553 回答
3

类似于 Peter Lyon 的回答,但使用 pytest 所需的确切代码,您可以将以下内容添加到 pytest 模块 (my_test_module.py) 的底部:

if __name__ == "__main__":
    pytest.main(["my_test_module.py", "-s"])

然后您可以从命令行调用调试器:

pdb3 my_test_module.py

繁荣。您在调试器中并且能够输入调试器命令。此方法使您的测试代码不会被 set_trace() 调用弄乱,并将在 pytest 中“正常”运行。

于 2020-05-06T12:57:12.050 回答
2

我对py.test不熟悉,放到unittest中,你做如下。也许 py.test 是类似的:

在您的测试模块(mytestmodule.py)中:

if __name__ == "__main__":
    unittest.main(module="mytestmodule")

然后运行测试

python -m pdb mytestmodule.py

您将获得一个交互式 pdb shell。

查看文档,看起来 py.test 有一个--pdb命令行选项:

http://codespeak.net/py/dist/test/features.html

于 2010-04-21T05:18:08.967 回答
1

在不编辑源文件的情况下添加和删除断点

虽然您可以通过在代码中添加breakpoint()orset_trace()语句来添加断点,但这种方法存在两个问题:

  • 首先,一旦开始运行代码,就无法删除断点。我经常发现,一旦我开始运行我的代码并到达一个初始断点,我想放置另一个断点并删除初始断点。将我放入调试器后breakpoint(),我可以添加额外的断点,但我无法删除最初的断点。虽然可以通过将初始breakpoint语句放在更高的位置来缓解这种情况,但如果您有参数化测试,那么即使是有限的。我可能会发现自己cont经常重复。
  • 其次,它需要更改源代码。您需要记住breakpoint()在将任何代码提交到版本控制之前删除所有命令,您必须在切换分支之前删除它们等。我有时发现我想使用调试器来比较两个分支之间的测试运行,并且必须编辑源代码每次都添加断点的代码会使该操作变得相当慢且更容易出错。我什至可能想在我正在调用的库中添加一个断点,在这种情况下,我正在编辑的文件甚至可能不在我的 git 存储库中,而是在我的 conda 环境深处,增加了忘记删除它的风险。在我看来,编辑文件以添加断点是丑陋的。

要在不编辑任何源文件的情况下以交互方式添加和删除pytest断点,可以按如下方式调用(在 bash shell 中):

python -mipdb $(type -p pytest) -s test_fileset.py

-s标志在这里至关重要,因为它阻止 pytest 与 stdin 和 stdout 混淆,并且当在调试器中运行时,pytest 将无法与 stdin 和 stdout 混淆,一切都会出错。对于不同的 shell,确切的调用语法会有所不同。

于 2020-10-07T10:20:07.273 回答
0

只需使用:pytest --trace test_your_test.py。这将在测试开始时调用 Python 调试器

于 2021-04-06T18:20:31.570 回答