目录:
- 安装及入门
- 使用和调用方法
- 原有TestSuite使用方法
- 断言的编写和报告
- Pytest fixtures:清晰 模块化 易扩展
- 使用Marks标记测试用例
- Monkeypatching,对模块和环境进行Mock
- 使用tmp目录和文件
- 捕获stdout及stderr输出
- 捕获警告信息
- 模块及测试文件中集成doctest测试
- skip及xfail: 处理不能成功的测试用例
- Fixture方法及测试用例的参数化
- 缓存: 使用跨执行状态
- unittest.TestCase支持
- 运行Nose用例
- 安装和使用插件
- 插件编写
- 编写钩子(hook)方法
- API参考
- 优质集成实践
- 片状测试
- Pytest导入机制及sys.path/PYTHONPATH
- 配置选项
- 示例及自定义技巧
- Bash自动补全设置
unittest.TestCase支持
pytest
支持unittest
开箱即用的基于Python 的测试。它旨在利用现有unittest
的测试套件将pytest用作测试运行器,并允许逐步调整测试套件以充分利用pytest的功能。
要使用运行现有unittest
样式的测试套件pytest
,请键入:
pytest tests
pytest将自动收集unittest.TestCase
子类及其test
方法test_*.py
或*_test.py
文件。
几乎所有unittest
功能都受支持:
-
@unittest.skip
风格装饰; -
setUp/tearDown
; -
setUpClass/tearDownClass
; -
setUpModule/tearDownModule
;
到目前为止,pytest不支持以下功能:
- load_tests协议 ;
- 分测验 ;
开箱即用的好处
通过使用pytest运行测试套件,你可以使用多种功能,在大多数情况下无需修改现有代码:
- 获得更多信息性的追溯 ;
- stdout和stderr捕获;
-
使用
-k
和-m
标记测试选择选项 ; - 在第一次(或N次)故障后停止 ;
- -pdb命令行选项,用于调试测试失败(参见下面的注释);
- 使用pytest-xdist插件将测试分发给多个CPU ;
- 使用普通的assert-statements而不是
self.assert*
函数(unittest2pytest对此非常有帮助);
unittest.TestCase
子类中的pytest特性
以下pytest功能适用于unittest.TestCase
子类:
下面pytest功能不工作,也许永远也因不同的设计理念:
第三方插件可能运行也可能不运行,具体取决于插件和测试套件。
unittest.TestCase
使用标记将pytest灯具混合到子类中
运行unittest pytest
允许你使用其 夹具机制进行unittest.TestCase
样式测试。假设你至少浏览了pytest fixture功能,让我们跳转到一个集成pytest db_class
fixture,设置类缓存数据库对象,然后从unittest样式测试中引用它的示例:
# content of conftest.py
# we define a fixture function below and it will be "used" by
# referencing its name from tests
import pytest
@pytest.fixture(scope="class")
def db_class(request):
class DummyDB(object):
pass
# set a class attribute on the invoking test context
request.cls.db = DummyDB()
这定义了一个fixture函数db_class
- 如果使用的话 - 为每个测试类调用一次,并将class-level db
属性设置为一个DummyDB
实例。fixture函数通过接收一个特殊request
对象来实现这一点,该对象允许访问请求测试上下文,例如cls
属性,表示使用该fixture的类。该架构将夹具写入与实际测试代码分离,并允许通过最小参考(夹具名称)重新使用夹具。那么让unittest.TestCase
我们使用fixture定义编写一个实际的类:
# content of test_unittest_db.py
import unittest
import pytest
@pytest.mark.usefixtures("db_class")
class MyTest(unittest.TestCase):
def test_method1(self):
assert hasattr(self, "db")
assert 0, self.db # fail for demo purposes
def test_method2(self):
assert 0, self.db # fail for demo purposes
在@pytest.mark.usefixtures("db_class")
类的装饰可确保pytest固定函数db_class
被调用每一次班。由于故意失败的断言语句,我们可以看看self.db
回溯中的值:
$ pytest test_unittest_db.py
=========================== test session starts ============================
platform linux -- Python 3.x.y, pytest-4.x.y, py-1.x.y, pluggy-0.x.y
cachedir: $PYTHON_PREFIX/.pytest_cache
rootdir: $REGENDOC_TMPDIR
collected 2 items
test_unittest_db.py FF [100%]
================================= FAILURES =================================
___________________________ MyTest.test_method1 ____________________________
self = <test_unittest_db.MyTest testMethod=test_method1>
def test_method1(self):
assert hasattr(self, "db")
> assert 0, self.db # fail for demo purposes
E AssertionError: <conftest.db_class.<locals>.DummyDB object at 0xdeadbeef>
E assert 0
test_unittest_db.py:9: AssertionError
___________________________ MyTest.test_method2 ____________________________
self = <test_unittest_db.MyTest testMethod=test_method2>
def test_method2(self):
> assert 0, self.db # fail for demo purposes
E AssertionError: <conftest.db_class.<locals>.DummyDB object at 0xdeadbeef>
E assert 0
test_unittest_db.py:12: AssertionError
========================= 2 failed in 0.12 seconds =========================
这个默认的pytest回溯显示两个测试方法共享同一个self.db
实例,这是我们在编写上面的类范围的fixture函数时的意图。
使用autouse灯具和访问其他灯具
虽然通常更好地明确声明对给定测试需要使用的灯具,但有时你可能想要在给定的上下文中自动使用灯具。毕竟,传统的unittest-setup风格要求使用这种隐含的夹具编写,而且很有可能,你习惯它或者喜欢它。
你可以使用标记夹具功能@pytest.fixture(autouse=True)
并在要使用它的上下文中定义夹具功能。让我们看一个initdir
夹具,它使一个TestCase
类的所有测试方法都 在一个预先初始化的临时目录中执行samplefile.ini
。我们的initdir
fixture本身使用pytest builtin tmpdir fixture来委托创建一个per-test临时目录:
# content of test_unittest_cleandir.py
import pytest
import unittest
class MyTest(unittest.TestCase):
@pytest.fixture(autouse=True)
def initdir(self, tmpdir):
tmpdir.chdir() # change to pytest-provided temporary directory
tmpdir.join("samplefile.ini").write("# testdata")
def test_method(self):
with open("samplefile.ini") as f:
s = f.read()
assert "testdata" in s
由于该autouse
标志,initdir
fixture函数将用于定义它的类的所有方法。这是@pytest.mark.usefixtures("initdir")
在类中使用标记的快捷方式,如上例所示。
运行此测试模块......:
$ pytest -q test_unittest_cleandir.py
. [100%]
1 passed in 0.12 seconds
...给我们一个通过测试,因为initdir
夹具功能在之前执行test_method
。
注意
unittest.TestCase
方法不能直接接收fixture参数作为实现可能会导致运行通用unittest.TestCase测试套件的能力。
以上usefixtures
和autouse
示例应该有助于将pytest灯具混合到unittest套件中。
你也可以逐步从子类化转移unittest.TestCase
到普通断言 ,然后开始逐步从完整的pytest功能集中受益。
注意
从unittest.TestCase
子类运行测试--pdb
将禁用针对发生异常的情况的tearDown和cleanup方法。这允许对所有在其tearDown机器中具有重要逻辑的应用程序进行适当的事后调试。但是,支持此功能会产生以下副作用:如果人们覆盖unittest.TestCase
__call__
或者run
需要以debug
相同的方式覆盖(对于标准unittest也是如此)。
注意
由于两个框架之间的架构差异,在unittest
测试call
阶段而不是在pytest
标准setup
和teardown
阶段中执行基于测试的设置和拆卸。在某些情况下,这一点非常重要,特别是在推理错误时。例如,如果unittest
基于a 的套件在设置期间出现错误,pytest
则在其setup
阶段期间将报告没有错误, 并且将在此期间引发错误call
。