2010-11-06 06:37:25 +08:00
|
|
|
|
|
|
|
|
2011-09-06 17:43:42 +08:00
|
|
|
Basic patterns and examples
|
2010-11-06 06:37:25 +08:00
|
|
|
==========================================================
|
|
|
|
|
2020-06-08 21:03:10 +08:00
|
|
|
How to change command line options defaults
|
|
|
|
-------------------------------------------
|
|
|
|
|
|
|
|
It can be tedious to type the same series of command line options
|
|
|
|
every time you use ``pytest``. For example, if you always want to see
|
|
|
|
detailed info on skipped and xfailed tests, as well as have terser "dot"
|
|
|
|
progress output, you can write it into a configuration file:
|
|
|
|
|
|
|
|
.. code-block:: ini
|
|
|
|
|
|
|
|
# content of pytest.ini
|
|
|
|
[pytest]
|
|
|
|
addopts = -ra -q
|
|
|
|
|
|
|
|
|
|
|
|
Alternatively, you can set a ``PYTEST_ADDOPTS`` environment variable to add command
|
|
|
|
line options while the environment is in use:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
|
|
|
|
|
|
|
export PYTEST_ADDOPTS="-v"
|
|
|
|
|
|
|
|
Here's how the command-line is built in the presence of ``addopts`` or the environment variable:
|
|
|
|
|
|
|
|
.. code-block:: text
|
|
|
|
|
|
|
|
<pytest.ini:addopts> $PYTEST_ADDOPTS <extra command-line arguments>
|
|
|
|
|
|
|
|
So if the user executes in the command-line:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
|
|
|
|
|
|
|
pytest -m slow
|
|
|
|
|
|
|
|
The actual command line executed is:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
|
|
|
|
|
|
|
pytest -ra -q -v -m slow
|
|
|
|
|
|
|
|
Note that as usual for other command-line applications, in case of conflicting options the last one wins, so the example
|
|
|
|
above will show verbose output because ``-v`` overwrites ``-q``.
|
|
|
|
|
|
|
|
|
2018-02-28 04:58:51 +08:00
|
|
|
.. _request example:
|
|
|
|
|
2011-09-06 17:43:42 +08:00
|
|
|
Pass different values to a test function, depending on command line options
|
2010-11-06 06:37:25 +08:00
|
|
|
----------------------------------------------------------------------------
|
|
|
|
|
2010-11-26 20:26:56 +08:00
|
|
|
.. regendoc:wipe
|
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
Suppose we want to write a test that depends on a command line option.
|
2016-08-23 10:35:41 +08:00
|
|
|
Here is a basic pattern to achieve this:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-06 06:37:25 +08:00
|
|
|
|
|
|
|
# content of test_sample.py
|
|
|
|
def test_answer(cmdopt):
|
|
|
|
if cmdopt == "type1":
|
2018-06-03 11:29:28 +08:00
|
|
|
print("first")
|
2010-11-06 06:37:25 +08:00
|
|
|
elif cmdopt == "type2":
|
2018-06-03 11:29:28 +08:00
|
|
|
print("second")
|
|
|
|
assert 0 # to see what was printed
|
2010-11-06 06:37:25 +08:00
|
|
|
|
|
|
|
|
|
|
|
For this to work we need to add a command line option and
|
2021-03-13 06:16:47 +08:00
|
|
|
provide the ``cmdopt`` through a :ref:`fixture function <fixture>`:
|
2016-08-23 10:35:41 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-06 06:37:25 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
2012-10-07 19:06:17 +08:00
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
def pytest_addoption(parser):
|
2018-06-03 11:29:28 +08:00
|
|
|
parser.addoption(
|
|
|
|
"--cmdopt", action="store", default="type1", help="my option: type1 or type2"
|
|
|
|
)
|
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2012-10-07 19:06:17 +08:00
|
|
|
@pytest.fixture
|
|
|
|
def cmdopt(request):
|
2012-11-06 21:09:12 +08:00
|
|
|
return request.config.getoption("--cmdopt")
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
Let's run this without supplying our new option:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -q test_sample.py
|
2019-01-06 03:19:40 +08:00
|
|
|
F [100%]
|
|
|
|
================================= FAILURES =================================
|
|
|
|
_______________________________ test_answer ________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
cmdopt = 'type1'
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
def test_answer(cmdopt):
|
|
|
|
if cmdopt == "type1":
|
2018-06-05 09:11:27 +08:00
|
|
|
print("first")
|
2010-11-06 06:37:25 +08:00
|
|
|
elif cmdopt == "type2":
|
2018-06-05 09:11:27 +08:00
|
|
|
print("second")
|
|
|
|
> assert 0 # to see what was printed
|
2010-11-06 06:37:25 +08:00
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
test_sample.py:6: AssertionError
|
2019-01-06 03:19:40 +08:00
|
|
|
--------------------------- Captured stdout call ---------------------------
|
2010-11-06 06:37:25 +08:00
|
|
|
first
|
2020-03-11 22:23:25 +08:00
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED test_sample.py::test_answer - assert 0
|
2019-09-18 21:11:59 +08:00
|
|
|
1 failed in 0.12s
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
And now with supplying a command line option:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -q --cmdopt=type2
|
2019-01-06 03:19:40 +08:00
|
|
|
F [100%]
|
|
|
|
================================= FAILURES =================================
|
|
|
|
_______________________________ test_answer ________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
cmdopt = 'type2'
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
def test_answer(cmdopt):
|
|
|
|
if cmdopt == "type1":
|
2018-06-05 09:11:27 +08:00
|
|
|
print("first")
|
2010-11-06 06:37:25 +08:00
|
|
|
elif cmdopt == "type2":
|
2018-06-05 09:11:27 +08:00
|
|
|
print("second")
|
|
|
|
> assert 0 # to see what was printed
|
2010-11-06 06:37:25 +08:00
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-06 06:37:25 +08:00
|
|
|
test_sample.py:6: AssertionError
|
2019-01-06 03:19:40 +08:00
|
|
|
--------------------------- Captured stdout call ---------------------------
|
2010-11-06 06:37:25 +08:00
|
|
|
second
|
2020-03-11 22:23:25 +08:00
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED test_sample.py::test_answer - assert 0
|
2019-09-18 21:11:59 +08:00
|
|
|
1 failed in 0.12s
|
2010-11-06 06:37:25 +08:00
|
|
|
|
2021-05-07 15:48:46 +08:00
|
|
|
You can see that the command line option arrived in our test.
|
|
|
|
|
|
|
|
We could add simple validation for the input by listing the choices:
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
# content of conftest.py
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_addoption(parser):
|
|
|
|
parser.addoption(
|
|
|
|
"--cmdopt",
|
|
|
|
action="store",
|
|
|
|
default="type1",
|
|
|
|
help="my option: type1 or type2",
|
|
|
|
choices=("type1", "type2"),
|
|
|
|
)
|
|
|
|
|
|
|
|
Now we'll get feedback on a bad argument:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
|
|
|
|
|
|
|
$ pytest -q --cmdopt=type3
|
|
|
|
ERROR: usage: pytest [options] [file_or_dir] [file_or_dir] [...]
|
|
|
|
pytest: error: argument --cmdopt: invalid choice: 'type3' (choose from 'type1', 'type2')
|
|
|
|
|
2021-10-04 14:56:26 +08:00
|
|
|
|
2021-05-07 15:48:46 +08:00
|
|
|
If you need to provide more detailed error messages, you can use the
|
2023-12-07 16:41:51 +08:00
|
|
|
``type`` parameter and raise :exc:`pytest.UsageError`:
|
2021-05-07 15:48:46 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
# content of conftest.py
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
|
|
def type_checker(value):
|
|
|
|
msg = "cmdopt must specify a numeric type as typeNNN"
|
|
|
|
if not value.startswith("type"):
|
|
|
|
raise pytest.UsageError(msg)
|
|
|
|
try:
|
|
|
|
int(value[4:])
|
|
|
|
except ValueError:
|
|
|
|
raise pytest.UsageError(msg)
|
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
def pytest_addoption(parser):
|
|
|
|
parser.addoption(
|
|
|
|
"--cmdopt",
|
|
|
|
action="store",
|
|
|
|
default="type1",
|
|
|
|
help="my option: type1 or type2",
|
|
|
|
type=type_checker,
|
|
|
|
)
|
|
|
|
|
|
|
|
This completes the basic pattern. However, one often rather wants to
|
|
|
|
process command line options outside of the test and rather pass in
|
|
|
|
different or more complex objects.
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2011-09-06 17:43:42 +08:00
|
|
|
Dynamically adding command line options
|
2010-12-07 19:14:12 +08:00
|
|
|
--------------------------------------------------------------
|
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
|
|
|
Through :confval:`addopts` you can statically add command line
|
|
|
|
options for your project. You can also dynamically modify
|
2016-08-23 10:35:41 +08:00
|
|
|
the command line arguments before they get processed:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2010-12-07 19:14:12 +08:00
|
|
|
|
2019-04-27 21:11:25 +08:00
|
|
|
# setuptools plugin
|
2010-12-07 19:14:12 +08:00
|
|
|
import sys
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
|
|
|
2018-04-23 09:56:18 +08:00
|
|
|
def pytest_load_initial_conftests(args):
|
2018-06-03 11:29:28 +08:00
|
|
|
if "xdist" in sys.modules: # pytest-xdist plugin
|
2010-12-07 19:14:12 +08:00
|
|
|
import multiprocessing
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2010-12-07 19:14:12 +08:00
|
|
|
num = max(multiprocessing.cpu_count() / 2, 1)
|
|
|
|
args[:] = ["-n", str(num)] + args
|
|
|
|
|
2021-10-22 20:47:57 +08:00
|
|
|
If you have the :pypi:`xdist plugin <pytest-xdist>` installed
|
2010-12-07 19:14:12 +08:00
|
|
|
you will now always perform test runs using a number
|
|
|
|
of subprocesses close to your CPU. Running in an empty
|
2018-11-24 13:41:22 +08:00
|
|
|
directory with the above conftest.py:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-12-07 19:14:12 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 0 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-08-30 23:43:47 +08:00
|
|
|
========================== no tests ran in 0.12s ===========================
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2011-02-09 21:55:21 +08:00
|
|
|
.. _`excontrolskip`:
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2011-09-06 17:43:42 +08:00
|
|
|
Control skipping of tests according to command line option
|
2010-11-21 04:35:55 +08:00
|
|
|
--------------------------------------------------------------
|
|
|
|
|
2010-11-26 20:26:56 +08:00
|
|
|
.. regendoc:wipe
|
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
Here is a ``conftest.py`` file adding a ``--runslow`` command
|
2017-08-09 10:51:07 +08:00
|
|
|
line option to control skipping of ``pytest.mark.slow`` marked tests:
|
2016-08-23 10:35:41 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
|
|
|
|
|
|
|
import pytest
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
def pytest_addoption(parser):
|
2018-06-03 11:29:28 +08:00
|
|
|
parser.addoption(
|
|
|
|
"--runslow", action="store_true", default=False, help="run slow tests"
|
|
|
|
)
|
|
|
|
|
2017-08-09 10:51:07 +08:00
|
|
|
|
2019-06-07 00:41:15 +08:00
|
|
|
def pytest_configure(config):
|
|
|
|
config.addinivalue_line("markers", "slow: mark test as slow to run")
|
|
|
|
|
|
|
|
|
2017-08-09 10:51:07 +08:00
|
|
|
def pytest_collection_modifyitems(config, items):
|
|
|
|
if config.getoption("--runslow"):
|
|
|
|
# --runslow given in cli: do not skip slow tests
|
|
|
|
return
|
|
|
|
skip_slow = pytest.mark.skip(reason="need --runslow option to run")
|
|
|
|
for item in items:
|
|
|
|
if "slow" in item.keywords:
|
|
|
|
item.add_marker(skip_slow)
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
We can now write a test module like this:
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
# content of test_module.py
|
2010-11-21 04:35:55 +08:00
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
2015-10-01 03:53:34 +08:00
|
|
|
def test_func_fast():
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2017-08-09 10:51:07 +08:00
|
|
|
@pytest.mark.slow
|
2010-11-21 04:35:55 +08:00
|
|
|
def test_func_slow():
|
|
|
|
pass
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
and when running it will see a skipped "slow" test:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -rs # "-rs" means report details on the little 's'
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 2 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
test_module.py .s [100%]
|
2019-05-09 05:50:08 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
========================= short test summary info ==========================
|
2019-01-31 00:25:38 +08:00
|
|
|
SKIPPED [1] test_module.py:8: need --runslow option to run
|
2019-08-30 23:43:47 +08:00
|
|
|
======================= 1 passed, 1 skipped in 0.12s =======================
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
Or run it including the ``slow`` marked test:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest --runslow
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 2 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
test_module.py .. [100%]
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-08-30 23:43:47 +08:00
|
|
|
============================ 2 passed in 0.12s =============================
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2020-03-03 00:08:37 +08:00
|
|
|
.. _`__tracebackhide__`:
|
|
|
|
|
2011-09-06 17:43:42 +08:00
|
|
|
Writing well integrated assertion helpers
|
2020-03-03 00:08:37 +08:00
|
|
|
-----------------------------------------
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2010-11-26 20:26:56 +08:00
|
|
|
.. regendoc:wipe
|
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
If you have a test helper function called from a test you can
|
|
|
|
use the ``pytest.fail`` marker to fail a test with a certain message.
|
|
|
|
The test support function will not show up in the traceback if you
|
|
|
|
set the ``__tracebackhide__`` option somewhere in the helper function.
|
2016-08-23 10:35:41 +08:00
|
|
|
Example:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
|
|
|
# content of test_checkconfig.py
|
|
|
|
import pytest
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
def checkconfig(x):
|
|
|
|
__tracebackhide__ = True
|
|
|
|
if not hasattr(x, "config"):
|
2022-04-28 22:30:16 +08:00
|
|
|
pytest.fail(f"not configured: {x}")
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
|
|
|
|
def test_something():
|
|
|
|
checkconfig(42)
|
|
|
|
|
2014-01-18 19:31:33 +08:00
|
|
|
The ``__tracebackhide__`` setting influences ``pytest`` showing
|
2010-11-21 04:35:55 +08:00
|
|
|
of tracebacks: the ``checkconfig`` function will not be shown
|
2016-03-21 00:12:50 +08:00
|
|
|
unless the ``--full-trace`` command line option is specified.
|
2018-11-24 13:41:22 +08:00
|
|
|
Let's run our little function:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -q test_checkconfig.py
|
2019-01-06 03:19:40 +08:00
|
|
|
F [100%]
|
|
|
|
================================= FAILURES =================================
|
|
|
|
______________________________ test_something ______________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
def test_something():
|
|
|
|
> checkconfig(42)
|
|
|
|
E Failed: not configured: 42
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2018-06-05 09:11:27 +08:00
|
|
|
test_checkconfig.py:11: Failed
|
2020-03-11 22:23:25 +08:00
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED test_checkconfig.py::test_something - Failed: not configured: 42
|
2019-09-18 21:11:59 +08:00
|
|
|
1 failed in 0.12s
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2016-04-20 17:07:34 +08:00
|
|
|
If you only want to hide certain exceptions, you can set ``__tracebackhide__``
|
|
|
|
to a callable which gets the ``ExceptionInfo`` object. You can for example use
|
2016-08-23 10:35:41 +08:00
|
|
|
this to make sure unexpected exception types aren't hidden:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2016-04-20 16:25:33 +08:00
|
|
|
|
2016-04-20 17:07:34 +08:00
|
|
|
import operator
|
2022-04-28 22:30:16 +08:00
|
|
|
|
2016-04-20 16:25:33 +08:00
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2016-04-20 16:25:33 +08:00
|
|
|
class ConfigException(Exception):
|
|
|
|
pass
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2016-04-20 16:25:33 +08:00
|
|
|
def checkconfig(x):
|
2018-06-03 11:29:28 +08:00
|
|
|
__tracebackhide__ = operator.methodcaller("errisinstance", ConfigException)
|
2016-04-20 16:25:33 +08:00
|
|
|
if not hasattr(x, "config"):
|
2022-04-28 22:30:16 +08:00
|
|
|
raise ConfigException(f"not configured: {x}")
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2016-04-20 16:25:33 +08:00
|
|
|
|
|
|
|
def test_something():
|
|
|
|
checkconfig(42)
|
|
|
|
|
|
|
|
This will avoid hiding the exception traceback on unrelated exceptions (i.e.
|
|
|
|
bugs in assertion helpers).
|
|
|
|
|
|
|
|
|
2014-01-18 19:31:33 +08:00
|
|
|
Detect if running from within a pytest run
|
2010-11-21 04:35:55 +08:00
|
|
|
--------------------------------------------------------------
|
|
|
|
|
2010-11-26 20:26:56 +08:00
|
|
|
.. regendoc:wipe
|
|
|
|
|
2010-11-21 04:35:55 +08:00
|
|
|
Usually it is a bad idea to make application code
|
|
|
|
behave differently if called from a test. But if you
|
|
|
|
absolutely must find out if your application code is
|
2016-08-23 10:35:41 +08:00
|
|
|
running from a test you can do something like this:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
# content of your_module.py
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
_called_from_test = False
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
# content of conftest.py
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
def pytest_configure(config):
|
|
|
|
your_module._called_from_test = True
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
and then check for the ``your_module._called_from_test`` flag:
|
2016-08-23 10:35:41 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
if your_module._called_from_test:
|
2010-11-21 04:35:55 +08:00
|
|
|
# called from within a test run
|
2018-06-03 11:19:17 +08:00
|
|
|
...
|
2010-11-21 04:35:55 +08:00
|
|
|
else:
|
|
|
|
# called "normally"
|
2018-06-03 11:19:17 +08:00
|
|
|
...
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2019-11-12 10:02:47 +08:00
|
|
|
accordingly in your application.
|
2010-11-21 04:35:55 +08:00
|
|
|
|
2011-01-13 02:39:36 +08:00
|
|
|
Adding info to test report header
|
|
|
|
--------------------------------------------------------------
|
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
It's easy to present extra information in a ``pytest`` run:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2011-01-13 02:39:36 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
2014-01-18 19:31:33 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2011-01-13 02:39:36 +08:00
|
|
|
def pytest_report_header(config):
|
|
|
|
return "project deps: mylib-1.1"
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
which will add the string to the test header accordingly:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2011-01-13 02:39:36 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2011-01-13 02:39:36 +08:00
|
|
|
project deps: mylib-1.1
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 0 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-08-30 23:43:47 +08:00
|
|
|
========================== no tests ran in 0.12s ===========================
|
2011-01-13 02:39:36 +08:00
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
It is also possible to return a list of strings which will be considered as several
|
|
|
|
lines of information. You may consider ``config.getoption('verbose')`` in order to
|
|
|
|
display more information if applicable:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2011-01-13 02:39:36 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2011-01-13 02:39:36 +08:00
|
|
|
def pytest_report_header(config):
|
2018-06-03 11:29:28 +08:00
|
|
|
if config.getoption("verbose") > 0:
|
2011-01-13 02:39:36 +08:00
|
|
|
return ["info1: did you know that ...", "did you?"]
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
which will add info only when run with "--v":
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2011-01-13 02:39:36 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -v
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y -- $PYTHON_PREFIX/bin/python
|
2021-10-04 14:56:26 +08:00
|
|
|
cachedir: .pytest_cache
|
2011-01-13 02:39:36 +08:00
|
|
|
info1: did you know that ...
|
|
|
|
did you?
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2011-01-13 02:39:36 +08:00
|
|
|
collecting ... collected 0 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-08-30 23:43:47 +08:00
|
|
|
========================== no tests ran in 0.12s ===========================
|
2011-01-13 02:39:36 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
and nothing when run plainly:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2011-01-13 02:39:36 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 0 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-08-30 23:43:47 +08:00
|
|
|
========================== no tests ran in 0.12s ===========================
|
2011-11-09 01:20:56 +08:00
|
|
|
|
2020-11-08 09:44:04 +08:00
|
|
|
Profiling test duration
|
2011-11-09 01:20:56 +08:00
|
|
|
--------------------------
|
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
|
|
|
.. versionadded: 2.2
|
|
|
|
|
|
|
|
If you have a slow running large test suite you might want to find
|
2016-08-23 10:35:41 +08:00
|
|
|
out which tests are the slowest. Let's make an artificial test suite:
|
2011-11-09 01:20:56 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
.. code-block:: python
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
# content of test_some_are_slow.py
|
2011-12-05 18:10:48 +08:00
|
|
|
import time
|
2011-11-09 01:20:56 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2011-11-09 01:20:56 +08:00
|
|
|
def test_funcfast():
|
2017-08-09 05:04:21 +08:00
|
|
|
time.sleep(0.1)
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2011-11-09 01:20:56 +08:00
|
|
|
def test_funcslow1():
|
2017-08-09 05:04:21 +08:00
|
|
|
time.sleep(0.2)
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2011-11-09 01:20:56 +08:00
|
|
|
def test_funcslow2():
|
2017-08-09 05:04:21 +08:00
|
|
|
time.sleep(0.3)
|
2011-12-05 18:10:48 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
Now we can profile which test functions execute the slowest:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2011-11-09 01:20:56 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest --durations=3
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-07 19:06:17 +08:00
|
|
|
collected 3 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
test_some_are_slow.py ... [100%]
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2020-07-09 05:51:01 +08:00
|
|
|
=========================== slowest 3 durations ============================
|
2018-02-20 09:43:59 +08:00
|
|
|
0.30s call test_some_are_slow.py::test_funcslow2
|
2020-01-21 00:53:31 +08:00
|
|
|
0.20s call test_some_are_slow.py::test_funcslow1
|
2020-03-11 22:23:25 +08:00
|
|
|
0.10s call test_some_are_slow.py::test_funcfast
|
2019-08-30 23:43:47 +08:00
|
|
|
============================ 3 passed in 0.12s =============================
|
2012-10-18 18:24:50 +08:00
|
|
|
|
2020-11-08 09:44:04 +08:00
|
|
|
Incremental testing - test steps
|
2012-10-18 18:24:50 +08:00
|
|
|
---------------------------------------------------
|
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
|
|
|
Sometimes you may have a testing situation which consists of a series
|
|
|
|
of test steps. If one step fails it makes no sense to execute further
|
|
|
|
steps as they are all expected to fail anyway and their tracebacks
|
|
|
|
add no insight. Here is a simple ``conftest.py`` file which introduces
|
2016-08-23 10:35:41 +08:00
|
|
|
an ``incremental`` marker which is to be used on classes:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-10-18 18:24:50 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
|
|
|
|
2020-03-11 22:23:25 +08:00
|
|
|
from typing import Dict, Tuple
|
2022-04-28 22:30:16 +08:00
|
|
|
|
2020-03-11 22:23:25 +08:00
|
|
|
import pytest
|
|
|
|
|
2020-01-27 17:45:17 +08:00
|
|
|
# store history of failures per test class name and per index in parametrize (if parametrize used)
|
|
|
|
_test_failed_incremental: Dict[str, Dict[Tuple[int, ...], str]] = {}
|
2012-10-18 18:24:50 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def pytest_runtest_makereport(item, call):
|
2012-10-18 21:06:55 +08:00
|
|
|
if "incremental" in item.keywords:
|
2020-01-27 17:45:17 +08:00
|
|
|
# incremental marker is used
|
2012-10-18 18:24:50 +08:00
|
|
|
if call.excinfo is not None:
|
2020-01-27 17:45:17 +08:00
|
|
|
# the test has failed
|
|
|
|
# retrieve the class name of the test
|
|
|
|
cls_name = str(item.cls)
|
|
|
|
# retrieve the index of the test (if parametrize is used in combination with incremental)
|
|
|
|
parametrize_index = (
|
|
|
|
tuple(item.callspec.indices.values())
|
|
|
|
if hasattr(item, "callspec")
|
|
|
|
else ()
|
|
|
|
)
|
|
|
|
# retrieve the name of the test function
|
|
|
|
test_name = item.originalname or item.name
|
|
|
|
# store in _test_failed_incremental the original name of the failed test
|
|
|
|
_test_failed_incremental.setdefault(cls_name, {}).setdefault(
|
|
|
|
parametrize_index, test_name
|
|
|
|
)
|
2012-10-18 18:24:50 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def pytest_runtest_setup(item):
|
2012-10-18 21:06:55 +08:00
|
|
|
if "incremental" in item.keywords:
|
2020-01-27 17:45:17 +08:00
|
|
|
# retrieve the class name of the test
|
|
|
|
cls_name = str(item.cls)
|
|
|
|
# check if a previous test has failed for this class
|
|
|
|
if cls_name in _test_failed_incremental:
|
|
|
|
# retrieve the index of the test (if parametrize is used in combination with incremental)
|
|
|
|
parametrize_index = (
|
|
|
|
tuple(item.callspec.indices.values())
|
|
|
|
if hasattr(item, "callspec")
|
|
|
|
else ()
|
|
|
|
)
|
|
|
|
# retrieve the name of the first test function to fail for this class name and index
|
|
|
|
test_name = _test_failed_incremental[cls_name].get(parametrize_index, None)
|
|
|
|
# if name found, test has failed for the combination of class name & test name
|
|
|
|
if test_name is not None:
|
2022-04-28 22:30:16 +08:00
|
|
|
pytest.xfail(f"previous test failed ({test_name})")
|
2020-01-27 17:45:17 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
|
|
|
|
These two hook implementations work together to abort incremental-marked
|
2016-08-23 10:35:41 +08:00
|
|
|
tests in a class. Here is a test module example:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-10-18 18:24:50 +08:00
|
|
|
|
|
|
|
# content of test_step.py
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
@pytest.mark.incremental
|
2019-08-07 03:40:27 +08:00
|
|
|
class TestUserHandling:
|
2012-10-18 18:24:50 +08:00
|
|
|
def test_login(self):
|
|
|
|
pass
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def test_modification(self):
|
|
|
|
assert 0
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def test_deletion(self):
|
|
|
|
pass
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def test_normal():
|
|
|
|
pass
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
If we run this:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2012-10-18 18:24:50 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -rx
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-10-18 18:24:50 +08:00
|
|
|
collected 4 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
test_step.py .Fx. [100%]
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
================================= FAILURES =================================
|
|
|
|
____________________ TestUserHandling.test_modification ____________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2021-10-04 14:56:26 +08:00
|
|
|
self = <test_step.TestUserHandling object at 0xdeadbeef0001>
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-10-18 18:24:50 +08:00
|
|
|
def test_modification(self):
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2018-06-05 09:11:27 +08:00
|
|
|
test_step.py:11: AssertionError
|
2024-01-18 05:44:01 +08:00
|
|
|
================================ XFAILURES =================================
|
|
|
|
______________________ TestUserHandling.test_deletion ______________________
|
|
|
|
|
|
|
|
item = <Function test_deletion>
|
|
|
|
|
|
|
|
def pytest_runtest_setup(item):
|
|
|
|
if "incremental" in item.keywords:
|
|
|
|
# retrieve the class name of the test
|
|
|
|
cls_name = str(item.cls)
|
|
|
|
# check if a previous test has failed for this class
|
|
|
|
if cls_name in _test_failed_incremental:
|
|
|
|
# retrieve the index of the test (if parametrize is used in combination with incremental)
|
|
|
|
parametrize_index = (
|
|
|
|
tuple(item.callspec.indices.values())
|
|
|
|
if hasattr(item, "callspec")
|
|
|
|
else ()
|
|
|
|
)
|
|
|
|
# retrieve the name of the first test function to fail for this class name and index
|
|
|
|
test_name = _test_failed_incremental[cls_name].get(parametrize_index, None)
|
|
|
|
# if name found, test has failed for the combination of class name & test name
|
|
|
|
if test_name is not None:
|
|
|
|
> pytest.xfail(f"previous test failed ({test_name})")
|
|
|
|
E _pytest.outcomes.XFailed: previous test failed (test_modification)
|
|
|
|
|
|
|
|
conftest.py:47: XFailed
|
2019-01-06 03:19:40 +08:00
|
|
|
========================= short test summary info ==========================
|
2022-10-25 19:12:55 +08:00
|
|
|
XFAIL test_step.py::TestUserHandling::test_deletion - reason: previous test failed (test_modification)
|
2019-08-30 23:43:47 +08:00
|
|
|
================== 1 failed, 2 passed, 1 xfailed in 0.12s ==================
|
2012-10-18 18:24:50 +08:00
|
|
|
|
|
|
|
We'll see that ``test_deletion`` was not executed because ``test_modification``
|
|
|
|
failed. It is reported as an "expected failure".
|
|
|
|
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
Package/Directory-level fixtures (setups)
|
|
|
|
-------------------------------------------------------
|
|
|
|
|
|
|
|
If you have nested test directories, you can have per-directory fixture scopes
|
2021-04-01 21:13:12 +08:00
|
|
|
by placing fixture functions in a ``conftest.py`` file in that directory.
|
2012-11-07 18:11:40 +08:00
|
|
|
You can use all types of fixtures including :ref:`autouse fixtures
|
|
|
|
<autouse fixtures>` which are the equivalent of xUnit's setup/teardown
|
|
|
|
concept. It's however recommended to have explicit fixture references in your
|
2015-11-28 14:46:45 +08:00
|
|
|
tests or test classes rather than relying on implicitly executing
|
2012-11-07 18:11:40 +08:00
|
|
|
setup/teardown functions, especially if they are far away from the actual tests.
|
|
|
|
|
2017-01-01 01:54:47 +08:00
|
|
|
Here is an example for making a ``db`` fixture available in a directory:
|
2016-08-23 10:35:41 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
# content of a/conftest.py
|
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2019-08-07 03:40:27 +08:00
|
|
|
class DB:
|
2012-11-07 18:11:40 +08:00
|
|
|
pass
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2023-04-01 01:00:45 +08:00
|
|
|
@pytest.fixture(scope="package")
|
2012-11-07 18:11:40 +08:00
|
|
|
def db():
|
|
|
|
return DB()
|
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
and then a test module in that directory:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
# content of a/test_db.py
|
|
|
|
def test_a1(db):
|
|
|
|
assert 0, db # to show value
|
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
another test module:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
# content of a/test_db2.py
|
|
|
|
def test_a2(db):
|
|
|
|
assert 0, db # to show value
|
|
|
|
|
|
|
|
and then a module in a sister directory which will not see
|
2016-08-23 10:35:41 +08:00
|
|
|
the ``db`` fixture:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
# content of b/test_error.py
|
|
|
|
def test_root(db): # no db here, will error out
|
|
|
|
pass
|
2014-01-18 19:31:33 +08:00
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
We can run this:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2012-11-07 18:11:40 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-11-09 06:36:16 +08:00
|
|
|
collected 7 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2024-01-02 16:58:20 +08:00
|
|
|
a/test_db.py F [ 14%]
|
|
|
|
a/test_db2.py F [ 28%]
|
|
|
|
b/test_error.py E [ 42%]
|
|
|
|
test_step.py .Fx. [100%]
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
================================== ERRORS ==================================
|
|
|
|
_______________________ ERROR at setup of test_root ________________________
|
2021-10-04 14:56:26 +08:00
|
|
|
file /home/sweet/project/b/test_error.py, line 1
|
2012-11-07 18:11:40 +08:00
|
|
|
def test_root(db): # no db here, will error out
|
2016-08-02 02:46:34 +08:00
|
|
|
E fixture 'db' not found
|
2019-05-12 00:35:32 +08:00
|
|
|
> available fixtures: cache, capfd, capfdbinary, caplog, capsys, capsysbinary, doctest_namespace, monkeypatch, pytestconfig, record_property, record_testsuite_property, record_xml_attribute, recwarn, tmp_path, tmp_path_factory, tmpdir, tmpdir_factory
|
2016-08-18 20:27:16 +08:00
|
|
|
> use 'pytest --fixtures [testpath]' for help on them.
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2021-10-04 14:56:26 +08:00
|
|
|
/home/sweet/project/b/test_error.py:1
|
2019-01-06 03:19:40 +08:00
|
|
|
================================= FAILURES =================================
|
|
|
|
_________________________________ test_a1 __________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2024-01-02 16:58:20 +08:00
|
|
|
db = <conftest.DB object at 0xdeadbeef0002>
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-07 18:11:40 +08:00
|
|
|
def test_a1(db):
|
|
|
|
> assert 0, db # to show value
|
2024-01-02 16:58:20 +08:00
|
|
|
E AssertionError: <conftest.DB object at 0xdeadbeef0002>
|
2014-09-24 20:46:56 +08:00
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-07 18:11:40 +08:00
|
|
|
a/test_db.py:2: AssertionError
|
2019-01-06 03:19:40 +08:00
|
|
|
_________________________________ test_a2 __________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2024-01-02 16:58:20 +08:00
|
|
|
db = <conftest.DB object at 0xdeadbeef0002>
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-07 18:11:40 +08:00
|
|
|
def test_a2(db):
|
|
|
|
> assert 0, db # to show value
|
2024-01-02 16:58:20 +08:00
|
|
|
E AssertionError: <conftest.DB object at 0xdeadbeef0002>
|
2014-09-24 20:46:56 +08:00
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-07 18:11:40 +08:00
|
|
|
a/test_db2.py:2: AssertionError
|
2024-01-02 16:58:20 +08:00
|
|
|
____________________ TestUserHandling.test_modification ____________________
|
|
|
|
|
|
|
|
self = <test_step.TestUserHandling object at 0xdeadbeef0003>
|
|
|
|
|
|
|
|
def test_modification(self):
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
|
|
|
|
|
|
|
test_step.py:11: AssertionError
|
2020-03-11 22:23:25 +08:00
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED a/test_db.py::test_a1 - AssertionError: <conftest.DB object at 0x7...
|
|
|
|
FAILED a/test_db2.py::test_a2 - AssertionError: <conftest.DB object at 0x...
|
2024-01-02 16:58:20 +08:00
|
|
|
FAILED test_step.py::TestUserHandling::test_modification - assert 0
|
2020-03-11 22:23:25 +08:00
|
|
|
ERROR b/test_error.py::test_root
|
2019-08-30 23:43:47 +08:00
|
|
|
============= 3 failed, 2 passed, 1 xfailed, 1 error in 0.12s ==============
|
2012-11-07 18:11:40 +08:00
|
|
|
|
|
|
|
The two test modules in the ``a`` directory see the same ``db`` fixture instance
|
|
|
|
while the one test in the sister-directory ``b`` doesn't see it. We could of course
|
|
|
|
also define a ``db`` fixture in that sister directory's ``conftest.py`` file.
|
|
|
|
Note that each fixture is only instantiated if there is a test actually needing
|
|
|
|
it (unless you use "autouse" fixture which are always executed ahead of the first test
|
|
|
|
executing).
|
|
|
|
|
|
|
|
|
2020-11-08 09:44:04 +08:00
|
|
|
Post-process test reports / failures
|
2012-11-09 06:36:16 +08:00
|
|
|
---------------------------------------
|
|
|
|
|
|
|
|
If you want to postprocess test reports and need access to the executing
|
2014-01-18 19:31:33 +08:00
|
|
|
environment you can implement a hook that gets called when the test
|
2012-11-09 06:36:16 +08:00
|
|
|
"report" object is about to be created. Here we write out all failing
|
|
|
|
test calls and also access a fixture (if it was used by the test) in
|
|
|
|
case you want to query/look at it during your post processing. In our
|
2017-01-01 01:54:47 +08:00
|
|
|
case we just write some information out to a ``failures`` file:
|
2016-08-23 10:35:41 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-09 06:36:16 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
|
|
|
|
|
|
|
import os.path
|
|
|
|
|
2022-04-28 22:30:16 +08:00
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2023-06-13 03:30:06 +08:00
|
|
|
@pytest.hookimpl(wrapper=True, tryfirst=True)
|
2015-08-09 06:07:27 +08:00
|
|
|
def pytest_runtest_makereport(item, call):
|
2012-11-09 06:36:16 +08:00
|
|
|
# execute all other hooks to obtain the report object
|
2023-06-13 03:30:06 +08:00
|
|
|
rep = yield
|
2012-11-09 06:36:16 +08:00
|
|
|
|
|
|
|
# we only look at actual failing test calls, not setup/teardown
|
2014-01-18 19:31:33 +08:00
|
|
|
if rep.when == "call" and rep.failed:
|
2012-11-09 06:36:16 +08:00
|
|
|
mode = "a" if os.path.exists("failures") else "w"
|
2023-07-09 02:40:05 +08:00
|
|
|
with open("failures", mode, encoding="utf-8") as f:
|
2012-11-09 06:36:16 +08:00
|
|
|
# let's also access a fixture for the fun of it
|
2021-03-14 03:22:54 +08:00
|
|
|
if "tmp_path" in item.fixturenames:
|
|
|
|
extra = " ({})".format(item.funcargs["tmp_path"])
|
2012-11-09 06:36:16 +08:00
|
|
|
else:
|
|
|
|
extra = ""
|
2014-01-18 19:31:33 +08:00
|
|
|
|
2012-11-09 06:36:16 +08:00
|
|
|
f.write(rep.nodeid + extra + "\n")
|
2015-08-09 06:07:27 +08:00
|
|
|
|
2023-06-13 03:30:06 +08:00
|
|
|
return rep
|
|
|
|
|
2012-11-09 06:36:16 +08:00
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
if you then have failing tests:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-09 06:36:16 +08:00
|
|
|
|
|
|
|
# content of test_module.py
|
2021-03-14 03:22:54 +08:00
|
|
|
def test_fail1(tmp_path):
|
2014-01-18 19:31:33 +08:00
|
|
|
assert 0
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
|
|
|
2012-11-09 06:36:16 +08:00
|
|
|
def test_fail2():
|
2014-01-18 19:31:33 +08:00
|
|
|
assert 0
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
and run them:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2012-11-09 06:36:16 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest test_module.py
|
2019-01-06 03:19:40 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
rootdir: /home/sweet/project
|
2012-11-09 06:36:16 +08:00
|
|
|
collected 2 items
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
test_module.py FF [100%]
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2019-01-06 03:19:40 +08:00
|
|
|
================================= FAILURES =================================
|
|
|
|
________________________________ test_fail1 ________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2021-10-04 14:56:26 +08:00
|
|
|
tmp_path = PosixPath('PYTEST_TMPDIR/test_fail10')
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2021-03-14 03:22:54 +08:00
|
|
|
def test_fail1(tmp_path):
|
2012-11-09 06:36:16 +08:00
|
|
|
> assert 0
|
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-09 06:36:16 +08:00
|
|
|
test_module.py:2: AssertionError
|
2019-01-06 03:19:40 +08:00
|
|
|
________________________________ test_fail2 ________________________________
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2012-11-09 06:36:16 +08:00
|
|
|
def test_fail2():
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
2018-07-04 08:58:18 +08:00
|
|
|
|
2018-06-05 09:11:27 +08:00
|
|
|
test_module.py:6: AssertionError
|
2020-03-11 22:23:25 +08:00
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED test_module.py::test_fail1 - assert 0
|
|
|
|
FAILED test_module.py::test_fail2 - assert 0
|
2019-08-30 23:43:47 +08:00
|
|
|
============================ 2 failed in 0.12s =============================
|
2012-11-09 06:36:16 +08:00
|
|
|
|
2019-02-15 21:10:37 +08:00
|
|
|
you will have a "failures" file which contains the failing test ids:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
2012-11-09 06:36:16 +08:00
|
|
|
|
|
|
|
$ cat failures
|
2015-09-22 20:02:11 +08:00
|
|
|
test_module.py::test_fail1 (PYTEST_TMPDIR/test_fail10)
|
|
|
|
test_module.py::test_fail2
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
Making test result information available in fixtures
|
|
|
|
-----------------------------------------------------------
|
|
|
|
|
|
|
|
.. regendoc:wipe
|
|
|
|
|
|
|
|
If you want to make test result reports available in fixture finalizers
|
2016-08-23 10:35:41 +08:00
|
|
|
here is a little example implemented via a local plugin:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
# content of conftest.py
|
2023-03-18 04:56:41 +08:00
|
|
|
from typing import Dict
|
2012-11-14 16:39:21 +08:00
|
|
|
import pytest
|
2023-03-18 04:56:41 +08:00
|
|
|
from pytest import StashKey, CollectReport
|
2012-11-14 16:39:21 +08:00
|
|
|
|
2022-11-28 04:09:56 +08:00
|
|
|
phase_report_key = StashKey[Dict[str, CollectReport]]()
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2023-06-13 03:30:06 +08:00
|
|
|
@pytest.hookimpl(wrapper=True, tryfirst=True)
|
2015-08-09 06:07:27 +08:00
|
|
|
def pytest_runtest_makereport(item, call):
|
2012-11-14 16:39:21 +08:00
|
|
|
# execute all other hooks to obtain the report object
|
2023-06-13 03:30:06 +08:00
|
|
|
rep = yield
|
2012-11-14 16:39:21 +08:00
|
|
|
|
2022-11-28 04:09:56 +08:00
|
|
|
# store test results for each phase of a call, which can
|
2012-11-14 16:39:21 +08:00
|
|
|
# be "setup", "call", "teardown"
|
2022-11-28 04:09:56 +08:00
|
|
|
item.stash.setdefault(phase_report_key, {})[rep.when] = rep
|
2023-06-13 03:30:06 +08:00
|
|
|
|
|
|
|
return rep
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def something(request):
|
2016-06-08 07:59:58 +08:00
|
|
|
yield
|
|
|
|
# request.node is an "item" because we use the default
|
|
|
|
# "function" scope
|
2022-11-28 04:09:56 +08:00
|
|
|
report = request.node.stash[phase_report_key]
|
|
|
|
if report["setup"].failed:
|
|
|
|
print("setting up a test failed or skipped", request.node.nodeid)
|
|
|
|
elif ("call" not in report) or report["call"].failed:
|
|
|
|
print("executing test failed or skipped", request.node.nodeid)
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
|
2016-08-23 10:35:41 +08:00
|
|
|
if you then have failing tests:
|
|
|
|
|
|
|
|
.. code-block:: python
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
# content of test_module.py
|
2014-01-18 19:31:33 +08:00
|
|
|
|
2012-11-14 16:39:21 +08:00
|
|
|
import pytest
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-11-14 16:39:21 +08:00
|
|
|
@pytest.fixture
|
|
|
|
def other():
|
|
|
|
assert 0
|
2014-01-18 19:31:33 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-11-14 16:39:21 +08:00
|
|
|
def test_setup_fails(something, other):
|
|
|
|
pass
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-11-14 16:39:21 +08:00
|
|
|
def test_call_fails(something):
|
2014-01-18 19:31:33 +08:00
|
|
|
assert 0
|
2012-11-14 16:39:21 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2012-11-14 16:39:21 +08:00
|
|
|
def test_fail2():
|
2014-01-18 19:31:33 +08:00
|
|
|
assert 0
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
and run it:
|
|
|
|
|
|
|
|
.. code-block:: pytest
|
2012-11-14 16:39:21 +08:00
|
|
|
|
2016-06-21 22:16:57 +08:00
|
|
|
$ pytest -s test_module.py
|
2023-03-18 04:56:41 +08:00
|
|
|
=========================== test session starts ============================
|
2024-01-02 16:58:20 +08:00
|
|
|
platform linux -- Python 3.x.y, pytest-8.x.y, pluggy-1.x.y
|
2023-03-18 04:56:41 +08:00
|
|
|
rootdir: /home/sweet/project
|
|
|
|
collected 3 items
|
|
|
|
|
|
|
|
test_module.py Esetting up a test failed or skipped test_module.py::test_setup_fails
|
|
|
|
Fexecuting test failed or skipped test_module.py::test_call_fails
|
|
|
|
F
|
|
|
|
|
|
|
|
================================== ERRORS ==================================
|
|
|
|
____________________ ERROR at setup of test_setup_fails ____________________
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def other():
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
|
|
|
|
|
|
|
test_module.py:7: AssertionError
|
|
|
|
================================= FAILURES =================================
|
|
|
|
_____________________________ test_call_fails ______________________________
|
|
|
|
|
|
|
|
something = None
|
|
|
|
|
|
|
|
def test_call_fails(something):
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
|
|
|
|
|
|
|
test_module.py:15: AssertionError
|
|
|
|
________________________________ test_fail2 ________________________________
|
|
|
|
|
|
|
|
def test_fail2():
|
|
|
|
> assert 0
|
|
|
|
E assert 0
|
|
|
|
|
|
|
|
test_module.py:19: AssertionError
|
|
|
|
========================= short test summary info ==========================
|
|
|
|
FAILED test_module.py::test_call_fails - assert 0
|
|
|
|
FAILED test_module.py::test_fail2 - assert 0
|
|
|
|
ERROR test_module.py::test_setup_fails - assert 0
|
|
|
|
======================== 2 failed, 1 error in 0.12s ========================
|
2012-11-14 16:39:21 +08:00
|
|
|
|
|
|
|
You'll see that the fixture finalizers could use the precise reporting
|
|
|
|
information.
|
|
|
|
|
2018-03-07 07:40:07 +08:00
|
|
|
.. _pytest current test env:
|
|
|
|
|
2017-07-19 04:18:34 +08:00
|
|
|
``PYTEST_CURRENT_TEST`` environment variable
|
|
|
|
--------------------------------------------
|
|
|
|
|
2019-04-28 23:37:58 +08:00
|
|
|
|
2017-07-19 04:18:34 +08:00
|
|
|
|
|
|
|
Sometimes a test session might get stuck and there might be no easy way to figure out
|
|
|
|
which test got stuck, for example if pytest was run in quiet mode (``-q``) or you don't have access to the console
|
2020-03-27 09:40:25 +08:00
|
|
|
output. This is particularly a problem if the problem happens only sporadically, the famous "flaky" kind of tests.
|
2017-07-19 04:18:34 +08:00
|
|
|
|
2020-03-27 09:40:25 +08:00
|
|
|
``pytest`` sets the :envvar:`PYTEST_CURRENT_TEST` environment variable when running tests, which can be inspected
|
2021-10-22 20:47:57 +08:00
|
|
|
by process monitoring utilities or libraries like :pypi:`psutil` to discover which test got stuck if necessary:
|
2017-07-19 04:18:34 +08:00
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
import psutil
|
|
|
|
|
|
|
|
for pid in psutil.pids():
|
|
|
|
environ = psutil.Process(pid).environ()
|
2018-06-03 11:29:28 +08:00
|
|
|
if "PYTEST_CURRENT_TEST" in environ:
|
2017-07-19 04:18:34 +08:00
|
|
|
print(f'pytest process {pid} running: {environ["PYTEST_CURRENT_TEST"]}')
|
|
|
|
|
|
|
|
During the test session pytest will set ``PYTEST_CURRENT_TEST`` to the current test
|
2020-03-27 09:40:25 +08:00
|
|
|
:ref:`nodeid <nodeids>` and the current stage, which can be ``setup``, ``call``,
|
|
|
|
or ``teardown``.
|
2017-07-19 04:18:34 +08:00
|
|
|
|
|
|
|
For example, when running a single test function named ``test_foo`` from ``foo_module.py``,
|
|
|
|
``PYTEST_CURRENT_TEST`` will be set to:
|
|
|
|
|
|
|
|
#. ``foo_module.py::test_foo (setup)``
|
|
|
|
#. ``foo_module.py::test_foo (call)``
|
|
|
|
#. ``foo_module.py::test_foo (teardown)``
|
|
|
|
|
|
|
|
In that order.
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
The contents of ``PYTEST_CURRENT_TEST`` is meant to be human readable and the actual format
|
|
|
|
can be changed between releases (even bug fixes) so it shouldn't be relied on for scripting
|
|
|
|
or automation.
|
|
|
|
|
2018-10-02 22:54:59 +08:00
|
|
|
.. _freezing-pytest:
|
|
|
|
|
2018-05-18 16:19:46 +08:00
|
|
|
Freezing pytest
|
2016-07-27 19:49:00 +08:00
|
|
|
---------------
|
2014-07-29 09:40:23 +08:00
|
|
|
|
|
|
|
If you freeze your application using a tool like
|
2016-07-27 08:29:07 +08:00
|
|
|
`PyInstaller <https://pyinstaller.readthedocs.io>`_
|
|
|
|
in order to distribute it to your end-users, it is a good idea to also package
|
|
|
|
your test runner and run your tests using the frozen application. This way packaging
|
|
|
|
errors such as dependencies not being included into the executable can be detected early
|
|
|
|
while also allowing you to send test files to users so they can run them in their
|
2016-07-27 19:49:00 +08:00
|
|
|
machines, which can be useful to obtain more information about a hard to reproduce bug.
|
2016-07-27 08:29:07 +08:00
|
|
|
|
2016-07-27 19:49:00 +08:00
|
|
|
Fortunately recent ``PyInstaller`` releases already have a custom hook
|
2018-05-18 16:19:46 +08:00
|
|
|
for pytest, but if you are using another tool to freeze executables
|
2016-07-27 19:49:00 +08:00
|
|
|
such as ``cx_freeze`` or ``py2exe``, you can use ``pytest.freeze_includes()``
|
|
|
|
to obtain the full list of internal pytest modules. How to configure the tools
|
|
|
|
to find the internal modules varies from tool to tool, however.
|
2014-07-29 09:40:23 +08:00
|
|
|
|
2018-05-18 16:19:46 +08:00
|
|
|
Instead of freezing the pytest runner as a separate executable, you can make
|
2016-07-27 19:49:00 +08:00
|
|
|
your frozen program work as the pytest runner by some clever
|
2018-05-18 16:19:46 +08:00
|
|
|
argument handling during program startup. This allows you to
|
2016-07-27 19:49:00 +08:00
|
|
|
have a single executable, which is usually more convenient.
|
2018-01-10 23:44:26 +08:00
|
|
|
Please note that the mechanism for plugin discovery used by pytest
|
2021-06-05 23:03:59 +08:00
|
|
|
(setuptools entry points) doesn't work with frozen executables so pytest
|
2018-05-18 16:19:46 +08:00
|
|
|
can't find any third party plugins automatically. To include third party plugins
|
2018-01-10 23:44:26 +08:00
|
|
|
like ``pytest-timeout`` they must be imported explicitly and passed on to pytest.main.
|
2014-07-29 09:40:23 +08:00
|
|
|
|
2016-07-27 19:49:00 +08:00
|
|
|
.. code-block:: python
|
2014-07-29 09:40:23 +08:00
|
|
|
|
|
|
|
# contents of app_main.py
|
|
|
|
import sys
|
2022-04-28 22:30:16 +08:00
|
|
|
|
2018-01-10 23:44:26 +08:00
|
|
|
import pytest_timeout # Third party plugin
|
2014-07-29 09:40:23 +08:00
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
if len(sys.argv) > 1 and sys.argv[1] == "--pytest":
|
2014-07-29 09:40:23 +08:00
|
|
|
import pytest
|
2018-06-03 11:29:28 +08:00
|
|
|
|
2018-01-10 23:44:26 +08:00
|
|
|
sys.exit(pytest.main(sys.argv[2:], plugins=[pytest_timeout]))
|
2014-07-29 09:40:23 +08:00
|
|
|
else:
|
|
|
|
# normal application execution: at this point argv can be parsed
|
|
|
|
# by your argument-parsing library of choice as usual
|
|
|
|
...
|
|
|
|
|
2016-08-07 04:58:17 +08:00
|
|
|
|
2016-07-27 19:49:00 +08:00
|
|
|
This allows you to execute tests using the frozen
|
2019-02-15 21:10:37 +08:00
|
|
|
application with standard ``pytest`` command-line options:
|
|
|
|
|
|
|
|
.. code-block:: bash
|
2014-07-29 09:40:23 +08:00
|
|
|
|
2023-10-11 05:16:24 +08:00
|
|
|
./app_main --pytest --verbose --tb=long --junit=xml=results.xml test-suite/
|