pytest是python的单测框架,使用灵活,插件丰富,以下是pytest常用的插件
pip install 就行,只有有这些插件,下面的某些命令行才生效
使用pytest.ini文件可以修改默认规则
import pytest
if __name__ == '__main__':
pytest.main(["-vs", "./test_demo/test_demo1.py"])
这样就可以运行所有用例
pytest -vs ./test_demo/test_demo1.py
不论是主函数模式还是命令行模式都会读取这个配置文件,该文件需要使用gbk编码,下面是这个配置文件的例子
[pytest]
# 命令行参数,用空格分隔
addopts = -vs
# 测试用例文件夹,可以自己配置
testpaths = ./test_demo
# 配置测试搜索的模块文件名称
python_files = test*.py
# 配置测试搜索的类名
python_classes = Test*
# 配置搜索的函数名
python_functions = test
定义三个组,冒烟:smoke,用户管理:user_manager,作业管理:worker_manager
import pytest
class TestDemo:
@pytest.mark.somke
def test_case1(self):
print("1")
@pytest.mark.user_manage
def test_case2(self):
print("2")
@pytest.mark.worker_manage
def test_case3(self):
print("3")
markers =
smoke:冒烟测试
user_manage:用户管理
worker_manage:作业管理
import pytest
if __name__ == '__main__':
pytest.main(["-vs", "-m smoke or usermanage"])
运行单组
import pytest
if __name__ == '__main__':
pytest.main(["-vs", "-m smoke"])
直接使用装饰器@pytest.mark.skip(reason=“原因填写”)
使用装饰器@pytest.mark.skipif(条件, 原因)
例子:
import pytest
class TestDemo:
age = 18
@pytest.mark.smoke
def test_case1(self):
print("1")
@pytest.mark.usermanage
@pytest.mark.skipif(age < 18, "未成年")
def test_case2(self):
print("2")
@pytest.mark.workermanage
@pytest.mark.skip(reason="原因填写")
def test_case3(self):
print("3")
为什么需要前后置?比如执行用例前需要做一些准备工作,比如打开浏览器,在执行用例后需要一些后置工作,比如关闭浏览器
在每个模块执行前会调用setup_module方法,在每个模块执行后会使用teardown_module方法。
例子:
import pytest
def setup_module():
print("模块用例前执行")
def teardown_module():
print("模块用例后执行")
class TestDemo:
def test_case1(self):
print("1")
def test_case2(self):
print("2")
def test_case3(self):
print("3")
class TestDemo2:
def test_case4(self):
print("4")
结果:
test_demo/test_demo2.py::TestDemo::test_case1 模块用例前执行
1
PASSED
test_demo/test_demo2.py::TestDemo::test_case2 2
PASSED
test_demo/test_demo2.py::TestDemo::test_case3 3
PASSED
test_demo/test_demo2.py::TestDemo2::test_case4 4
PASSED模块用例后执行
类级别函数 setup_class/teardown_class 对类有效,位于类中,在测试类中前后调用一次。
class TestDemo:
def setup_class(self):
print("类级别前置")
def test_case1(self):
print("1")
def test_case2(self):
print("2")
def test_case3(self):
print("3")
def teardown_class(self):
print("类级别后置")
test_demo/test_demo2.py::TestDemo::test_case1 模块用例前执行
类级别前置
1
PASSED
test_demo/test_demo2.py::TestDemo::test_case2 2
PASSED
test_demo/test_demo2.py::TestDemo::test_case3 3
PASSED类级别后置
模块用例后执行
方法级别函数 setup_method/teardown_method和setup/teardown对类有效,也位于类中,这两个效果一样,在测试类中每个测试方法前后调用一次。
class TestDemo:
def setup_method(self):
print("方法级别前置")
def test_case1(self):
print("1")
def test_case2(self):
print("2")
def test_case3(self):
print("3")
def teardown_method(self):
print("方法级别后置")
test_demo/test_demo3.py::TestDemo::test_case1 方法级别前置
1
PASSED方法级别后置
test_demo/test_demo3.py::TestDemo::test_case2 方法级别前置
2
PASSED方法级别后置
test_demo/test_demo3.py::TestDemo::test_case3 方法级别前置
3
PASSED方法级别后置
import pytest
@pytest.fixture(scope="function", params=["1", "2", "3"], autouse=False, ids=None, name="new_name")
def my_feature(request):
i = request.param
print("前置")
yield i
print("后置")
class TestDemo:
def test_case1(self, new_name):
print(new_name)
print("1")
结果
test_demo/test_demo4.py::TestDemo::test_case1[1] 前置
1
1
PASSED后置
test_demo/test_demo4.py::TestDemo::test_case1[2] 前置
2
1
PASSED后置
test_demo/test_demo4.py::TestDemo::test_case1[3] 前置
3
1
PASSED后置
fixture为session级别是可以跨.py模块调用的,也就是当我们有多个.py文件的用例的时候,如果多个用例只需调用一次fixture,那就可以设置为scope=“session”,并且写到conftest.py文件里。
conftest.py文件名称时固定的,pytest会自动识别该文件。放到项目的根目录下就可以全局调用了,如果放到某个package下,那就在改package内有效。
例子:
在包下创建conftest.py,注意,该配置只在本包生效
和之前一样使用
结果还是和之前一样。
pytest -vs --html ./report/report.html
主要用的装饰器是@pytest.mark.parametrize(argnames, argvalues)
import pytest
class TestDemo:
@pytest.mark.parametrize("args",[(4399, 'AAAA'), (2012, 'BBBB')])
def test_case1(self, args):
print(args)
import pytest
class TestDemo:
@pytest.mark.parametrize("arg1,arg2", [(4399, 'AAAA'), (2012, 'BBBB')])
def test_case1(self, arg1, arg2):
print(arg1, arg2)