【问题标题】:pytest: Mark a test as a "must pass" and stop testing if it failspytest:将测试标记为“必须通过”,如果失败则停止测试
【发布时间】:2020-04-10 06:16:49
【问题描述】:

我正在尝试实现一个新的pytest 标记,称为@pytest.mark.must_pass,以指示如果标记的测试失败,pytest 应该跳过所有后续测试并终止。

如果标记的测试失败,我已经能够使用pytest_runtest_call 挂钩让 pytest 终止,但我使用的是pytest.exit,它不会打印回溯,也不会显示测试的失败指示有问题。

我需要将此失败显示为任何其他测试失败,除了 pytest 在打印需要打印的任何内容以详细说明失败后停止测试。

到目前为止我的代码:

# Copied this implementation from _pytest.runner
def pytest_runtest_call(item):
    _update_current_test_var(item, "call")
    try:
        del sys.last_type
        del sys.last_value
        del sys.last_traceback
    except AttributeError:
        pass
    try:
        item.runtest()
    except Exception:
        # Store trace info to allow postmortem debugging
        type, value, tb = sys.exc_info()
        assert tb is not None
        tb = tb.tb_next  # Skip *this* frame
        sys.last_type = type
        sys.last_value = value
        sys.last_traceback = tb
        del type, value, tb  # Get rid of these in this frame

        # If test is marked as must pass, stop testing here
        if item.iter_markers(name = "must_pass"):
            pytest.exit('Test marked as "must_pass" failed, terminating.')

        raise

pytest 中是否已经有这样做的机制?

任何帮助将不胜感激。

谢谢。

【问题讨论】:

  • 根据您提供的代码,一旦您到达must_pass 标记,pytest 将始终退出
  • @aws_apprentice 是的,但它不会显示为失败的测试。例如,它不会显示回溯,也不会以红色打印“FAILED”。
  • 那么,如果遇到此标记不再运行任何测试而只是停止,那么所需的行为到底是什么?
  • 是的,当标记为“必须通过”的测试失败时,它应该将失败显示为任何其他测试失败,并跳过所有剩余的测试,然后退出。

标签: python pytest


【解决方案1】:

所以这可以通过使用pytest_runtest_makereportpytest_runtest_setup来实现

在您的conftest.py 中,您可以放置​​以下内容:

import pytest


def pytest_runtest_makereport(item, call):
    if item.iter_markers(name='must_pass'):
        if call.excinfo is not None:
            parent = item.parent
            parent._mpfailed = item


def pytest_runtest_setup(item):
    must_pass_failed = getattr(item.parent, '_mpfailed', None)
    if must_pass_failed is not None:
        pytest.skip('must pass test failed (%s)' % must_pass_failed.name)

现在,当我们使用以下内容对其进行测试时:

import pytest


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


def test_foo_1():
    assert foo(1, 1) == 2


@pytest.mark.must_pass
def test_foo_2():
    assert foo(2, 2) == 6


def test_foo_3():
    assert foo(3, 3) == 6


def test_foo_4():
    assert foo(4, 4) == 8

我们看到了想要的输出:

     ▲ = pytest test.py 
=============================================================== test session starts ================================================================
platform darwin -- Python 3.6.5, pytest-4.6.2, py-1.8.0, pluggy-0.12.0
rootdir: /Users/foo/Desktop/testing, inifile: pytest.ini
plugins: cov-2.7.1
collected 4 items                                                                                                                                  

test.py .Fss                                                                                                                                 [100%]

===================================================================== FAILURES =====================================================================
____________________________________________________________________ test_foo_2 ____________________________________________________________________

    @pytest.mark.must_pass
    def test_foo_2():
>       assert foo(2, 2) == 6
E       assert 4 == 6
E        +  where 4 = foo(2, 2)

test.py:14: AssertionError
================================================== 1 failed, 1 passed, 2 skipped in 0.08 seconds ===================================================

【讨论】:

  • 谢谢,这正是我需要的:)
【解决方案2】:

将代码粘贴到接受的答案中会产生以下结果:

'must_pass' not found in `markers` configuration option

对于任何来到这里想要使用——而不是实现——这个的人,同样可以使用 pytest-dependency 来实现:

  1. pip install pytest-dependency
  2. 请参阅this answer 了解用法。

【讨论】:

  • 如前所述,不需要为此添加第 3 方库,只需在 pytest.ini 中注册标记,如图所示 here
猜你喜欢
  • 1970-01-01
  • 2015-02-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-04-14
  • 1970-01-01
  • 2021-04-17
  • 1970-01-01
相关资源
最近更新 更多