我有一个固定装置,可以在测试期间创建项目列表。我想要另一个参数化由第一个生成的值的夹具。

范例程式码

import random
import pytest

@pytest.fixture
def values():
    return [random.randint(0, 100) for _ in range(10)]


@pytest.fixture
def value(request):
    return request.param


@pytest.mark.parametrize("value", params=values):
def test_function(value):
    assert value > 0

上面的代码的问题在于values是一个函数而不是一个列表。我做了很多挖掘工作,但没有找到任何方法来拆开夹具来参数化另一个夹具。

我知道我可以通过values固定装置并在测试中对其进行迭代,但这不是一个好的解决方案,因为我想查看导致测试失败的值。

我也对替代解决方案持开放态度,例如,是否有可能从已启动的测试中运行子测试。

最佳答案

这似乎是对夹具概念及其与参数概念的区别的误解。
Pytest有两个主要阶段:

  • 收集阶段,目标是创建要运行的测试“节点”列表。一个测试“节点”对应一个测试ID,表示每个参数一个值。在此阶段,不执行固定装置,仅读取装饰标记(包含参数)。因此,只有在修饰符中声明为的参数才能影响此阶段。
  • 一个执行阶段,其中运行每个测试节点。在运行之前,所有尚未安装的必需夹具都已安装。因此,夹具功能仅在此阶段中执行。他们的结果无法修改上一阶段已完成的测试列表。

  • 在您的示例中,您希望夹具设置的结果(阶段B)更改要创建的测试列表(阶段A):这在设计上是不可能的。您必须在其他地方创建此列表,例如在init hook中的pytest conftest.py中创建该列表,或者在任何测试模块中简单地将其创建为共享变量,然后在测试或夹具的参数中引用该列表。
    另请参见这个非常相似的问题:Parametrizing tests depending of also parametrized values in pytest
    请注意,作为对hoefling's comment to your question的补充,您现在可以在参数列表中使用经过参数设置的夹具:我已经在pytest-cases插件中添加了此功能以进行评估,以便最终我们可以建议将其合并到pytest中(请参阅this discussion,因此请随时提供反馈!)。不幸的是,由于上述基本原因,这不能解决您在本文中描述的确切问题。

    关于Python Pytest解压夹具,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/50231627/

    10-13 09:36