2015-04-25 17:29:11 +08:00
|
|
|
|
.. _plugins:
|
|
|
|
|
.. _`writing-plugins`:
|
|
|
|
|
|
|
|
|
|
Writing plugins
|
|
|
|
|
===============
|
|
|
|
|
|
|
|
|
|
It is easy to implement `local conftest plugins`_ for your own project
|
|
|
|
|
or `pip-installable plugins`_ that can be used throughout many projects,
|
|
|
|
|
including third party projects. Please refer to :ref:`using plugins` if you
|
|
|
|
|
only want to use but not write plugins.
|
|
|
|
|
|
|
|
|
|
A plugin contains one or multiple hook functions. :ref:`Writing hooks <writinghooks>`
|
|
|
|
|
explains the basics and details of how you can write a hook function yourself.
|
|
|
|
|
``pytest`` implements all aspects of configuration, collection, running and
|
2018-02-08 06:53:31 +08:00
|
|
|
|
reporting by calling :ref:`well specified hooks <hook-reference>` of the following plugins:
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2018-02-08 06:53:31 +08:00
|
|
|
|
* builtin plugins: loaded from pytest's internal ``_pytest`` directory.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2015-12-11 05:00:01 +08:00
|
|
|
|
* :ref:`external plugins <extplugins>`: modules discovered through
|
2015-04-25 17:29:11 +08:00
|
|
|
|
`setuptools entry points`_
|
|
|
|
|
|
|
|
|
|
* `conftest.py plugins`_: modules auto-discovered in test directories
|
|
|
|
|
|
|
|
|
|
In principle, each hook call is a ``1:N`` Python function call where ``N`` is the
|
|
|
|
|
number of registered implementation functions for a given specification.
|
2016-03-21 00:01:04 +08:00
|
|
|
|
All specifications and implementations follow the ``pytest_`` prefix
|
2015-04-25 17:29:11 +08:00
|
|
|
|
naming convention, making them easy to distinguish and find.
|
|
|
|
|
|
|
|
|
|
.. _`pluginorder`:
|
|
|
|
|
|
|
|
|
|
Plugin discovery order at tool startup
|
|
|
|
|
--------------------------------------
|
|
|
|
|
|
|
|
|
|
``pytest`` loads plugin modules at tool startup in the following way:
|
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
1. by scanning the command line for the ``-p no:name`` option
|
|
|
|
|
and *blocking* that plugin from being loaded (even builtin plugins can
|
|
|
|
|
be blocked this way). This happens before normal command-line parsing.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
2. by loading all builtin plugins.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
3. by scanning the command line for the ``-p name`` option
|
|
|
|
|
and loading the specified plugin. This happens before normal command-line parsing.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
4. by loading all plugins registered through `setuptools entry points`_.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
5. by loading all plugins specified through the :envvar:`PYTEST_PLUGINS` environment variable.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-13 01:11:43 +08:00
|
|
|
|
6. by loading all :file:`conftest.py` files as inferred by the command line
|
|
|
|
|
invocation:
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-09-20 03:14:28 +08:00
|
|
|
|
- if no test paths are specified, use the current dir as a test path
|
2020-09-13 01:11:43 +08:00
|
|
|
|
- if exists, load ``conftest.py`` and ``test*/conftest.py`` relative
|
|
|
|
|
to the directory part of the first test path. After the ``conftest.py``
|
|
|
|
|
file is loaded, load all plugins specified in its
|
|
|
|
|
:globalvar:`pytest_plugins` variable if present.
|
|
|
|
|
|
|
|
|
|
Note that pytest does not find ``conftest.py`` files in deeper nested
|
|
|
|
|
sub directories at tool startup. It is usually a good idea to keep
|
|
|
|
|
your ``conftest.py`` file in the top level test or project root directory.
|
|
|
|
|
|
|
|
|
|
7. by recursively loading all plugins specified by the
|
|
|
|
|
:globalvar:`pytest_plugins` variable in ``conftest.py`` files.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
.. _`pytest/plugin`: http://bitbucket.org/pytest-dev/pytest/src/tip/pytest/plugin/
|
|
|
|
|
.. _`conftest.py plugins`:
|
|
|
|
|
.. _`localplugin`:
|
|
|
|
|
.. _`local conftest plugins`:
|
|
|
|
|
|
|
|
|
|
conftest.py: local per-directory plugins
|
|
|
|
|
----------------------------------------
|
|
|
|
|
|
|
|
|
|
Local ``conftest.py`` plugins contain directory-specific hook
|
|
|
|
|
implementations. Hook Session and test running activities will
|
|
|
|
|
invoke all hooks defined in ``conftest.py`` files closer to the
|
|
|
|
|
root of the filesystem. Example of implementing the
|
|
|
|
|
``pytest_runtest_setup`` hook so that is called for tests in the ``a``
|
|
|
|
|
sub directory but not for other directories::
|
|
|
|
|
|
|
|
|
|
a/conftest.py:
|
|
|
|
|
def pytest_runtest_setup(item):
|
|
|
|
|
# called for running each test in 'a' directory
|
2018-11-22 16:15:14 +08:00
|
|
|
|
print("setting up", item)
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
a/test_sub.py:
|
|
|
|
|
def test_sub():
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
test_flat.py:
|
|
|
|
|
def test_flat():
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
Here is how you might run it::
|
|
|
|
|
|
2018-04-06 02:34:26 +08:00
|
|
|
|
pytest test_flat.py --capture=no # will not show "setting up"
|
|
|
|
|
pytest a/test_sub.py --capture=no # will show "setting up"
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2017-07-28 07:02:18 +08:00
|
|
|
|
.. note::
|
2015-04-25 17:29:11 +08:00
|
|
|
|
If you have ``conftest.py`` files which do not reside in a
|
|
|
|
|
python package directory (i.e. one containing an ``__init__.py``) then
|
|
|
|
|
"import conftest" can be ambiguous because there might be other
|
2017-07-20 08:11:17 +08:00
|
|
|
|
``conftest.py`` files as well on your ``PYTHONPATH`` or ``sys.path``.
|
2016-01-20 23:35:27 +08:00
|
|
|
|
It is thus good practice for projects to either put ``conftest.py``
|
2015-04-25 17:29:11 +08:00
|
|
|
|
under a package scope or to never import anything from a
|
2017-07-20 08:11:17 +08:00
|
|
|
|
``conftest.py`` file.
|
|
|
|
|
|
|
|
|
|
See also: :ref:`pythonpath`.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-10-17 19:26:30 +08:00
|
|
|
|
.. note::
|
|
|
|
|
Some hooks should be implemented only in plugins or conftest.py files situated at the
|
|
|
|
|
tests root directory due to how pytest discovers plugins during startup,
|
|
|
|
|
see the documentation of each hook for details.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2015-08-26 10:54:05 +08:00
|
|
|
|
Writing your own plugin
|
|
|
|
|
-----------------------
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
If you want to write a plugin, there are many real-life examples
|
|
|
|
|
you can copy from:
|
|
|
|
|
|
|
|
|
|
* a custom collection example plugin: :ref:`yaml plugin`
|
2018-02-08 06:53:31 +08:00
|
|
|
|
* builtin plugins which provide pytest's own functionality
|
2021-03-12 04:30:47 +08:00
|
|
|
|
* many :ref:`external plugins <plugin-list>` providing additional features
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2018-02-08 06:53:31 +08:00
|
|
|
|
All of these plugins implement :ref:`hooks <hook-reference>` and/or :ref:`fixtures <fixture>`
|
2015-04-25 17:29:11 +08:00
|
|
|
|
to extend and add functionality.
|
|
|
|
|
|
2015-08-26 10:54:05 +08:00
|
|
|
|
.. note::
|
|
|
|
|
Make sure to check out the excellent
|
|
|
|
|
`cookiecutter-pytest-plugin <https://github.com/pytest-dev/cookiecutter-pytest-plugin>`_
|
|
|
|
|
project, which is a `cookiecutter template <https://github.com/audreyr/cookiecutter>`_
|
|
|
|
|
for authoring plugins.
|
|
|
|
|
|
|
|
|
|
The template provides an excellent starting point with a working plugin,
|
2017-07-15 17:45:28 +08:00
|
|
|
|
tests running with tox, a comprehensive README file as well as a
|
|
|
|
|
pre-configured entry-point.
|
2015-08-26 10:54:05 +08:00
|
|
|
|
|
|
|
|
|
Also consider :ref:`contributing your plugin to pytest-dev<submitplugin>`
|
2015-04-25 17:29:11 +08:00
|
|
|
|
once it has some happy users other than yourself.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
.. _`setuptools entry points`:
|
|
|
|
|
.. _`pip-installable plugins`:
|
|
|
|
|
|
|
|
|
|
Making your plugin installable by others
|
|
|
|
|
----------------------------------------
|
|
|
|
|
|
|
|
|
|
If you want to make your plugin externally available, you
|
|
|
|
|
may define a so-called entry point for your distribution so
|
|
|
|
|
that ``pytest`` finds your plugin module. Entry points are
|
2021-10-22 20:47:57 +08:00
|
|
|
|
a feature that is provided by :std:doc:`setuptools:index`. pytest looks up
|
2015-04-25 17:29:11 +08:00
|
|
|
|
the ``pytest11`` entrypoint to discover its
|
|
|
|
|
plugins and you can thus make your plugin available by defining
|
|
|
|
|
it in your setuptools-invocation:
|
|
|
|
|
|
|
|
|
|
.. sourcecode:: python
|
|
|
|
|
|
|
|
|
|
# sample ./setup.py file
|
|
|
|
|
from setuptools import setup
|
|
|
|
|
|
2022-06-24 08:54:51 +08:00
|
|
|
|
|
|
|
|
|
name_of_plugin = "myproject" # register plugin with this name
|
2015-04-25 17:29:11 +08:00
|
|
|
|
setup(
|
|
|
|
|
name="myproject",
|
2018-06-03 11:29:28 +08:00
|
|
|
|
packages=["myproject"],
|
2015-04-25 17:29:11 +08:00
|
|
|
|
# the following makes a plugin available to pytest
|
2022-06-24 08:54:51 +08:00
|
|
|
|
entry_points={"pytest11": [f"{name_of_plugin} = myproject.pluginmodule"]},
|
2016-03-23 07:42:52 +08:00
|
|
|
|
# custom PyPI classifier for pytest plugins
|
2018-06-03 11:29:28 +08:00
|
|
|
|
classifiers=["Framework :: Pytest"],
|
2015-04-25 17:29:11 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
If a package is installed this way, ``pytest`` will load
|
|
|
|
|
``myproject.pluginmodule`` as a plugin which can define
|
2022-06-24 08:54:51 +08:00
|
|
|
|
:ref:`hooks <hook-reference>`. Confirm registration with ``pytest --trace-config``
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2016-03-23 07:42:52 +08:00
|
|
|
|
.. note::
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2016-03-23 07:42:52 +08:00
|
|
|
|
Make sure to include ``Framework :: Pytest`` in your list of
|
2019-07-16 04:50:45 +08:00
|
|
|
|
`PyPI classifiers <https://pypi.org/classifiers/>`_
|
2016-03-23 07:42:52 +08:00
|
|
|
|
to make it easy for users to find your plugin.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
|
2018-03-02 04:54:54 +08:00
|
|
|
|
.. _assertion-rewriting:
|
|
|
|
|
|
2016-07-17 19:30:21 +08:00
|
|
|
|
Assertion Rewriting
|
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
|
|
One of the main features of ``pytest`` is the use of plain assert
|
|
|
|
|
statements and the detailed introspection of expressions upon
|
|
|
|
|
assertion failures. This is provided by "assertion rewriting" which
|
|
|
|
|
modifies the parsed AST before it gets compiled to bytecode. This is
|
|
|
|
|
done via a :pep:`302` import hook which gets installed early on when
|
2017-11-27 03:46:06 +08:00
|
|
|
|
``pytest`` starts up and will perform this rewriting when modules get
|
2020-02-16 17:30:25 +08:00
|
|
|
|
imported. However, since we do not want to test different bytecode
|
|
|
|
|
from what you will run in production, this hook only rewrites test modules
|
|
|
|
|
themselves (as defined by the :confval:`python_files` configuration option),
|
|
|
|
|
and any modules which are part of plugins.
|
|
|
|
|
Any other imported module will not be rewritten and normal assertion behaviour
|
|
|
|
|
will happen.
|
2016-07-17 19:30:21 +08:00
|
|
|
|
|
|
|
|
|
If you have assertion helpers in other modules where you would need
|
|
|
|
|
assertion rewriting to be enabled you need to ask ``pytest``
|
2017-11-27 03:46:06 +08:00
|
|
|
|
explicitly to rewrite this module before it gets imported.
|
2016-07-17 19:30:21 +08:00
|
|
|
|
|
|
|
|
|
.. autofunction:: pytest.register_assert_rewrite
|
2018-03-02 04:54:54 +08:00
|
|
|
|
:noindex:
|
2016-07-17 19:30:21 +08:00
|
|
|
|
|
|
|
|
|
This is especially important when you write a pytest plugin which is
|
|
|
|
|
created using a package. The import hook only treats ``conftest.py``
|
|
|
|
|
files and any modules which are listed in the ``pytest11`` entrypoint
|
|
|
|
|
as plugins. As an example consider the following package::
|
|
|
|
|
|
|
|
|
|
pytest_foo/__init__.py
|
|
|
|
|
pytest_foo/plugin.py
|
|
|
|
|
pytest_foo/helper.py
|
|
|
|
|
|
|
|
|
|
With the following typical ``setup.py`` extract:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
setup(..., entry_points={"pytest11": ["foo = pytest_foo.plugin"]}, ...)
|
2016-07-17 19:30:21 +08:00
|
|
|
|
|
2017-11-27 03:46:06 +08:00
|
|
|
|
In this case only ``pytest_foo/plugin.py`` will be rewritten. If the
|
2016-07-17 19:30:21 +08:00
|
|
|
|
helper module also contains assert statements which need to be
|
2017-11-27 03:46:06 +08:00
|
|
|
|
rewritten it needs to be marked as such, before it gets imported.
|
|
|
|
|
This is easiest by marking it for rewriting inside the
|
2016-07-17 19:30:21 +08:00
|
|
|
|
``__init__.py`` module, which will always be imported first when a
|
|
|
|
|
module inside a package is imported. This way ``plugin.py`` can still
|
|
|
|
|
import ``helper.py`` normally. The contents of
|
|
|
|
|
``pytest_foo/__init__.py`` will then need to look like this:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
pytest.register_assert_rewrite("pytest_foo.helper")
|
2016-07-17 19:30:21 +08:00
|
|
|
|
|
|
|
|
|
|
2015-04-25 17:29:11 +08:00
|
|
|
|
Requiring/Loading plugins in a test module or conftest file
|
|
|
|
|
-----------------------------------------------------------
|
|
|
|
|
|
2020-07-10 20:50:03 +08:00
|
|
|
|
You can require plugins in a test module or a ``conftest.py`` file using :globalvar:`pytest_plugins`:
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2017-02-15 21:57:03 +08:00
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
pytest_plugins = ["name1", "name2"]
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
When the test module or conftest plugin is loaded the specified plugins
|
2017-02-15 21:57:03 +08:00
|
|
|
|
will be loaded as well. Any module can be blessed as a plugin, including internal
|
|
|
|
|
application modules:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
pytest_plugins = "myapp.testsupport.myplugin"
|
|
|
|
|
|
2020-07-10 20:50:03 +08:00
|
|
|
|
:globalvar:`pytest_plugins` are processed recursively, so note that in the example above
|
|
|
|
|
if ``myapp.testsupport.myplugin`` also declares :globalvar:`pytest_plugins`, the contents
|
2017-02-15 21:57:03 +08:00
|
|
|
|
of the variable will also be loaded as plugins, and so on.
|
|
|
|
|
|
2018-03-13 05:32:51 +08:00
|
|
|
|
.. _`requiring plugins in non-root conftests`:
|
2018-02-21 07:27:24 +08:00
|
|
|
|
|
|
|
|
|
.. note::
|
2020-07-10 20:50:03 +08:00
|
|
|
|
Requiring plugins using :globalvar:`pytest_plugins` variable in non-root
|
2018-02-21 07:27:24 +08:00
|
|
|
|
``conftest.py`` files is deprecated.
|
|
|
|
|
|
|
|
|
|
This is important because ``conftest.py`` files implement per-directory
|
|
|
|
|
hook implementations, but once a plugin is imported, it will affect the
|
|
|
|
|
entire directory tree. In order to avoid confusion, defining
|
2020-07-10 20:50:03 +08:00
|
|
|
|
:globalvar:`pytest_plugins` in any ``conftest.py`` file which is not located in the
|
2018-02-21 07:27:24 +08:00
|
|
|
|
tests root directory is deprecated, and will raise a warning.
|
|
|
|
|
|
2017-02-15 23:15:53 +08:00
|
|
|
|
This mechanism makes it easy to share fixtures within applications or even
|
2017-06-13 10:45:35 +08:00
|
|
|
|
external applications without the need to create external plugins using
|
2017-02-15 21:57:03 +08:00
|
|
|
|
the ``setuptools``'s entry point technique.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2020-07-10 20:50:03 +08:00
|
|
|
|
Plugins imported by :globalvar:`pytest_plugins` will also automatically be marked
|
2017-06-13 10:45:35 +08:00
|
|
|
|
for assertion rewriting (see :func:`pytest.register_assert_rewrite`).
|
2017-02-15 21:57:03 +08:00
|
|
|
|
However for this to have any effect the module must not be
|
2017-01-01 01:54:47 +08:00
|
|
|
|
imported already; if it was already imported at the time the
|
2020-07-10 20:50:03 +08:00
|
|
|
|
:globalvar:`pytest_plugins` statement is processed, a warning will result and
|
2017-11-27 03:46:06 +08:00
|
|
|
|
assertions inside the plugin will not be rewritten. To fix this you
|
2016-07-17 19:30:21 +08:00
|
|
|
|
can either call :func:`pytest.register_assert_rewrite` yourself before
|
|
|
|
|
the module is imported, or you can arrange the code to delay the
|
|
|
|
|
importing until after the plugin is registered.
|
|
|
|
|
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
Accessing another plugin by name
|
|
|
|
|
--------------------------------
|
|
|
|
|
|
|
|
|
|
If a plugin wants to collaborate with code from
|
|
|
|
|
another plugin it can obtain a reference through
|
|
|
|
|
the plugin manager like this:
|
|
|
|
|
|
|
|
|
|
.. sourcecode:: python
|
|
|
|
|
|
2018-02-23 02:48:59 +08:00
|
|
|
|
plugin = config.pluginmanager.get_plugin("name_of_plugin")
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
|
|
|
|
If you want to look at the names of existing plugins, use
|
2016-03-21 00:12:50 +08:00
|
|
|
|
the ``--trace-config`` option.
|
2015-04-25 17:29:11 +08:00
|
|
|
|
|
2019-03-31 11:22:30 +08:00
|
|
|
|
|
|
|
|
|
.. _registering-markers:
|
|
|
|
|
|
|
|
|
|
Registering custom markers
|
|
|
|
|
--------------------------
|
|
|
|
|
|
|
|
|
|
If your plugin uses any markers, you should register them so that they appear in
|
|
|
|
|
pytest's help text and do not :ref:`cause spurious warnings <unknown-marks>`.
|
|
|
|
|
For example, the following plugin would register ``cool_marker`` and
|
|
|
|
|
``mark_with`` for all users:
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
def pytest_configure(config):
|
|
|
|
|
config.addinivalue_line("markers", "cool_marker: this one is for cool tests.")
|
|
|
|
|
config.addinivalue_line(
|
|
|
|
|
"markers", "mark_with(arg, arg2): this marker takes arguments."
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
2015-04-28 17:54:53 +08:00
|
|
|
|
Testing plugins
|
|
|
|
|
---------------
|
|
|
|
|
|
2017-07-15 22:48:02 +08:00
|
|
|
|
pytest comes with a plugin named ``pytester`` that helps you write tests for
|
|
|
|
|
your plugin code. The plugin is disabled by default, so you will have to enable
|
|
|
|
|
it before you can use it.
|
2015-04-28 17:54:53 +08:00
|
|
|
|
|
2017-07-15 22:48:02 +08:00
|
|
|
|
You can do so by adding the following line to a ``conftest.py`` file in your
|
|
|
|
|
testing directory:
|
2015-04-28 17:54:53 +08:00
|
|
|
|
|
2017-07-15 22:48:02 +08:00
|
|
|
|
.. code-block:: python
|
2015-04-28 17:54:53 +08:00
|
|
|
|
|
2017-07-15 22:48:02 +08:00
|
|
|
|
# content of conftest.py
|
|
|
|
|
|
|
|
|
|
pytest_plugins = ["pytester"]
|
|
|
|
|
|
|
|
|
|
Alternatively you can invoke pytest with the ``-p pytester`` command line
|
|
|
|
|
option.
|
|
|
|
|
|
2021-05-24 17:31:28 +08:00
|
|
|
|
This will allow you to use the :py:class:`pytester <pytest.Pytester>`
|
2017-07-15 22:48:02 +08:00
|
|
|
|
fixture for testing your plugin code.
|
|
|
|
|
|
|
|
|
|
Let's demonstrate what you can do with the plugin with an example. Imagine we
|
|
|
|
|
developed a plugin that provides a fixture ``hello`` which yields a function
|
|
|
|
|
and we can invoke this function with one optional parameter. It will return a
|
|
|
|
|
string value of ``Hello World!`` if we do not supply a value or ``Hello
|
|
|
|
|
{value}!`` if we do supply a string value.
|
|
|
|
|
|
2017-07-20 02:14:46 +08:00
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
2019-10-31 04:25:50 +08:00
|
|
|
|
def pytest_addoption(parser):
|
2018-06-03 11:29:28 +08:00
|
|
|
|
group = parser.getgroup("helloworld")
|
2017-07-20 02:14:46 +08:00
|
|
|
|
group.addoption(
|
2018-06-03 11:29:28 +08:00
|
|
|
|
"--name",
|
|
|
|
|
action="store",
|
|
|
|
|
dest="name",
|
|
|
|
|
default="World",
|
|
|
|
|
help='Default "name" for hello().',
|
2017-07-20 02:14:46 +08:00
|
|
|
|
)
|
|
|
|
|
|
2018-06-03 11:29:28 +08:00
|
|
|
|
|
2017-07-20 02:14:46 +08:00
|
|
|
|
@pytest.fixture
|
|
|
|
|
def hello(request):
|
2018-06-03 11:29:28 +08:00
|
|
|
|
name = request.config.getoption("name")
|
2017-07-20 02:14:46 +08:00
|
|
|
|
|
|
|
|
|
def _hello(name=None):
|
|
|
|
|
if not name:
|
2018-06-03 11:29:28 +08:00
|
|
|
|
name = request.config.getoption("name")
|
2022-04-28 22:30:16 +08:00
|
|
|
|
return f"Hello {name}!"
|
2017-07-20 02:14:46 +08:00
|
|
|
|
|
|
|
|
|
return _hello
|
|
|
|
|
|
|
|
|
|
|
2021-01-02 04:21:39 +08:00
|
|
|
|
Now the ``pytester`` fixture provides a convenient API for creating temporary
|
2017-07-15 22:48:02 +08:00
|
|
|
|
``conftest.py`` files and test files. It also allows us to run the tests and
|
|
|
|
|
return a result object, with which we can assert the tests' outcomes.
|
|
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
2021-01-02 04:21:39 +08:00
|
|
|
|
def test_hello(pytester):
|
2017-07-15 22:48:02 +08:00
|
|
|
|
"""Make sure that our plugin works."""
|
|
|
|
|
|
|
|
|
|
# create a temporary conftest.py file
|
2021-01-02 04:21:39 +08:00
|
|
|
|
pytester.makeconftest(
|
2018-06-03 11:29:28 +08:00
|
|
|
|
"""
|
2017-07-15 22:48:02 +08:00
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(params=[
|
|
|
|
|
"Brianna",
|
|
|
|
|
"Andreas",
|
|
|
|
|
"Floris",
|
|
|
|
|
])
|
|
|
|
|
def name(request):
|
|
|
|
|
return request.param
|
2018-06-03 11:29:28 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2017-07-15 22:48:02 +08:00
|
|
|
|
|
|
|
|
|
# create a temporary pytest test file
|
2021-01-02 04:21:39 +08:00
|
|
|
|
pytester.makepyfile(
|
2018-06-03 11:29:28 +08:00
|
|
|
|
"""
|
2017-07-15 22:48:02 +08:00
|
|
|
|
def test_hello_default(hello):
|
|
|
|
|
assert hello() == "Hello World!"
|
|
|
|
|
|
|
|
|
|
def test_hello_name(hello, name):
|
|
|
|
|
assert hello(name) == "Hello {0}!".format(name)
|
2018-06-03 11:29:28 +08:00
|
|
|
|
"""
|
|
|
|
|
)
|
2015-04-28 17:54:53 +08:00
|
|
|
|
|
2017-07-15 22:48:02 +08:00
|
|
|
|
# run all tests with pytest
|
2021-01-02 04:21:39 +08:00
|
|
|
|
result = pytester.runpytest()
|
2017-07-15 22:48:02 +08:00
|
|
|
|
|
|
|
|
|
# check that all 4 tests passed
|
|
|
|
|
result.assert_outcomes(passed=4)
|
|
|
|
|
|
|
|
|
|
|
2021-09-22 06:16:52 +08:00
|
|
|
|
Additionally it is possible to copy examples to the ``pytester``'s isolated environment
|
|
|
|
|
before running pytest on it. This way we can abstract the tested logic to separate files,
|
|
|
|
|
which is especially useful for longer tests and/or longer ``conftest.py`` files.
|
|
|
|
|
|
|
|
|
|
Note that for ``pytester.copy_example`` to work we need to set `pytester_example_dir`
|
|
|
|
|
in our ``pytest.ini`` to tell pytest where to look for example files.
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
|
.. code-block:: ini
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2018-06-27 14:28:21 +08:00
|
|
|
|
# content of pytest.ini
|
2018-06-27 12:52:54 +08:00
|
|
|
|
[pytest]
|
|
|
|
|
pytester_example_dir = .
|
|
|
|
|
|
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
|
.. code-block:: python
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2018-06-27 14:28:21 +08:00
|
|
|
|
# content of test_example.py
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
|
|
|
|
|
2021-01-02 04:21:39 +08:00
|
|
|
|
def test_plugin(pytester):
|
|
|
|
|
pytester.copy_example("test_example.py")
|
|
|
|
|
pytester.runpytest("-k", "test_example")
|
2018-11-24 13:41:22 +08:00
|
|
|
|
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
|
|
|
|
def test_example():
|
2018-11-24 13:41:22 +08:00
|
|
|
|
pass
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2018-11-24 13:41:22 +08:00
|
|
|
|
.. code-block:: pytest
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2018-06-27 14:28:21 +08:00
|
|
|
|
$ pytest
|
2018-06-27 12:52:54 +08:00
|
|
|
|
=========================== test session starts ============================
|
2021-12-07 17:13:36 +08:00
|
|
|
|
platform linux -- Python 3.x.y, pytest-7.x.y, pluggy-1.x.y
|
2021-10-04 14:56:26 +08:00
|
|
|
|
rootdir: /home/sweet/project, configfile: pytest.ini
|
2018-06-27 14:28:21 +08:00
|
|
|
|
collected 2 items
|
|
|
|
|
|
|
|
|
|
test_example.py .. [100%]
|
|
|
|
|
|
2020-12-13 05:21:28 +08:00
|
|
|
|
============================ 2 passed in 0.12s =============================
|
2018-06-27 12:52:54 +08:00
|
|
|
|
|
2017-07-20 01:42:33 +08:00
|
|
|
|
For more information about the result object that ``runpytest()`` returns, and
|
2017-07-15 22:48:02 +08:00
|
|
|
|
the methods that it provides please check out the :py:class:`RunResult
|
|
|
|
|
<_pytest.pytester.RunResult>` documentation.
|