From 93a68cdfb4731b3592002a2dbdd7d18e8afb7919 Mon Sep 17 00:00:00 2001 From: Niklas Meinzer Date: Fri, 28 Jun 2019 18:13:47 +0200 Subject: [PATCH 1/2] doc(plugin_hooks): Improve documentation for writing plugin hooks. --- changelog/5517.doc.rst | 1 + doc/en/writing_plugins.rst | 52 ++++++++++++++++++++++++++++++++++++-- 2 files changed, 51 insertions(+), 2 deletions(-) create mode 100644 changelog/5517.doc.rst diff --git a/changelog/5517.doc.rst b/changelog/5517.doc.rst new file mode 100644 index 000000000..92102c983 --- /dev/null +++ b/changelog/5517.doc.rst @@ -0,0 +1 @@ +Improve "Declaring new hooks" section in chapter "Writing Plugins" diff --git a/doc/en/writing_plugins.rst b/doc/en/writing_plugins.rst index 469ee57ca..4fb1948b9 100644 --- a/doc/en/writing_plugins.rst +++ b/doc/en/writing_plugins.rst @@ -621,12 +621,60 @@ the new plugin: Hooks are usually declared as do-nothing functions that contain only documentation describing when the hook will be called and what return values -are expected. +are expected. The names of the functions must start with `pytest_` otherwise pytest won't recognize them. -For an example, see `newhooks.py`_ from `xdist `_. +Here's an example. Let's assume this code is in the ``hooks.py`` module. + +.. code-block:: python + + def pytest_my_hook(config): + """ + Receives the pytest config and does things with it + """ + +To register the hooks with pytest they need to be structured in their own module or class. This +class or module can then be passed to the ``pluginmanager`` using the ``pytest_addhooks`` function +(which itself is a hook exposed by pytest). + +.. code-block:: python + + def pytest_addhooks(pluginmanager): + """ This example assumes the hooks are grouped in the 'hooks' module. """ + from my_app.tests import hooks + pluginmanager.add_hookspecs(hooks) + +For a real world example, see `newhooks.py`_ from `xdist `_. .. _`newhooks.py`: https://github.com/pytest-dev/pytest-xdist/blob/974bd566c599dc6a9ea291838c6f226197208b46/xdist/newhooks.py +Hooks may be called both from fixtures or from other hooks. In both cases, hooks are called +through the ``hook`` object, available in the ``config`` object. Most hooks receive a +``config`` object directly, while fixtures may use the ``pytestconfig`` fixture which provides the same object. + +.. code-block:: python + + @pytest.fixture() + def my_fixture(pytestconfig): + # call the hook called "pytest_my_hook" + # `result` will be a list of return values from all registered functions. + result = pytestconfig.hook.pytest_my_hook(config=pytestconfig) + +.. note:: + Hooks receive parameters using only keyword arguments. + +Now your hook is ready to be used. To register a function at the hook, other plugins or users must +now simply define the function ``pytest_my_hook`` with the correct signature in their ``conftest.py``. + +Example: + +.. code-block:: python + + def pytest_my_hook(config): + """ + Print all active hooks to the screen. + """ + print(config.hook) + Optionally using hooks from 3rd party plugins --------------------------------------------- From 789a3662ce7aa1dad3eea208d6d8dc191f50e506 Mon Sep 17 00:00:00 2001 From: Bruno Oliveira Date: Sat, 29 Jun 2019 09:11:09 -0300 Subject: [PATCH 2/2] Fix linting --- doc/en/writing_plugins.rst | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/doc/en/writing_plugins.rst b/doc/en/writing_plugins.rst index 4fb1948b9..1596eeb54 100644 --- a/doc/en/writing_plugins.rst +++ b/doc/en/writing_plugins.rst @@ -641,6 +641,7 @@ class or module can then be passed to the ``pluginmanager`` using the ``pytest_a def pytest_addhooks(pluginmanager): """ This example assumes the hooks are grouped in the 'hooks' module. """ from my_app.tests import hooks + pluginmanager.add_hookspecs(hooks) For a real world example, see `newhooks.py`_ from `xdist `_. @@ -656,7 +657,7 @@ through the ``hook`` object, available in the ``config`` object. Most hooks rece @pytest.fixture() def my_fixture(pytestconfig): # call the hook called "pytest_my_hook" - # `result` will be a list of return values from all registered functions. + # 'result' will be a list of return values from all registered functions. result = pytestconfig.hook.pytest_my_hook(config=pytestconfig) .. note::