doc: Reformat/Modernize some code (#9900)

Found because I was curious what https://pypi.org/project/shed/ does with pytest.
This commit is contained in:
Florian Bruhin 2022-04-28 16:30:16 +02:00 committed by GitHub
parent eb8b3ad929
commit e580534df0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 45 additions and 36 deletions

View File

@ -25,7 +25,7 @@ example: specifying and selecting acceptance tests
self.tmpdir = request.config.mktemp(request.function.__name__, numbered=True) self.tmpdir = request.config.mktemp(request.function.__name__, numbered=True)
def run(self, *cmd): def run(self, *cmd):
""" called by test code to execute an acceptance test. """ """called by test code to execute an acceptance test."""
self.tmpdir.chdir() self.tmpdir.chdir()
return subprocess.check_output(cmd).decode() return subprocess.check_output(cmd).decode()

View File

@ -375,7 +375,7 @@ specifies via named environments:
envnames = [mark.args[0] for mark in item.iter_markers(name="env")] envnames = [mark.args[0] for mark in item.iter_markers(name="env")]
if envnames: if envnames:
if item.config.getoption("-E") not in envnames: if item.config.getoption("-E") not in envnames:
pytest.skip("test requires env in {!r}".format(envnames)) pytest.skip(f"test requires env in {envnames!r}")
A test file using this local plugin: A test file using this local plugin:
@ -528,7 +528,7 @@ test function. From a conftest file we can read it like this:
def pytest_runtest_setup(item): def pytest_runtest_setup(item):
for mark in item.iter_markers(name="glob"): for mark in item.iter_markers(name="glob"):
print("glob args={} kwargs={}".format(mark.args, mark.kwargs)) print(f"glob args={mark.args} kwargs={mark.kwargs}")
sys.stdout.flush() sys.stdout.flush()
Let's run this without capturing output and see what we get: Let's run this without capturing output and see what we get:
@ -558,6 +558,7 @@ for your particular platform, you could use the following plugin:
# content of conftest.py # content of conftest.py
# #
import sys import sys
import pytest import pytest
ALL = set("darwin linux win32".split()) ALL = set("darwin linux win32".split())
@ -567,7 +568,7 @@ for your particular platform, you could use the following plugin:
supported_platforms = ALL.intersection(mark.name for mark in item.iter_markers()) supported_platforms = ALL.intersection(mark.name for mark in item.iter_markers())
plat = sys.platform plat = sys.platform
if supported_platforms and plat not in supported_platforms: if supported_platforms and plat not in supported_platforms:
pytest.skip("cannot run on platform {}".format(plat)) pytest.skip(f"cannot run on platform {plat}")
then tests will be skipped if they were specified for a different platform. then tests will be skipped if they were specified for a different platform.
Let's do a little test file to show how this looks like: Let's do a little test file to show how this looks like:

View File

@ -663,6 +663,7 @@ as a complement to ``raises``. For example:
.. code-block:: python .. code-block:: python
from contextlib import contextmanager from contextlib import contextmanager
import pytest import pytest

View File

@ -342,7 +342,7 @@ Example:
def checkconfig(x): def checkconfig(x):
__tracebackhide__ = True __tracebackhide__ = True
if not hasattr(x, "config"): if not hasattr(x, "config"):
pytest.fail("not configured: {}".format(x)) pytest.fail(f"not configured: {x}")
def test_something(): def test_something():
@ -376,6 +376,7 @@ this to make sure unexpected exception types aren't hidden:
.. code-block:: python .. code-block:: python
import operator import operator
import pytest import pytest
@ -386,7 +387,7 @@ this to make sure unexpected exception types aren't hidden:
def checkconfig(x): def checkconfig(x):
__tracebackhide__ = operator.methodcaller("errisinstance", ConfigException) __tracebackhide__ = operator.methodcaller("errisinstance", ConfigException)
if not hasattr(x, "config"): if not hasattr(x, "config"):
raise ConfigException("not configured: {}".format(x)) raise ConfigException(f"not configured: {x}")
def test_something(): def test_something():
@ -565,6 +566,7 @@ an ``incremental`` marker which is to be used on classes:
# content of conftest.py # content of conftest.py
from typing import Dict, Tuple from typing import Dict, Tuple
import pytest import pytest
# store history of failures per test class name and per index in parametrize (if parametrize used) # store history of failures per test class name and per index in parametrize (if parametrize used)
@ -608,7 +610,7 @@ an ``incremental`` marker which is to be used on classes:
test_name = _test_failed_incremental[cls_name].get(parametrize_index, None) 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 name found, test has failed for the combination of class name & test name
if test_name is not None: if test_name is not None:
pytest.xfail("previous test failed ({})".format(test_name)) pytest.xfail(f"previous test failed ({test_name})")
These two hook implementations work together to abort incremental-marked These two hook implementations work together to abort incremental-marked
@ -802,9 +804,10 @@ case we just write some information out to a ``failures`` file:
# content of conftest.py # content of conftest.py
import pytest
import os.path import os.path
import pytest
@pytest.hookimpl(tryfirst=True, hookwrapper=True) @pytest.hookimpl(tryfirst=True, hookwrapper=True)
def pytest_runtest_makereport(item, call): def pytest_runtest_makereport(item, call):
@ -1066,6 +1069,7 @@ like ``pytest-timeout`` they must be imported explicitly and passed on to pytest
# contents of app_main.py # contents of app_main.py
import sys import sys
import pytest_timeout # Third party plugin import pytest_timeout # Third party plugin
if len(sys.argv) > 1 and sys.argv[1] == "--pytest": if len(sys.argv) > 1 and sys.argv[1] == "--pytest":

View File

@ -238,7 +238,7 @@ file which provides an alternative explanation for ``Foo`` objects:
if isinstance(left, Foo) and isinstance(right, Foo) and op == "==": if isinstance(left, Foo) and isinstance(right, Foo) and op == "==":
return [ return [
"Comparing Foo instances:", "Comparing Foo instances:",
" vals: {} != {}".format(left.val, right.val), f" vals: {left.val} != {right.val}",
] ]
now, given this test module: now, given this test module:

View File

@ -199,7 +199,6 @@ across pytest invocations:
# content of test_caching.py # content of test_caching.py
import pytest import pytest
import time
def expensive_computation(): def expensive_computation():

View File

@ -253,6 +253,7 @@ which works in a similar manner to :ref:`raises <assertraises>`:
.. code-block:: python .. code-block:: python
import warnings import warnings
import pytest import pytest

View File

@ -239,7 +239,6 @@ which can then be used in your doctests directly:
>>> len(a) >>> len(a)
10 10
""" """
pass
Note that like the normal ``conftest.py``, the fixtures are discovered in the directory tree conftest is in. Note that like the normal ``conftest.py``, the fixtures are discovered in the directory tree conftest is in.
Meaning that if you put your doctest with your source code, the relevant conftest.py needs to be in the same directory tree. Meaning that if you put your doctest with your source code, the relevant conftest.py needs to be in the same directory tree.

View File

@ -398,9 +398,10 @@ access the fixture function:
.. code-block:: python .. code-block:: python
# content of conftest.py # content of conftest.py
import pytest
import smtplib import smtplib
import pytest
@pytest.fixture(scope="module") @pytest.fixture(scope="module")
def smtp_connection(): def smtp_connection():
@ -609,10 +610,10 @@ Here's what that might look like:
.. code-block:: python .. code-block:: python
# content of test_emaillib.py # content of test_emaillib.py
import pytest
from emaillib import Email, MailAdminClient from emaillib import Email, MailAdminClient
import pytest
@pytest.fixture @pytest.fixture
def mail_admin(): def mail_admin():
@ -683,10 +684,10 @@ Here's how the previous example would look using the ``addfinalizer`` method:
.. code-block:: python .. code-block:: python
# content of test_emaillib.py # content of test_emaillib.py
import pytest
from emaillib import Email, MailAdminClient from emaillib import Email, MailAdminClient
import pytest
@pytest.fixture @pytest.fixture
def mail_admin(): def mail_admin():
@ -752,10 +753,10 @@ above):
.. code-block:: python .. code-block:: python
# content of test_emaillib.py # content of test_emaillib.py
import pytest
from emaillib import Email, MailAdminClient from emaillib import Email, MailAdminClient
import pytest
@pytest.fixture @pytest.fixture
def setup(): def setup():
@ -1030,16 +1031,17 @@ read an optional server URL from the test module which uses our fixture:
.. code-block:: python .. code-block:: python
# content of conftest.py # content of conftest.py
import pytest
import smtplib import smtplib
import pytest
@pytest.fixture(scope="module") @pytest.fixture(scope="module")
def smtp_connection(request): def smtp_connection(request):
server = getattr(request.module, "smtpserver", "smtp.gmail.com") server = getattr(request.module, "smtpserver", "smtp.gmail.com")
smtp_connection = smtplib.SMTP(server, 587, timeout=5) smtp_connection = smtplib.SMTP(server, 587, timeout=5)
yield smtp_connection yield smtp_connection
print("finalizing {} ({})".format(smtp_connection, server)) print(f"finalizing {smtp_connection} ({server})")
smtp_connection.close() smtp_connection.close()
We use the ``request.module`` attribute to optionally obtain an We use the ``request.module`` attribute to optionally obtain an
@ -1193,15 +1195,16 @@ through the special :py:class:`request <FixtureRequest>` object:
.. code-block:: python .. code-block:: python
# content of conftest.py # content of conftest.py
import pytest
import smtplib import smtplib
import pytest
@pytest.fixture(scope="module", params=["smtp.gmail.com", "mail.python.org"]) @pytest.fixture(scope="module", params=["smtp.gmail.com", "mail.python.org"])
def smtp_connection(request): def smtp_connection(request):
smtp_connection = smtplib.SMTP(request.param, 587, timeout=5) smtp_connection = smtplib.SMTP(request.param, 587, timeout=5)
yield smtp_connection yield smtp_connection
print("finalizing {}".format(smtp_connection)) print(f"finalizing {smtp_connection}")
smtp_connection.close() smtp_connection.close()
The main change is the declaration of ``params`` with The main change is the declaration of ``params`` with
@ -1503,7 +1506,7 @@ to show the setup/teardown flow:
def test_2(otherarg, modarg): def test_2(otherarg, modarg):
print(" RUN test2 with otherarg {} and modarg {}".format(otherarg, modarg)) print(f" RUN test2 with otherarg {otherarg} and modarg {modarg}")
Let's run the tests in verbose mode and with looking at the print-output: Let's run the tests in verbose mode and with looking at the print-output:
@ -1604,6 +1607,7 @@ and declare its use in a test module via a ``usefixtures`` marker:
# content of test_setenv.py # content of test_setenv.py
import os import os
import pytest import pytest

View File

@ -73,7 +73,6 @@ messages. This is supported by the ``caplog`` fixture:
def test_foo(caplog): def test_foo(caplog):
caplog.set_level(logging.INFO) caplog.set_level(logging.INFO)
pass
By default the level is set on the root logger, By default the level is set on the root logger,
however as a convenience it is also possible to set the log level of any however as a convenience it is also possible to set the log level of any
@ -83,7 +82,6 @@ logger:
def test_foo(caplog): def test_foo(caplog):
caplog.set_level(logging.CRITICAL, logger="root.baz") caplog.set_level(logging.CRITICAL, logger="root.baz")
pass
The log levels set are restored automatically at the end of the test. The log levels set are restored automatically at the end of the test.
@ -161,9 +159,7 @@ the records for the ``setup`` and ``call`` stages during teardown like so:
x.message for x in caplog.get_records(when) if x.levelno == logging.WARNING x.message for x in caplog.get_records(when) if x.levelno == logging.WARNING
] ]
if messages: if messages:
pytest.fail( pytest.fail(f"warning messages encountered during testing: {messages}")
"warning messages encountered during testing: {}".format(messages)
)

View File

@ -69,6 +69,7 @@ It is also possible to skip the whole module using
.. code-block:: python .. code-block:: python
import sys import sys
import pytest import pytest
if not sys.platform.startswith("win"): if not sys.platform.startswith("win"):
@ -409,6 +410,7 @@ test instances when using parametrize:
.. code-block:: python .. code-block:: python
import sys import sys
import pytest import pytest

View File

@ -115,6 +115,7 @@ fixture definition:
# content of test_unittest_db.py # content of test_unittest_db.py
import unittest import unittest
import pytest import pytest
@ -194,10 +195,10 @@ creation of a per-test temporary directory:
.. code-block:: python .. code-block:: python
# content of test_unittest_cleandir.py # content of test_unittest_cleandir.py
import os
import pytest
import unittest import unittest
import pytest
class MyTest(unittest.TestCase): class MyTest(unittest.TestCase):
@pytest.fixture(autouse=True) @pytest.fixture(autouse=True)

View File

@ -183,9 +183,10 @@ You can specify additional plugins to ``pytest.main``:
.. code-block:: python .. code-block:: python
# content of myinvoke.py # content of myinvoke.py
import pytest
import sys import sys
import pytest
class MyPlugin: class MyPlugin:
def pytest_sessionfinish(self): def pytest_sessionfinish(self):

View File

@ -194,7 +194,7 @@ class or module can then be passed to the ``pluginmanager`` using the ``pytest_a
.. code-block:: python .. code-block:: python
def pytest_addhooks(pluginmanager): def pytest_addhooks(pluginmanager):
""" This example assumes the hooks are grouped in the 'sample_hook' module. """ """This example assumes the hooks are grouped in the 'sample_hook' module."""
from my_app.tests import sample_hook from my_app.tests import sample_hook
pluginmanager.add_hookspecs(sample_hook) pluginmanager.add_hookspecs(sample_hook)
@ -253,14 +253,14 @@ and use pytest_addoption as follows:
# default value # default value
@hookspec(firstresult=True) @hookspec(firstresult=True)
def pytest_config_file_default_value(): def pytest_config_file_default_value():
""" Return the default value for the config file command line option. """ """Return the default value for the config file command line option."""
# contents of myplugin.py # contents of myplugin.py
def pytest_addhooks(pluginmanager): def pytest_addhooks(pluginmanager):
""" This example assumes the hooks are grouped in the 'hooks' module. """ """This example assumes the hooks are grouped in the 'hooks' module."""
from . import hooks from . import hooks
pluginmanager.add_hookspecs(hooks) pluginmanager.add_hookspecs(hooks)

View File

@ -367,7 +367,7 @@ string value of ``Hello World!`` if we do not supply a value or ``Hello
def _hello(name=None): def _hello(name=None):
if not name: if not name:
name = request.config.getoption("name") name = request.config.getoption("name")
return "Hello {name}!".format(name=name) return f"Hello {name}!"
return _hello return _hello

View File

@ -32,7 +32,7 @@ which will usually be called once for all the functions:
.. code-block:: python .. code-block:: python
def setup_module(module): def setup_module(module):
""" setup any state specific to the execution of the given module.""" """setup any state specific to the execution of the given module."""
def teardown_module(module): def teardown_module(module):