pytest


Pytest测试框架

  • 日志模块: pytest.ini 配置日志, requests封装logging模块可以实现
  • 通知机器人: 钉钉获取webhook链接, jenkins安装dingtalk插件, 配置webhook
  • 其他特性- 钩子函数: 应用场景**pytest_terminal_summary**实现测试结果收集
  • 辅助函数, 环境变量: fixture函数

命令行参数

# -x
python -m pytest    # 全部执行
python -m pytest  -x     # 1次失败停止
python -m pytest  --maxfail 2    # 2次失败停止
# -k
python -m pytest -k 'Login and not ft' # 执行Login相关的测试用例, 单不执行带有ft的
# -q
python -m pytest -q   # 控制条输出简化信息
# -v
python -m pytest -v    # 控制台输出详细信息
# 指定路径执行
python -m pytest .\testcases\trina_union\test_m2.py::Test1::test_m1
# -m
python -m pytest -m pre #只执行带有pre标记的
# --lf 
python -m pytest --lf    # 只运行上次失败的用例
# -n
python -m pytest -n 5    # 多线程执行, 依赖pytest-xdist
# --reruns
python -m pytest --reruns 2    # 失败重试, 依赖pytest-rerunfailures
# -s
python -m putest -s # 显示print打印内容
# --pdb
python -m pytest --pdb # 用例失败进入pdb模式, 可查看变量

# --html
python -m pytest .\testcases\ --html  ./reports/report.html # 生成html测试报告(静态简陋)

# --alluredir
pytest testcases --alluredir=./reports"    # 生成allure报告

Setup_teardown

# 函数级别的初始化和收尾
class Test1:
    def teardown_method(self):
        logging.info("TearDown----")

    def setup_method(self):
        logging.info('Test1---setup_method')
# 类级别的初始化和收尾
class Test1:

    def setup_class():
        logging.info("setup_class----")

    def teardown_class():
        logging.info("teardown_class------")
        
# 模块级别的初始化和收尾
def setup_module():
    logging.info("setup_module----")

def teardown_module():
    logging.info("teardown_module------")

fixture

  • 设置在conftest.py或者测试用例文件中
  • 用于setip, teardown及返回测试数据
# scope: 影响层级
# autouse: 自动生效,无需测试用例引用
@pytest.fixture(scope='function', autouse=True)
def f1():
    logging.info("------fixture setup-------")
    yield 'fixture'
    logging.info("------fixture teardown-------")

class Test1:
    def test_m2(self, f1):
        logging.info(f1)    # 通过fixture函数名获取 yield返回值 'fixture'

@pytest.mark.usefixtures('f2')    # 引用fixture的方式2
def test_m1(self):
    logging.info('Exe-----testm2')
    assert 1==1

# params: 数据驱动, 循环返回列表数据
@pytest.fixture(params=[1,2,3])
def fp(request):
    yield request.param
   
class Test2:
    def test_m1(self, fp):
        logging.info(fp)    # 执行三次, 分别获取到 1, 2, 3
        
 
  

装饰器

# 数据驱动parametrize
class Test1:
    @pytest.mark.parametrize("v1, v2",
                             [
                                 (1, 9),
                                 (2, 8),
                                 (3, 7)
                             ])
    def test_m2(self, f1, v1, v2):
        logging.info(f1)
        logging.info(f"v1:{v1},v2:{v2}")
        assert 1==1
# 跳过 skip
@pytest.mark.skip(reason='for test')
def test_m1(self):
    logging.info('Exe-----testm2')
    assert 1==1
    
# 条件跳过skipif
@pytest.mark.skipif(condition='check_db(1)')
def test_m1(self):
    logging.info('Exe-----test_m1')
    
# mark: 用例标记
@pytest.mark.pre
class Test2:
    def test_m1(self):
        logging.info('Exe-----test_m1')     
# xfail 预期失败用例

Author: Feny Lau
Reprint policy: All articles in this blog are used except for special statements CC BY 4.0 reprint policy. If reproduced, please indicate source Feny Lau !
  TOC