前言

上一篇我们介绍了Pytest是如何进行参数化的,这一篇我们将给大家介绍的一些常用标记。

skip 和 xfail 标记

实际工作中,测试用例的执行可能会依赖于一些外部条件,例如:只能运行在某个特定的操作系统(Windows),或者我们本身期望它们测试失败,例如:被某个已知的Bug所阻塞;如果我们能为这些用例提前打上标记,那么pytest就相应地预处理它们,并提供一个更加准确的测试报告。
pytest 比较常用的标记有:

skip:只有当某些条件得到满足时,才执行测试用例,否则跳过整个测试用例的执行;例如,在非Windows平台上跳过只支持Windows系统的用例;xfail:因为一个确切的原因,我们知道这个用例会失败;例如,对某个未实现的功能的测试,或者阻塞于某个已知Bug的测试;

pytest默认不显示skip和xfail用例的详细信息,但我们可以通过 -r 选项来自定义这种行为:通常,我们使用一个字母作为一种类型的代表,具体的规则如下:

(f)ailed, (E)rror, (s)kipped, (x)failed, (X)passed, (p)assed, (P)assed with output, (a)ll except passed(p/P), or (A)ll

如显示结果为XFAIL、XPASS和SKIPPED的用例:

pytest -rsxX

skip 跳过测试用例执行

1.@pytest.mark.skip装饰器 跳过执行某个用例最简单的方式就是使用@pytest.mark.skip装饰器,并且可以设置一个可选参数reason,表明跳过的原因

@pytest.mark.skip(reason="no way of currently testing this")def test_the_unknown():    ...

1.@pytest.mark.skipif装饰器 如果我们想有条件的跳过某些测试用例的执行,可以使用@pytest.mark.skipif装饰器

import sys,pytest@pytest.mark.skipif(sys.version_info < (3, 6), reason="requires python3.6 or higher")def test_function():    ...

在大型的测试项目中,可以在一个文件中定义所有的执行条件,需要时就引入到模块中

xxx.pyminversion = pytest.mark.skipif(sys.version_info < (3, 8),                                reason='请使用 python 3.8 或者更高的版本。')

引用模块

from xxx import minversion@minversiondef test_two():    pass

1.跳过测试类 在类上应用@pytest.mark.skip或@pytest.mark.skipif, 会作用于类中的每一个用例

import pytest@pytest.mark.skip("作用于类中的每一个用例,所以 pytest 共收集到两个 SKIPPED 的用例。")class TestMyClass():    def test_one(self):        assert True    def test_two(self):        assert True

1.跳过测试模块

import pytestpytestmark = pytest.mark.skip('作用于模块中的每一个用例,所以 pytest 共收集到两个 SKIPPED 的用例。')def test_one():    assert Truedef test_two():    assert True

xfail 标记用例为预期失败的

1.我们可以使用@pytest.mark.xfail标记用例,表示期望这个用例执行失败;
用例会正常执行,只是失败时不再显示堆栈信息,最终的结果有两个:用例执行失败时(xfailed:符合预期的失败)、用例执行成功时(xpassed:不符合预期的成功)

@pytest.mark.xfaildef test_func():    assert func(5) == 6if __name__ == '__main__':    pytest.main(['-s', 'test_para.py'])

1.我们也可以通过pytest.xfail方法在用例执行过程中直接标记用例结果为XFAIL,并跳过剩余的部分:

def test_func():    print('111')    pytest.xfail("it's not ok")    print(222)============================= test session starts ==============================platform darwin -- Python 3.7.3, pytest-6.0.1, py-1.9.0, pluggy-0.13.1rootdir: /Users/zhangcheng/Documents/project/7.18/test_pycollected 1 itemtest_para.py 111x============================== 1 xfailed in 0.06s ==============================

1.xfail 它也可以接收一个python表达式,表明只有满足条件时才标记用例;

@pytest.mark.xfail(sys.version_info >= (3, 6), reason="python3.6 api changes")def test_function():    ...

1.结合pytest.param方法

import pytestimport sys@pytest.mark.parametrize(    ('n', 'expected'),    [(2, 1),     pytest.param(2, 1, marks=pytest.mark.xfail(), id='XPASS'),     pytest.param(0, 1, marks=pytest.mark.xfail(raises=ZeroDivisionError), id='XFAIL'),     pytest.param(1, 2, marks=pytest.mark.skip(reason='无效的参数,跳过执行')),     pytest.param(1, 2, marks=pytest.mark.skipif(sys.version_info <= (3, 8), reason='请使用3.8及以上版本的python。'))])def test_params(n, expected):    assert 2 / n == expectedpytest -s -rsxXp test_para.py====================================================== short test summary info ======================================================SKIPPED [1] test_para.py:31: 无效的参数,跳过执行SKIPPED [1] test_para.py:31: 请使用3.8及以上版本的python。XFAIL test_para.py::test_params[XFAIL]XPASS test_para.py::test_params[XPASS] PASSED test_para.py::test_params[2-1]

生成测试报告

Pytest 也可以很方便的生成测试报告,但前提是要先安装pytest-html插件。

1.安装 pytest-html 插件

pip install pytest-html

1.新建run.py (在新目录下):

if __name__ == '__main__':    pytest.main(['-s', '-v', '../test_case', '--html=../reports/report.html'])

好了,经过这5篇的讲解,Pytest 的常用方法已经介绍完毕,当然,大家在使用过程中肯定还会遇到许多细节问题,这就要求大家自己去尝试解决,加深理解,这样才能更好的掌握Pytest这一框架。

谢谢大家!

鲲鹏!