dirtyhand-tester Pytest 顺序执行,依赖执行,参数化执行

simonpatrick · 2024年05月14日 · 最后由 大卡卡 回复于 2024年06月01日 · 7892 次阅读

Pytest 顺序执行,依赖执行,参数化执行

进行下面实验前,需要安装 python 和 pytest,我一般用 poetry 来管理,如何创建就是以下几个命令:

poetry init 
poetry add pytest --group test

0- 被测对象

假设被测试对象就是一个计算加减乘除的一个服务,这里把一个方法就等同于一个 API 接口,实际上也确实没有
太大区别,一个是本地调用,一个是远程调用,远程调用需要一个客户端,本地调用直接就用这个类调用了

"""
service functions for calculation
"""


def add(a, b):
    return a + b


def subtract(a, b):
    return a - b


def multiply(a, b):
    return a * b


def divide(a, b):
    return a * 1.0 / b

1. 使用 pytest 进行测试这个服务的常见场景和解法

  1. 单独测试一个方法
  2. 测试方法 B 但是需要先执行测试方法 A
  3. 参数化测试,参数可能是静态写死的,也可能是动态的

1.1 pytest-单独测试一个方法

使用如下方法就可以进行单独的方法测试,基本原则就是: 一个方法一个测试,测试数据都固定

import allure
import pytest

from app.module_a.service import subtract, add


@allure.feature("减法结果为正")
def test_subtract_positive():
    result = subtract(2, 1)
    assert result == 1


@allure.feature("减法结果为为负数")
def test_subtract_negative():
    result = subtract(1, 2)
    assert result == -1

1.2 Pytest-测试方法 B 但是需要先执行测试方法 A

如果遇到: 测试方法 B 但是需要先执行测试方法 A,这里面有一些小问题就是:

  1. 顺序不能绝对保证: 如果通过运行 pytest 的命令行时候,有可能不能确保测试 A 一定跑在测试 B 前面
  2. 测试可读性问题: s 不仔细看代码,不能看出来 - 测试方法 B 但是需要先执行测试方法 A

这个问题可以通过 pytest 的插件 pytest-order 来解决:

poetry add pytest-order

测试代码如下: 通过@pytest.mark.order定于显示的定于运行顺序,好处有两个:

  1. 确保顺序
  2. 提高测试可读性,一眼就看出来那个先跑那个后跑
  3. 不好的地方是: 单独运行第二个测试: test_subtract_order 会报错,但是因为写了 order 标识, 大概一下子也能明白怎么回事
context = {}


@pytest.mark.order(1)
def test_add_positive():
    result = add(1, 2)
    assert result == 3
    context['USED_FOR_SUBTRACT'] = result


@allure.feature("被减数是加法函数计算出来的结果")
@pytest.mark.order(2)
def test_subtract_order():
    result = subtract(1, context['USED_FOR_SUBTRACT'])
    assert result == -2

如果使用 pytest-dependency 插件解决,代码也类似,就不多说明了:

context = {}

@pytest.mark.order(1)
@user8cy()
def test_add_positive():
    result = add(1, 2)
    assert result == 3
    context['USED_FOR_SUBTRACT'] = result

@user9cy(depends=["test_add_positive"])
def test_subtract_dep():
    result = subtract(1, context['USED_FOR_SUBTRACT'])
    assert result == -2

这里面注意一点: 用来存数据的用字典会好那么一点,为了避免一些想不到的麻烦我自己就一直用字典了,也没有用 global 这样的东西。

1.3 pytest - 参数化执行

  • 有时很多 case 都很类似,只要给数据自动跑就行了,不想写很多单独的测试方法了,就用参数化执行, 最简单的就是直接给上固定的数据, 以下代码一看就明白
fixed_params = [(1, 2, 3), (-1, 1, 0)]


@user10ize('a,b,expected', fixed_params)
def test_add_parameterized(a, b, expected):
    actual = add(a, b)
    assert actual == expected
  • 有时呢,参数化的时候有一两个数字是需要动态获得以下的,那么可以用以下方式简单处理以下, 就是写一个函数区构建这些参数化的测试数据
def dynamic_cases():
    d_params = [(1, 2, 3), (-1, 1, 0)]
    dynamic_case = (1, add(1, 2), 4)
    d_params.append(dynamic_case)
    return d_params


@user11ize('a,b,expected', dynamic_cases())
def test_add_parameterized(a, b, expected):
    actual = add(a, b)
    assert actual == expected

2. 使用的一些小结

有时可能更复杂,那么其实呢,我自己的做法就是:

  1. 混合使用顺序 order/依赖空 dependency 和参数化化进行处理
  2. 能参数化的和复杂的 case 分开,比如: A. 一些异常错误数据,出错信息的测试,这些参数化可能容易一些就参数化了 B. 复杂场景的,计算,拿前一个返回再构建后一个的人参之类的,干脆就单独写一个测试方法了
  3. 不去想一个统一的解法,写代码和在测试平台上写测试用例的最大的区别就是:测试平台一般想用一种方法解决所有问题, 而自己写代码的坏,就是完全自己控制灵活度, 各有好坏,既然自己写代码了, 那就多多利用写代码的灵活度解决自己的问题,有时找一个可以适用所有的方法可能很麻烦,不如直接手写一个测试方法来的简单直接 还节约一点时间。
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
共收到 7 条回复 时间 点赞
耿晓 回复

这个不错,这个真不错

耿晓 回复

可以可以很棒的分享 感谢

耿晓 回复

你这代码不还是没有指定顺序执行吗, 只是把模块和用例的执行顺序用 ID 记录了, 执行顺序的那个 order 值怎么指定呢;
还是说在 allure 的 feature 和 story 中加入生成器, 就会自动按照顺序执行了..

pytest 的执行顺序不是从上往下吗?第 2 点直接把前置的测试方法放在前面就行了吧?

你说得对哦铁铁,使用生成器确实不能解决作者的问题,使用生成器只能解决写死数字致使的后续编辑时需要重新编辑排序数字的问题,并且是在特定的顺序执行的情况下才能达到预期效果。
那有什么好的办法可以解决及不想将排序写死,还能控制用例执行顺序的方法呢?

这个我有一定经验,不知道符合不符合你们要求。之前我测试路由器的时候需要添加 ACL 规则,他也是一个有编号的,在添加的时候不是顺序的使用 1,2,3,4,而是实有 5,10,15,20 这样的编号或者是 10,20,30,40,这样后面需要插入的时候就有空闲调整顺序了。

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册