自动化测试框架设计-04开发pytest插件
- 2022-01-18 09:00:00
- 虫师
- 转贴:
- 公众号
- 2960
本章我们介绍如何开发pytest插件,在上一篇文章中我们介绍了4个python单元测试框架。大概分两类,一类是必须有类继承的,例如 QTAF
和 unittest
, 另一类是可以没有类继承,例如nose/nose2
和 pytest
。对于没有可以没有类继承的框架,开发难度会稍大一些。
pytest扩展能力
如果我们需要给pytest增加额外的扩展能力,那么有三种方式。
1. 钩子函数
利用conftest.py
这个特殊的问题,可以创建钩子函数。
目录结构:
pytest_sample/
├── conftest.py
└── test_sample.py
在conftest.py
文件中实现如下功能。
import pytest
@pytest.fixture
def
hello
():
return
"hello 虫师"
定义一个函数hello()
,并使用pytest.fixture
装饰器对其进行装饰。fixture的概念我们前面已经做介绍。这里fixture
默认的级别为function
,可以理解为被装饰的函数会在每个功能前被执行。
然后,在test_sample.py
测试文件中调动钩子函数。
# 调用钩子函数hello
def
test_case
(hello): print(
"hello:", hello)
assert hello ==
"hello 虫师"
在测试用例中钩子函数hello()
作为测试用例的参数hello
被调用了。断言 hello
函数返回的结果是否为“hello 虫师”
执行用例:
>
pytest -vs test_sample.py ================================= test session starts ===========================
collected 1 item
test_sample.py::test_case hello: hello 虫师
PASSED
================================== 1 passed in 2.61s =============================
2. 用例装饰器
我们比较常用的用例装饰是parametrize
,用法如下。
import pytest
@pytest.mark.parametrize(
'a, b',
[
(
1,
2),
(
2,
3),
(
3,
4),
]
)
def
test_add
(a, b): print(
f'a:
{a}, b:
{b}')
assert a +
1 == b
使用pytest.mark.parametrize()
装饰器,装饰test_add()
测试用例。定义a和b为参数变量,每次取一组数据进行测试。
运行结果
>
pytest -vs test_sample.py ================================= test session starts ===========================
collected 3 items
test_sample.py::test_add[1-2] a:1, b:2
PASSED
test_sample.py::test_add[2-3] a:2, b:3
PASSED
test_sample.py::test_add[3-4] a:3, b:4
PASSED
================================== 3 passed in 2.51s =============================
3.命令行参数
通过pytest
命令执行用例的时候作为参数传值。以pytest
的扩展插件pytest-base-url
为例。
安装
>
pip install pytest-base-url
编写用例如下:
def test_example (base_url):
print( "base_url:", base_url)
assert "http" in base_url
这里同样用到了钩子函数base_url
, 但base_url的参数定义是在执行用例的时候作为参数传入的。
运行测试
>
pytest -vs test_sample.py --base-url https://www.baidu.com ================================= test session starts ===========================
collected 1 item
test_sample.py::test_example base_url: https://www.baidu.com
PASSED
================================== 1 passed in 2.51s =============================
pytest扩展插件
实现pytest-hello插件
我么暂且称这个插件为pytest-hello
。
创建pytest_hello.py
文件,实现代码如下:
import pytest
from typing import Any, Optional
def pytest_configure (config: Any) -> None:
""" register an additional marker
"""
config.addinivalue_line(
"markers", "env(name): mark test to run only on named environment"
)
def pytest_runtest_setup (item: Any) -> None:
"""
Called to perform the setup phase for a test item.
"""
env_names = [mark.args[ 0] for mark in item.iter_markers(name= "env")]
if env_names:
if item.config.getoption( "--env") not in env_names:
pytest.skip( "test requires env in {!r}".format(env_names))
@pytest.fixture(scope="function")
def hello (hello_name: str) -> str:
"""
hello Hook function
"""
return f"hello, {hello_name}"
@pytest.fixture(scope="function") def hello_name (pytestconfig: Any) -> Optional[str]:
"""
hello_name Hook function
"""
names = pytestconfig.getoption( "--hello" )
if len(names) == 0:
return "虫师"
if len(names) == 1:
return names[ 0]
return names[ 0]
def pytest_addoption (parser: Any) -> None:
"""
关键字 :自动化测试,项目管理软件,bug管理,敏捷开发工具
- 联系人:阿道
- 联系方式: 17762006160
- 地址:青岛市黄岛区长江西路118号青铁广场18楼