4

从技术上讲,我已经解决了我正在处理的问题,但我不禁觉得我的解决方案很丑:

我有一个 pytest 套件,我可以在两种模式下运行:本地模式(用于开发测试;一切都通过 Chrome 在我的开发盒上运行)和 Seriousface 回归测试模式(用于 CI;该套件可以在无数浏览器上运行和操作系统)。我有一个命令行标志可以在两种模式之间切换,--test-local. 如果它在那里,我以本地模式运行。如果它不存在,我会以严肃的方式运行。这是我的做法:

# contents of conftest.py
import pytest

def pytest_addoption(parser):
    parser.addoption("--test-local", action="store_true", default=False, help="run locally instead of in seriousface mode")

def pytest_generate_tests(metafunc):
    if "dummy" in metafunc.fixturenames:
        if metafunc.config.getoption("--test-local"):
            driverParams = [(True, None)]
        else:
            driverParams = [(False, "seriousface setting 1"), (False, "seriousface setting 2")]
        metafunc.parameterize("dummy", driverParams)

@pytest.fixture(scope="function")
def driver(dummy):
    _driver = makeDriverStuff(dummy[0], dummy[1])
    yield _driver
    _driver.cleanup()

@pytest.fixture
def dummy():
    pass

问题是,那个dummy装置很可怕。我尝试过直接pytest_generate_tests对夹具进行参数化driver,但它最终会替换夹具,而不是仅仅将东西放入其中,因此cleanup()在测试完成时永远不会被调用。使用 dummy 可以让我用参数元组替换 dummy,以便将其传递到driver().

但是,重申一下,我所拥有的确实有效,感觉就像是一个 janky hack。

4

1 回答 1

5

您可以尝试不同的方法:不是动态选择测试的参数集,而是在其上声明所有参数集,但在启动时取消选择不相关的参数集。

# r.py
import pytest

real = pytest.mark.real
mock = pytest.mark.mock

@pytest.mark.parametrize('a, b', [
    real((True, 'serious 1')),
    real((True, 'serious 2')),
    mock((False, 'fancy mock')),
    (None, 'always unmarked!'),
])
def test_me(a, b):
    print([a, b])

然后按如下方式运行:

pytest -ra -v -s   r.py -m real        # strictly marked sets
pytest -ra -v -s   r.py -m mock        # strictly marked sets
pytest -ra -v -s   r.py -m 'not real'  # incl. non-marked sets
pytest -ra -v -s   r.py -m 'not mock'  # incl. non-marked sets

此外,skipif标记可用于选定的参数集(与取消选择不同)。pytest 原生支持它们。但是语法很丑。在 pytest 的Skip/xfail with parametrize部分中查看更多信息。

官方手册还包含与您在Custom marker and command line option to control test runs中的问题完全相同的情况。但是,它也没有-m测试取消选择那么优雅,并且更适合复杂的运行时条件,而不是先验已知的测试结构。

于 2017-10-10T22:25:28.087 回答