web-dev-qa-db-ja.com

一部のテストが失敗した場合でも、すべてのPyTestテストを実行するにはどうすればよいですか?

一部のユニットテストが失敗した場合でも、PyTestですべてのユニットテストを実行する方法を探しています。これを行う簡単な方法があるはずです。 CLiオプションを確認し、このサイトで同様の質問/回答を探しましたが、何も表示されませんでした。これがすでに回答されている場合は申し訳ありません。

たとえば、PyTestコードと一緒に次のコードスニペットを考えます。

def parrot(i):
    return i

def test_parrot():
    assert parrot(0) == 0
    assert parrot(1) == 1
    assert parrot(2) == 1
    assert parrot(2) == 2

デフォルトでは、実行は最初の失敗で停​​止します。

$ python -m pytest fail_me.py 
=================== test session starts ===================
platform linux2 -- Python 2.7.10, pytest-2.9.1, py-1.4.31, pluggy-0.3.1
rootdir: /home/npsrt/Documents/repo/codewars, inifile: 
collected 1 items 

fail_me.py F

=================== FAILURES ===================
___________________ test_parrot ___________________

    def test_parrot():
        assert parrot(0) == 0
        assert parrot(1) == 1
>       assert parrot(2) == 1
E       assert 2 == 1
E        +  where 2 = parrot(2)

fail_me.py:7: AssertionError
=================== 1 failed in 0.05 seconds ===================

私がやりたいのは、PyTestが最初の失敗に遭遇した後でもコードを実行し続けることです。

16
Jeff Wright

すべてのテストを実行しました。あなたが書いたテストは1つだけで、そのテストが実行されました!

(Google TestのEXPECTマクロのように)アサーションが失敗した場合にテストが続行される、致命的でないアサーションが必要な場合は、その機能を提供する pytest-expect を試してください。ここに彼らのサイトが与える例があります:

def test_func(expect):
    expect('a' == 'b')
    expect(1 != 1)
    a = 1
    b = 2
    expect(a == b, 'a:%s b:%s' % (a,b))

期待値の失敗によってテストが停止することはなく、失敗した期待値がすべて報告されることがわかります。

$ python -m pytest test_expect.py
================ test session starts =================
platform darwin -- Python 2.7.9 -- py-1.4.26 -- pytest-2.7.0
rootdir: /Users/okken/example, inifile: 
plugins: expect
collected 1 items 

test_expect.py F

====================== FAILURES ======================
_____________________ test_func ______________________
>    expect('a' == 'b')
test_expect.py:2
--------
>    expect(1 != 1)
test_expect.py:3
--------
>    expect(a == b, 'a:%s b:%s' % (a,b))
a:1 b:2
test_expect.py:6
--------
Failed Expectations:3
============== 1 failed in 0.01 seconds ==============

他の人がすでに述べたように、理想的には複数のテストを記述し、それぞれにアサーションを1つだけ持つことになります(これはハードリミットではなく、適切なガイドラインです)。

@pytest.mark.parametrize デコレータはこれを簡単にします:

import pytest

def parrot(i):
    return i

@pytest.mark.parametrize('inp, expected', [(0, 0), (1, 1), (2, 1), (2, 2)])
def test_parrot(inp, expected):
    assert parrot(inp) == expected

-vで実行すると:

parrot.py::test_parrot[0-0] PASSED
parrot.py::test_parrot[1-1] PASSED
parrot.py::test_parrot[2-1] FAILED
parrot.py::test_parrot[2-2] PASSED

=================================== FAILURES ===================================
_______________________________ test_parrot[2-1] _______________________________

inp = 2, expected = 1

    @pytest.mark.parametrize('inp, expected', [(0, 0), (1, 1), (2, 1), (2, 2)])
    def test_parrot(inp, expected):
>       assert parrot(inp) == expected
E       assert 2 == 1
E        +  where 2 = parrot(2)

parrot.py:8: AssertionError
====================== 1 failed, 3 passed in 0.01 seconds ======================
11
The Compiler

--maxfail引数でこれを制御できるはずです。デフォルトではエラーが発生しても停止しないようになっていると思います。そのため、オーバーライドしている場所について、py.testの設定ファイルを確認します。

7
Daenyth

Pytestプラグイン pytest-check はpytest-expectのリライトです(以前はここで推奨されていましたが、古くなっています)。次のように「ソフト」アサートを実行できます。

GitHubリポジトリの例:

import pytest_check as check

def test_example():
    a = 1
    b = 2
    c = [2, 4, 6]
    check.greater(a, b)
    check.less_equal(b, a)
    check.is_in(a, c, "Is 1 in the list")
    check.is_not_in(b, c, "make sure 2 isn't in list")
2
ChrisGS