Jan*_*nus 9 python testing pytest
假设我有一个测试表示为带有assert-statements的简单脚本(请参阅背景以了解原因),例如
import foo
assert foo(3) == 4
Run Code Online (Sandbox Code Playgroud)
我如何将这个脚本包含在我的 pytest 测试套件中——以一种很好的方式?
我尝试了两种有效但不太好的方法:
一种方法是将脚本命名为测试,但这会使整个 pytest 发现在测试失败时失败。
我目前的方法是从测试函数中导入脚本:
def test_notebooks():
notebook_folder = Path(__file__).parent / 'notebooks'
for notebook in notebook_folder.glob('*.py'):
import_module(f'{notebook_folder.name}.{notebook.stem}')
Run Code Online (Sandbox Code Playgroud)
这是有效的,但脚本不会单独报告,并且测试失败有一个长而蜿蜒的堆栈跟踪:
__________________________________________________ test_notebooks ___________________________________________________
def test_notebooks():
notebook_folder = Path(__file__).parent / 'notebooks'
for notebook in notebook_folder.glob('*.py'):
> import_module(f'{notebook_folder.name}.{notebook.stem}')
test_notebooks.py:7:
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
envs\anaconda\lib\importlib\__init__.py:127: in import_module
return _bootstrap._gcd_import(name[level:], package, level)
<frozen importlib._bootstrap>:1006: in _gcd_import
... (9 lines removed)...
<frozen importlib._bootstrap>:219: in _call_with_frames_removed
???
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
> assert False
E AssertionError
notebooks\notebook_2.py:1: AssertionError
Run Code Online (Sandbox Code Playgroud)
我在脚本文件中进行测试的原因是它们实际上是 Jupyter 笔记本.py,由优秀的jupytext插件保存为带有标记的文件。
这些笔记本被转换为 html 文件,可以交互使用来学习系统,并作为廉价的功能测试。
从测试函数调用脚本没有任何问题,因此您的方法非常好。但是,我会使用参数化而不是在 for 循环中运行脚本;这样你就可以很好地为每个脚本执行一次测试。如果你不喜欢长的回溯,你可以在自定义的pytest_exception_interacthookimpl 中剪切它们。例子:
# conftest.py
def pytest_exception_interact(node, call, report):
excinfo = call.excinfo
if 'script' in node.funcargs:
excinfo.traceback = excinfo.traceback.cut(path=node.funcargs['script'])
report.longrepr = node.repr_failure(excinfo)
Run Code Online (Sandbox Code Playgroud)
参数化测试:
# test_spam.py
import pathlib
import runpy
import pytest
scripts = pathlib.Path(__file__, '..', 'scripts').resolve().glob('*.py')
@pytest.mark.parametrize('script', scripts)
def test_script_execution(script):
runpy.run_path(script)
Run Code Online (Sandbox Code Playgroud)
测试执行结果(为了测试,我创建了带有单行的简单脚本,如assert False或1 / 0:
$ pytest -v
======================================= test session starts ========================================
platform linux -- Python 3.6.8, pytest-4.6.3, py-1.8.0, pluggy-0.12.0 -- /home/hoefling/projects/.venvs/stackoverflow/bin/python3.6
cachedir: .pytest_cache
rootdir: /home/hoefling/projects/private/stackoverflow/so-56807698
plugins: mock-1.10.4, cov-2.7.1, forked-1.0.2, xdist-1.28.0, django-3.4.8
collected 3 items
test_spam.py::test_script_execution[script0] PASSED
test_spam.py::test_script_execution[script1] FAILED
test_spam.py::test_script_execution[script2] FAILED
============================================= FAILURES =============================================
____________________________________ test_script_runpy[script1] ____________________________________
> assert False
E AssertionError
scripts/script_3.py:1: AssertionError
____________________________________ test_script_runpy[script2] ____________________________________
> 1 / 0
E ZeroDivisionError: division by zero
scripts/script_2.py:1: ZeroDivisionError
================================ 2 failed, 1 passed in 0.07 seconds ================================
Run Code Online (Sandbox Code Playgroud)
如果你不喜欢上面的解决方案,我能想到的另一件事是实现你自己的测试收集和执行协议。例子:
# conftest.py
import pathlib
import runpy
import pytest
def pytest_collect_file(parent, path):
p = pathlib.Path(str(path))
if p.suffix == '.py' and p.parent.name == 'scripts':
return Script(path, parent)
class Script(pytest.File):
def collect(self):
yield ScriptItem(self.name, self)
class ScriptItem(pytest.Item):
def runtest(self):
runpy.run_path(self.fspath)
def repr_failure(self, excinfo):
excinfo.traceback = excinfo.traceback.cut(path=self.fspath)
return super().repr_failure(excinfo)
Run Code Online (Sandbox Code Playgroud)
这将收集目录中的每个.py文件scripts,将每个脚本包装在一个测试用例中并runpy在测试执行时调用。执行日志看起来几乎相同,只是测试命名不同。