自定义pytest参数化测试名称

Customizing pytest parameterized test name

我进行了以下测试:

@pytest.mark.parametrize(
    "nums",
    [[3, 1, 5, 4, 2], [2, 6, 4, 3, 1, 5], [1, 5, 6, 4, 3, 2]]
)
def test_cyclic_sort(nums):
    pass


@pytest.mark.parametrize(
    "nums, missing",
    [([4, 0, 3, 1], 2)]
)
def test_find_missing_number(nums, missing):
    pass

我想自定义测试名称以包含输入数组。我已阅读 pytest docs, and and this question,但 none 回答以下问题:

  1. 传递给 id 函数的是什么?在我上面的代码中,第一个测试采用一个参数,第二个采用两个参数。
  2. pytest 文档对 id 使用顶级函数,而我想将我的测试放在 class 中并使用 @staticmethod。尝试从 TestClass 内部使用 TestClass.static_method 引用静态方法会在 PyCharm 中产生错误;这样做的正确语法是什么?

编辑: 创建 https://github.com/pytest-dev/pytest/issues/8448.

当为 ids 关键字使用可调用对象时,将使用单个参数调用它:被参数化的测试参数的值。 callable ids return 一个字符串,将在方括号中用作测试名称后缀。

如果测试对多个值进行参数化,该函数仍将使用单个参数调用,但每个测试将调用多次。生成的名称将用破折号连接,类似于

"-".join([idfunc(val) for val in parameters])

例如:

test_something[val1-val2-val3]

Here is the join in the pytest source.

要使用静态方法,此语法有效:

class TestExample:

    @staticmethod
    def idfunc(val):
        return f"foo{val}"

    @pytest.mark.parametrize(
        "x, y",
        [
            [1, 2],
            ["a", "b"],
        ],
        ids=idfunc.__func__,
    )
    def test_vals(self, x, y):
        assert x
        assert y

这将生成两个测试,如上所述调用 idfunc 四次。

TestExample::test_vals[foo1-foo2]
TestExample::test_vals[fooa-foob]