参数化装置和测试方法

pytest 在多个级别启用测试参数化:

  • pytest.fixture() 允许参数化装置方法。
  • @pytest.mark.parametrize 允许在测试函数或类中定义多组参数和装置。
  • pytest_generate_tests 允许定义自定义参数化方案或扩展。

@pytest.mark.parametrize:参数化测试函数

内置的 pytest.mark.parametrize 装饰器可以对测试函数的参数进行参数化。下面是一个测试函数的典型示例,该函数执行检查某个输入是否会导致预期输出:

# content of test_expectation.py
import pytest


@pytest.mark.parametrize("test_input,expected", [("3+5", 8), ("2+4", 6), ("6*9", 42)])
def test_eval(test_input, expected):
    assert eval(test_input) == expected

在这里,@parametrize 装饰器定义了三个不同的 (test_input,expected) 元组,以便 test_eval 函数将依次使用它们运行 3 次:

$ pytest
=========================== test session starts ============================
platform linux -- Python 3.x.y, pytest-6.x.y, py-1.x.y, pluggy-0.x.y
cachedir: $PYTHON_PREFIX/.pytest_cache
rootdir: $REGENDOC_TMPDIR
collected 3 items

test_expectation.py ..F                                              [100%]

================================= FAILURES =================================
____________________________ test_eval[6*9-42] _____________________________

test_input = '6*9', expected = 42

    @pytest.mark.parametrize("test_input,expected", [("3+5", 8), ("2+4", 6), ("6*9", 42)])
    def test_eval(test_input, expected):
>       assert eval(test_input) == expected
E       AssertionError: assert 54 == 42
E        +  where 54 = eval('6*9')

test_expectation.py:6: AssertionError
========================= short test summary info ==========================
FAILED test_expectation.py::test_eval[6*9-42] - AssertionError: assert 54...
======================= 1 failed, 2 passed in 0.12s ========================

Note: 参数值按原样传递给测试(没有任何副本)。 例如,如果您将列表或字典作为参数值传递,并且测试用例代码对其进行了更改,则更改将反映在后续的测试用例调用中。

Note:默认情况下,pytest 会转义 unicode 字符串中用于参数化的任何非 ascii 字符,因为它有几个缺点。但是,如果您想在参数化中使用 unicode 字符串并在终端中按原样(非转义)查看它们,请在 pytest.ini 中使用此选项:

[pytest]
disable_test_id_escaping_and_forfeit_all_rights_to_community_support = True

但是请记住,这可能会导致不必要的副作用甚至错误,具体取决于所使用的操作系统和当前安装的插件,因此请自行承担风险。

如本示例中设计的那样,只有一对输入/输出值无法通过简单的测试功能。和通常使用测试函数参数一样,您可以在跟踪信息中看到输入和输出值。

请注意,您还可以在类或模块上使用参数化标记(请参阅使用属性标记测试函数),这将使用参数集调用多个函数。

也可以在参数化中标记单个测试实例,例如使用内置的 mark.xfail:

# content of test_expectation.py
import pytest


@pytest.mark.parametrize(
    "test_input,expected",
    [("3+5", 8), ("2+4", 6), pytest.param("6*9", 42, marks=pytest.mark.xfail)],
)
def test_eval(test_input, expected):
    assert eval(test_input) == expected

让我们执行它:

$ pytest
=========================== test session starts ============================
platform linux -- Python 3.x.y, pytest-6.x.y, py-1.x.y, pluggy-0.x.y
cachedir: $PYTHON_PREFIX/.pytest_cache
rootdir: $REGENDOC_TMPDIR
collected 3 items

test_expectation.py ..x                                              [100%]

======================= 2 passed, 1 xfailed in 0.12s =======================

之前导致失败的一个参数集现在显示为“xfailed”(预期失败)测试。

如果提供给参数化的值导致空列表 - 例如,如果它们是由某个函数动态生成的 - pytest 的行为由 empty_parameter_set_mark 选项定义。

要获得多个参数化参数的所有组合,您可以堆叠参数化装饰器:

import pytest


@pytest.mark.parametrize("x", [0, 1])
@pytest.mark.parametrize("y", [2, 3])
def test_foo(x, y):
    pass

这将在参数设置为 x=0/y=2、x=1/y=2、x=0/y=3 和 x=1/y=3 的情况下运行测试,按照装饰器的顺序耗尽参数.

基本 pytest_generate_tests 示例

有时您可能想要实现自己的参数化方案或实现一些动态来确定装置的参数或范围。为此,您可以使用在收集测试函数时调用 pytest_generate_tests 钩子。通过传入的 metafunc 对象,您可以检查请求的测试上下文,最重要的是,您可以调用 metafunc.parametrize() 来进行参数化。

例如,假设我们想要通过新的 pytest 命令行选项设置字符串输入来运行测试。让我们首先编写一个接受 stringinput 装置函数作为参数的简单测试:

# content of test_strings.py


def test_valid_string(stringinput):
    assert stringinput.isalpha()

现在我们添加一个 conftest.py 文件,其中包含添加的命令行选项和测试函数的参数化:

# content of conftest.py


def pytest_addoption(parser):
    parser.addoption(
        "--stringinput",
        action="append",
        default=[],
        help="list of stringinputs to pass to test functions",
    )


def pytest_generate_tests(metafunc):
    if "stringinput" in metafunc.fixturenames:
        metafunc.parametrize("stringinput", metafunc.config.getoption("stringinput"))

如果我们现在传递两个 stringinput 值,我们的测试将运行两次:

$ pytest -q --stringinput="hello" --stringinput="world" test_strings.py
..                                                                   [100%]
2 passed in 0.12s

让我们也运行一个会导致测试失败的字符串输入:

$ pytest -q --stringinput="!" test_strings.py
F                                                                    [100%]
================================= FAILURES =================================
___________________________ test_valid_string[!] ___________________________

stringinput = '!'

    def test_valid_string(stringinput):
>       assert stringinput.isalpha()
E       AssertionError: assert False
E        +  where False = <built-in method isalpha of str object at 0xdeadbeef>()
E        +    where <built-in method isalpha of str object at 0xdeadbeef> = '!'.isalpha

test_strings.py:4: AssertionError
========================= short test summary info ==========================
FAILED test_strings.py::test_valid_string[!] - AssertionError: assert False
1 failed in 0.12s

正如预期的那样,我们的测试用例失败了

如果您不指定 stringinput 它将被跳过,因为 metafunc.parametrize() 将使用空参数列表调用:

$ pytest -q -rs test_strings.py
s                                                                    [100%]
========================= short test summary info ==========================
SKIPPED [1] test_strings.py: got empty parameter set ['stringinput'], function test_valid_string at $REGENDOC_TMPDIR/test_strings.py:2
1 skipped in 0.12s

请注意,当使用不同的参数集多次调用 metafunc.parametrize 时,这些集合中的所有参数名称不能重复,否则将引发错误。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值