Skip to content

runner: correctly cleanup item _request/funcargs if an exception was reraised during call (e.g. KeyboardInterrupt)#13626

Merged
bluetech merged 1 commit into
pytest-dev:mainfrom
mon:insufficient_cleanup_on_keyboardinterrupt
Apr 12, 2026
Merged

runner: correctly cleanup item _request/funcargs if an exception was reraised during call (e.g. KeyboardInterrupt)#13626
bluetech merged 1 commit into
pytest-dev:mainfrom
mon:insufficient_cleanup_on_keyboardinterrupt

Conversation

@mon

@mon mon commented Jul 30, 2025

Copy link
Copy Markdown
Contributor

In my test suite, I have some objects that rely on garbage collection to be cleaned up correctly. This is not amazing, but it's how the code is structured for now. If I interrupt tests with Ctrl+C, or by manually raising KeyboardInterrupt in a test, these objects are not cleaned up any more.

call_and_report re-raises Exit and KeyboardInterrupt, which breaks the cleanup logic in runtestprotocol that unsets the item funcargs (which is where my objects end up living as references, as they're passed in as fixtures).

By just wrapping the entire block with try: ... finally: ..., cleanup works again as expected.

I'm going to hold off on updating changelog/AUTHORS until this PR is accepted!

Template checklist things:

  • [n/a] Include documentation when adding new features.
  • Include new tests or update existing tests when applicable (hard to test CPython reference/GC behaviour, suggestions appreciated).
  • Allow maintainers to push and squash when merging my commits. Please uncheck this if you prefer to squash the commits yourself.

If this change fixes an issue, please:

  • [n/a] Add text like closes #XYZW to the PR description and/or commits

Unless your change is trivial or a small documentation fix (e.g., a typo or reword of a small section) please:

  • Create a new changelog file in the changelog folder, with a name like <ISSUE NUMBER>.<TYPE>.rst. See changelog/README.rst for details.

  • Add yourself to AUTHORS in alphabetical order.

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

We also need test that demonstrates the problem and that it was fixed by this change

In general the suggestion is to never ever use gc for Ressource management

But test tools need to account for their users dealing with legacy codebases that have that misbehavior

@mon

mon commented Jul 30, 2025

Copy link
Copy Markdown
Contributor Author

Alright, I'll try and make a test case to validate the fix. Does it otherwise seem OK?

The real reason for the GC-based cleanup is because session scoped test fixtures are being too aggressively cleaned up by pytest, but that is a much harder problem for me to get my head around and presumably harder to actually solve. So here I am relying on Cpython specifics...

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

If you trade session scope with dunder del something is very wrong

Its a use case we explicitly condone

Its on the don't do and if you had to migrate away from it ASAP list

@mon

mon commented Jul 30, 2025

Copy link
Copy Markdown
Contributor Author

I'll revisit the reason that session scopes fixtures aren't working then, sounds like a significantly better long-term solution. I think it was something along the lines of #8102 (I am parametrizing these session fixtures).

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

Sounds like you want a retaining factory fixture

@mon

mon commented Aug 1, 2025

Copy link
Copy Markdown
Contributor Author

I appreciate you taking the time to help, this __del__ based cleanup has been bothering me since I convinced myself I needed it. I'm not entirely sure what a retaining factory fixture is, could you could give me an example to plug into my tests?

If context helps, here's a minimum repro of my issue. Looking at the --setup-plan output, because my session fixtures are a cartesian product and pytest seems to handle fixture cleanup on a per-test basis, I always get fixtures setup and finalised more than once. In this example it's just S1, but it scales with the number of tests.

conftest.py

import pytest

@pytest.fixture(scope="session")
def client(): pass
@pytest.fixture(scope="session")
def server(): pass

class C1: pass
class C2: pass
class S1: pass
class S2: pass

def pytest_generate_tests(metafunc: pytest.Metafunc):
    metafunc.parametrize("client", (C1, C2), scope="session", indirect=True)
    metafunc.parametrize("server", (S1, S2), scope="session", indirect=True)

test_test.py

def test_teardown_order1(client, server): pass
def test_teardown_order2(client, server): pass

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

In the posted case multiple setups/teardowns are expexted as only one parameter variant can be active at a time

A retaining fixture would be a non parameterized fixture one is3s as factory/ final cleanup

Setup would then as the factory for a instance

Ans teardowns would entirely be left to the factory

That way the parameterized fixtures would be reduced helpers that as the factory foe a instance

@mon

mon commented Aug 1, 2025

Copy link
Copy Markdown
Contributor Author

I think I understand, I'll try and get something working over the next few days. Will it roughly look like this?

@pytest.fixture(scope="session")
def factory():
    active = {}

    def fn(param: type):
        if existing := active.get(param):
            return existing
        
        active[param] = param()
        return active[param]

    yield fn

    for x in active.values():
        x.cleanup()

@pytest.fixture
def client(factory, request):
    yield factory(request.param)

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

@mon my understanding is that you actual/final goal would be more along the lines of having a parameterized global fixture of which more than one instance can be active at the same time and thus partially excluding it from the test reordering problems

@mon

mon commented Aug 5, 2025

Copy link
Copy Markdown
Contributor Author

I didn't quite understand "of which more than one instance can be active at the same time" since that was my problem to begin with, so I ended up going with my previous solution and it's worked great!

Is it worth closing this PR or is it a useful goal to clean things up regardless?

@RonnyPfannschmidt

Copy link
Copy Markdown
Member

Lifecycle management is hard currently we can't express the case of multiple can be active in the internal datastructures

This is a artifacts of how the fixture system organically evolved between 2010 and 2016

@bluetech bluetech force-pushed the insufficient_cleanup_on_keyboardinterrupt branch from b31abac to 30f06ec Compare April 12, 2026 13:15
@psf-chronographer psf-chronographer Bot added the bot:chronographer:provided (automation) changelog entry is part of PR label Apr 12, 2026

@bluetech bluetech left a comment

Copy link
Copy Markdown
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Regardless of the very true comments by @RonnyPfannschmidt (prefer to not rely on __del__), I do think this change makes sense, i.e. that the cleanup is supposed to happen also on Ctrl-C.

Writing a proper semantic regression test for this is hard because even Cpython doesn't guarantee "release on last reference" semantics. So I wrote a simple unit test that the _request is cleared instead. Also added a changelog entry.

…reraised during call (e.g. KeyboardInterrupt)

In my test suite, I have some objects that rely on garbage collection to
be cleaned up correctly. This is not amazing, but it's how the code is
structured for now. If I interrupt tests with Ctrl+C, or by manually
raising KeyboardInterrupt in a test, these objects are not cleaned up
any more.

call_and_report re-raises Exit and KeyboardInterrupt, which breaks the
cleanup logic in runtestprotocol that unsets the item funcargs (which is
where my objects end up living as references, as they're passed in as
fixtures).

By just wrapping the entire block with try: ... finally: ..., cleanup
works again as expected.

Co-authored-by: Ran Benita <ran@unusedvar.com>
@bluetech bluetech force-pushed the insufficient_cleanup_on_keyboardinterrupt branch from 30f06ec to 338851a Compare April 12, 2026 13:19
@bluetech bluetech merged commit 2d3d169 into pytest-dev:main Apr 12, 2026
33 checks passed
@mon

mon commented Apr 15, 2026

Copy link
Copy Markdown
Contributor Author

Fantastic to hear, thank you so much!

@mon mon deleted the insufficient_cleanup_on_keyboardinterrupt branch April 15, 2026 11:06
luketainton pushed a commit to luketainton/repos_PwnedPW that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ1bml0LXRlc3RzIl19-->

Reviewed-on: https://git.tainton.uk/repos/PwnedPW/pulls/340
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
luketainton pushed a commit to luketainton/repos_pypilot that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ0eXBlL2RlcGVuZGVuY2llcyJdfQ==-->

Reviewed-on: https://git.tainton.uk/repos/pypilot/pulls/454
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
luketainton pushed a commit to luketainton/repos_epage that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ0eXBlL2RlcGVuZGVuY2llcyJdfQ==-->

Reviewed-on: https://git.tainton.uk/repos/epage/pulls/225
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
luketainton pushed a commit to luketainton/luke_instant-msg-api that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ0eXBlL2RlcGVuZGVuY2llcyJdfQ==-->

Reviewed-on: https://git.tainton.uk/luke/instant-msg-api/pulls/263
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
luketainton pushed a commit to luketainton/repos_webexmemebot that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ0eXBlL2RlcGVuZGVuY2llcyJdfQ==-->

Reviewed-on: https://git.tainton.uk/repos/webexmemebot/pulls/590
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
luketainton pushed a commit to luketainton/repos_roboluke that referenced this pull request Jun 18, 2026
This PR contains the following updates:

| Package | Change | [Age](https://docs.renovatebot.com/merge-confidence/) | [Confidence](https://docs.renovatebot.com/merge-confidence/) |
|---|---|---|---|
| [pytest](https://github.com/pytest-dev/pytest) ([changelog](https://docs.pytest.org/en/stable/changelog.html)) | `<9.1.0,>=9.0.0` → `<9.1.1,>=9.1.0` | ![age](https://developer.mend.io/api/mc/badges/age/pypi/pytest/9.1.0?slim=true) | ![confidence](https://developer.mend.io/api/mc/badges/confidence/pypi/pytest/9.0.3/9.1.0?slim=true) |

---

### Release Notes

<details>
<summary>pytest-dev/pytest (pytest)</summary>

### [`v9.1.0`](https://github.com/pytest-dev/pytest/releases/tag/9.1.0)

[Compare Source](pytest-dev/pytest@9.0.3...9.1.0)

### pytest 9.1.0 (2026-06-13)

#### Removals and backward incompatible breaking changes

- [#&#8203;14533](pytest-dev/pytest#14533): When using `--doctest-modules`, autouse fixtures with `module`, `package` or `session` scope that are defined inline in Python test modules (not plugins or conftests) will now possibly execute twice.

  If this is undesirable, move the fixture definition to a `conftest.py` file if possible.

  Technical explanation for those interested:
  When using <span class="title-ref">--doctest-modules</span>, pytest possibly collects Python modules twice, once as `pytest.Module` and once as a `DoctestModule` (depending on the configuration).
  Due to improvements in pytest's fixture implementation, if e.g. the `DoctestModule` collects a fixture, it is now visible to it only, and not to the `Module`.
  This means that both need to register the fixtures independently.

#### Deprecations (removal in next major release)

- [#&#8203;10819](pytest-dev/pytest#10819): Added a deprecation warning for class-scoped fixtures defined as instance methods (without `@classmethod`). Such fixtures set attributes on a different instance than the test methods use, leading to unexpected behavior. Use `@classmethod` decorator instead -- by `yastcher`.

  See `10819` and `14011`.

- [#&#8203;12882](pytest-dev/pytest#12882): Calling `request.getfixturevalue() <pytest.FixtureRequest.getfixturevalue>` during teardown to request a fixture that was not already requested is now deprecated and will become an error in pytest 10.

  See `dynamic-fixture-request-during-teardown` for details.

- [#&#8203;13409](pytest-dev/pytest#13409): Using non-`~collections.abc.Collection` iterables (such as generators, iterators, or custom iterable objects) for the `argvalues` parameter in `@pytest.mark.parametrize <pytest.mark.parametrize ref>` and `metafunc.parametrize <pytest.Metafunc.parametrize>` is now deprecated.

  These iterables get exhausted after the first iteration,
  leading to tests getting unexpectedly skipped in cases such as running `pytest.main()` multiple times,
  using class-level parametrize decorators,
  or collecting tests multiple times.

  See `parametrize-iterators` for details and suggestions.

- [#&#8203;13946](pytest-dev/pytest#13946): The private `config.inicfg` attribute is now deprecated.
  Use `config.getini() <pytest.Config.getini>` to access configuration values instead.

  See `config-inicfg` for more details.

- [#&#8203;14004](pytest-dev/pytest#14004): Passing `baseid` to `~pytest.FixtureDef` or `nodeid` strings to fixture registration APIs is now deprecated. These are internal pytest APIs that are used by some plugins.

  Use the `node` parameter instead for fixture scoping. This enables more robust node-based
  matching instead of string prefix matching.
  If you've used `nodeid=None`, pass `node=session` instead.

  This will be removed in pytest 10.

- [#&#8203;14335](pytest-dev/pytest#14335): The method of configuring hooks using markers, deprecated since pytest 7.2, is now scheduled to be removed in pytest 10.
  See `hook-markers` for more details.

- [#&#8203;14434](pytest-dev/pytest#14434): The `--pastebin` option is now deprecated.
  The same functionality is now available in an external plugin, `pytest-pastebin`.
  See `pastebin-deprecated` for more details.

- [#&#8203;14513](pytest-dev/pytest#14513): The private `FixtureDef.has_location` attribute is now deprecated and will be removed in pytest 10.
  See `fixturedef-has-location-deprecated` for details.

- [#&#8203;1764](pytest-dev/pytest#1764): `pytest.console_main` is now deprecated and will be removed in pytest 10.
  It was never intended for programmatic use; use `pytest.main` instead.

#### New features

- [#&#8203;12376](pytest-dev/pytest#12376): Added `pytest.register_fixture()` to register fixtures using an imperative interface.

  This is an advanced function intended for use by plugins.

  Normally, fixtures should be registered declaratively using the `@pytest.fixture <pytest.fixture>` decorator.
  Pytest looks for these fixture definitions during the collection phase and registers them automatically.
  For some plugin usecases the declarative interface can be cumbersome or unviable, in which case this imperative interface can be used.

- [#&#8203;14023](pytest-dev/pytest#14023): Added <span class="title-ref">--report-chars</span> long CLI option.

- [#&#8203;14371](pytest-dev/pytest#14371): Added `--max-warnings` command-line option and `max_warnings` configuration option to fail the test run when the number of warnings exceeds a given threshold -- by `miketheman`.

- [#&#8203;6757](pytest-dev/pytest#6757): Added the `assertion_text_diff_style` configuration option, allowing
  string equality failures to be rendered as separate `Left:` and `Right:`
  blocks instead of `ndiff` output.

- [#&#8203;8395](pytest-dev/pytest#8395): Added support for `~datetime.datetime` and `~datetime.timedelta` comparisons with `pytest.approx`. An explicit `abs` or `rel` tolerance as a `~datetime.timedelta` is required and relative tolerance is not supported for datetime comparisons -- by `hamza-mobeen`.

#### Improvements in existing functionality

- [#&#8203;11225](pytest-dev/pytest#11225): `pytest.warns` now shows "Regex pattern did not match" instead of "DID NOT WARN" when warnings were emitted but the `match` pattern did not match.

- [#&#8203;11295](pytest-dev/pytest#11295): Improved output of `--fixtures-per-test` by excluding internal-implementation fixtures generated by `@pytest.mark.parametrize` and similar.

- [#&#8203;13241](pytest-dev/pytest#13241): `pytest.raises`, `pytest.warns` and `pytest.deprecated_call` now uses `ParamSpec` for the type hint to the (old and not recommended) callable overload, instead of `Any`. This allows type checkers to raise errors when passing incorrect function parameters.
  `func` can now also be passed as a kwarg, which the type hint previously showed as possible but didn't accept.

- [#&#8203;13862](pytest-dev/pytest#13862): Improved the readability of "DID NOT RAISE" error messages by using the exception type's name instead of its <span class="title-ref">repr</span>.

- [#&#8203;14026](pytest-dev/pytest#14026): Added test coverage for compiled regex patterns in `pytest.raises` match parameter.

- [#&#8203;14137](pytest-dev/pytest#14137): <span class="title-ref">pytest.ScopeName</span> is now public to allow using it in function signatures.

- [#&#8203;14342](pytest-dev/pytest#14342): Marked `yield_fixture` as deprecated to type checkers using the `deprecated` decorator. Note it
  `has originally been deprecated <yield-fixture-deprecated>` in pytest 6.2 already.

- [#&#8203;14373](pytest-dev/pytest#14373): Added type annotations for `pytest.approx`.

- [#&#8203;14430](pytest-dev/pytest#14430): When using `--setup-show`, a space is now printed after the test name (and possibly used fixtures), to separate it from the test result.

- [#&#8203;14441](pytest-dev/pytest#14441): Reduced the default number of `gc.collect()` passes in the `unraisableexception` plugin from 5 to 1 on CPython, where reference counting makes a single pass sufficient. PyPy retains 5 passes due to object resurrection via `__del__`. This can noticeably speed up test suites that trigger many pytester runs.

- [#&#8203;14461](pytest-dev/pytest#14461): Improved assertion failure explanations for equality comparisons between mapping objects that are not `dict` instances.

- [#&#8203;14513](pytest-dev/pytest#14513): The order in which fixture definitions overriding each other are resolved is now determined first by their *visibility* in the collection tree rather than by the order in which they are registered.

  A fixture defined for a more specific node (e.g. a module or an item) now always takes precedence over one with the same name defined for a more general node (e.g. the session), even when the more general one was registered later.
  Fixtures with non-comparable visibility or the same visibility keep the existing behavior of "last registered wins".
  This change is supposed to only affect plugins which register multiple fixtures programmatically with the same name.

- [#&#8203;14524](pytest-dev/pytest#14524): Add official Python 3.15 support.

- [#&#8203;1764](pytest-dev/pytest#1764): Improved argparse program name to show `pytest`, `python -m pytest`, or `pytest.main()` based on how pytest was invoked, making help and error messages clearer.

- [#&#8203;8265](pytest-dev/pytest#8265): Emit a `PytestCollectionWarning` when a module-level `__getattr__` returns `None` for `pytestmark` instead of raising `AttributeError`.

  Previously this caused a cryptic `TypeError: got None instead of Mark` error.
  Now pytest issues a helpful warning and continues collecting the module normally.

#### Bug fixes

- [#&#8203;13192](pytest-dev/pytest#13192): Fixed <span class="title-ref">|</span> (pipe) not being treated as a regex meta-character that needs escaping in `pytest.raises(match=...) <pytest.raises>`.

- [#&#8203;13484](pytest-dev/pytest#13484): Fixed `-W` option values being duplicated in `Config.known_args_namespace`.

- [#&#8203;13626](pytest-dev/pytest#13626): Fixed function-scoped fixture values being kept alive after a test was interrupted by `KeyboardInterrupt` or early exit,
  allowing them to potentially be released more promptly.

- [#&#8203;13784](pytest-dev/pytest#13784): Fixed `capteesys` producing doubled output when used with `--capture=no` (`-s`).

- [#&#8203;13817](pytest-dev/pytest#13817): Fixed a secondary <span class="title-ref">AttributeError</span> masking the original error when an option argument fails to initialize.

- [#&#8203;13884](pytest-dev/pytest#13884): Fixed rare internal IndexError caused by <span class="title-ref">builtins.compile</span> being overridden in client code.

- [#&#8203;13885](pytest-dev/pytest#13885): Fixed autouse fixtures defined inside a `unittest.TestCase` class running even when the class is decorated with `unittest.skip` or `unittest.skipIf` -- regression since pytest 8.1.0.

- [#&#8203;13917](pytest-dev/pytest#13917): `unittest.SkipTest` is no longer considered an interactive exception, i.e. `pytest_exception_interact` is no longer called for it.

- [#&#8203;13963](pytest-dev/pytest#13963): Fixed subtests running with `pytest-xdist` when their contexts contain objects that are not JSON-serializable.

  Fixes [pytest-dev/pytest-xdist#1273](pytest-dev/pytest-xdist#1273).

- [#&#8203;14004](pytest-dev/pytest#14004): Fixed conftest.py fixture scoping when `testpaths` points outside of the `rootdir <rootdir>`.

  Previously, fixtures from nested conftest.py files would incorrectly leak to sibling directories
  when using a relative `testpaths` like `../tests/sdk`.

  Conftest fixtures are now parsed during `Directory <pytest.Directory>` collection, using the `Directory` node for proper scoping.

- [#&#8203;14050](pytest-dev/pytest#14050): Display dictionary differences in assertion failures using the original key insertion order instead of sorted order.

- [#&#8203;14080](pytest-dev/pytest#14080): fix missing type annotations on `Pytester.makepyfile` and `Pytester.maketxtfile` methods.

- [#&#8203;14114](pytest-dev/pytest#14114): An exception from `pytest_fixture_post_finalizer` no longer prevents fixtures from being torn down, causing additional errors in the following tests.

- [#&#8203;14161](pytest-dev/pytest#14161): Fixed `monkeypatch.setattr() <pytest.MonkeyPatch.setattr>` leaving a stale entry on the undo stack when the underlying `setattr()` call fails (e.g. on immutable targets), causing an `AttributeError` crash during teardown.

- [#&#8203;14214](pytest-dev/pytest#14214): Fixed `-v` hint in `pytest.raises` match diff not working because assertion verbosity was not propagated.

- [#&#8203;14234](pytest-dev/pytest#14234): Allow `pytest.HIDDEN_PARAM <hidden-param>` in `@pytest.mark.parametrize(ids=...) <pytest.mark.parametrize ref>` typing.

- [#&#8203;14248](pytest-dev/pytest#14248): Fixed direct parametrization causing the static fixture closure (as reflected in `request.fixturenames <pytest.FixtureRequest.fixturenames>`) to omit fixtures that are requested transitively from overridden fixtures.

- [#&#8203;14263](pytest-dev/pytest#14263): Unraisable exceptions from finalizers are now collected during `pytest_unconfigure`, before pytest tears down the warning filters installed for the session. Previously the collection ran from a cleanup callback whose order relative to other plugins' cleanups was not guaranteed, so an active `error` filter could be removed before the exception surfaced and a late resource leak would pass silently. A `-W error` filter, or any filter matching `pytest.PytestUnraisableExceptionWarning`, now promotes these exceptions to failures regardless of plugin cleanup order.

- [#&#8203;14377](pytest-dev/pytest#14377): Fixed crash in <span class="title-ref">Config.get\_terminal\_writer</span> when an assertion fails with the `terminalreporter` plugin disabled.

- [#&#8203;14381](pytest-dev/pytest#14381): Fixed `-V` (short form of `--version`) to properly display the current version.

- [#&#8203;14389](pytest-dev/pytest#14389): Improved `pytest.raises(..., match=...) <pytest.raises>` failures to suppress the mismatched exception as a cause of the resulting `AssertionError`.

- [#&#8203;14392](pytest-dev/pytest#14392): Fixed a bug in `pytest.raises(match=...) <pytest.raises>` "fully escaped" detection, causing the regex diff display to be shown in some instances when the raw string diff display should be shown instead.

- [#&#8203;14442](pytest-dev/pytest#14442): Fixed a regression in pytest 9.0 where `--strict-markers` and `--strict-config` specified through `addopts` were silently ignored.

  Note that when targeting pytest >= 9.0, it's nicer to use `strict_markers` and `strict_config`, or `strict mode <strict mode>`.

- [#&#8203;14456](pytest-dev/pytest#14456): Fixed `pytest.approx` not recognizing types with `__array_interface__` as numpy-like arrays.

- [#&#8203;14474](pytest-dev/pytest#14474): Fixed a regression where `-k` and `-m` expressions containing both backslash characters in identifiers and string literal arguments would incorrectly raise a `SyntaxError` about escaping.

- [#&#8203;14483](pytest-dev/pytest#14483): Fixed JUnit XML report incorrectly escaping high Unicode codepoints (supplementary plane characters like emoji) in test failure messages. -- by `EternalRights`

- [#&#8203;14492](pytest-dev/pytest#14492): Fixed `Code.getargs()` incorrectly including local variable names in the returned argument tuple for functions with `*args` and/or `**kwargs`. The method was using `co_flags` bitmask values (`4` and `8`) directly as counts instead of converting them to `1` via `bool()`, and was not accounting for `co_kwonlyargcount` when `var=True`.

- [#&#8203;3697](pytest-dev/pytest#3697): Logging capture now works for non-propagating loggers.
  Previously only logs which reached the root logger were captured.
  This includes `caplog` and the "Captured log calls" test reporting.

- [#&#8203;3850](pytest-dev/pytest#3850): Fixed JUnit XML report: the `tests` attribute of the `<testsuite>` element now always matches the number of `<testcase>` elements in the file. In some cases (test passes but fails during teardown) the `tests` attribute would report an incorrect number of testcases in the XML file.

- [#&#8203;5848](pytest-dev/pytest#5848): `pytest_fixture_post_finalizer` is no longer called extra times for the same fixture teardown in some cases.

- [#&#8203;719](pytest-dev/pytest#719): Fixed `@pytest.mark.parametrize <pytest.mark.parametrize ref>` not unpacking single-element tuple values when using a string argnames with a trailing comma (e.g., `"arg,"`).

  The trailing comma form now correctly behaves like the tuple form `("arg",)`, treating argvalues as a list of tuples to unpack.

#### Improved documentation

- [#&#8203;11022](pytest-dev/pytest#11022): Document safer alternatives and scope guidance for monkeypatching standard library functions.
- [#&#8203;11307](pytest-dev/pytest#11307): Document that `@pytest.hookimpl(specname=...)` only works for function names starting with `pytest_`.
- [#&#8203;13038](pytest-dev/pytest#13038): Document that doctests do not support parametrized fixtures, including parametrized autouse fixtures.
- [#&#8203;13155](pytest-dev/pytest#13155): Clarified how the `request` fixture provides indirect parametrization values via `request.param`.
- [#&#8203;13304](pytest-dev/pytest#13304): Clarified in the documentation that hook implementations defined in `conftest.py` files are not available to other plugins during their `pytest_addoption()` execution, as conftest files are discovered and loaded after builtin and third-party plugins have been initialized. However, initial conftest files themselves can implement `pytest_addoption()` to add their own command-line options.
- [#&#8203;13902](pytest-dev/pytest#13902): Clarified how subtest progress markers are shown in the documentation.
- [#&#8203;14012](pytest-dev/pytest#14012): The `ini options ref` section of the API Reference now specified the type and default value of every configuration option.
- [#&#8203;14148](pytest-dev/pytest#14148): Documented a safe `pytestconfig.cache` access pattern when the
  `cacheprovider` plugin is disabled.
- [#&#8203;14303](pytest-dev/pytest#14303): The documentation is now built with Sphinx >= 9.
- [#&#8203;14465](pytest-dev/pytest#14465): Updated the hooks how-to page to link the `newhooks.py` file in `pytest-xdist` at tag `v3.8.0` instead of an unrelated 2017-era commit under the old layout. Pointing at a tag keeps the example in sync with the version users actually install, while remaining stable when the project's main branch moves on.

#### Miscellaneous internal changes

- [#&#8203;14582](pytest-dev/pytest#14582): Improved the recursion traceback test to exercise all requested traceback styles.

</details>

---

### Configuration

📅 **Schedule**: (UTC)

- Branch creation
  - At any time (no schedule defined)
- Automerge
  - At any time (no schedule defined)

🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied.

♻ **Rebasing**: Whenever PR is behind base branch, or you tick the rebase/retry checkbox.

🔕 **Ignore**: Close this PR and you won't be reminded about this update again.

---

 - [ ] <!-- rebase-check -->If you want to rebase/retry this PR, check this box

---

This PR has been generated by [Mend Renovate](https://github.com/renovatebot/renovate).
<!--renovate-debug:eyJjcmVhdGVkSW5WZXIiOiI0My4yMjAuMCIsInVwZGF0ZWRJblZlciI6IjQzLjIyMC4wIiwidGFyZ2V0QnJhbmNoIjoibWFpbiIsImxhYmVscyI6WyJ0eXBlL2RlcGVuZGVuY2llcyJdfQ==-->

Reviewed-on: https://git.tainton.uk/repos/roboluke/pulls/456
Co-authored-by: renovate[bot] <renovate-bot@git.tainton.uk>
Co-committed-by: renovate[bot] <renovate-bot@git.tainton.uk>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

bot:chronographer:provided (automation) changelog entry is part of PR

Projects

None yet

Development

Successfully merging this pull request may close these issues.

3 participants