Skip to main content

pytest plugin for running parallel tests

Project description

pytest-freethreaded

A Pytest plugin for helping verify that your tests and libraries are thread-safe with the Python 3.13 experimental freethreaded mode.

Why do I need this plugin?

Python 3.13 introduces a new mode called freethreaded that allows you to run Python code with the GIL disabled. This can be useful for running CPU-bound code in parallel, but it also means that your code must be thread-safe.

Some packages and libraries with good, mature test bases have marked their libraries as "compatible" with freethreaded, but they have only run the tests once and in a single-thread!

Pytest-xdist won't help you here, because it uses multiprocessing to parallelize tests, and the GIL is still enabled.

So we made this plugin to help you run your tests in a thread-pool with the GIL disabled, to help you identify if your tests are thread-safe.

And the first library we tested it on (which was marked as compatible) caused a segmentation fault in CPython! So you should give this a go if you're a package maintainer.

Installation

pip install pytest-freethreaded

Usage

We recommend using this plugin in a phased approach to identify if your tests can be run in a thread-pool with the GIL disabled, before disabling the GIL.

Phase 1: Your tests should pass without this plugin

First, your tests should be running, and you should have tests.

Phase 2: Run your tests with the GIL enabled, but with concurrency

This plugin will run each multiple times, in parallel. By default, it will execute each test 10 times, then repeat that 20 times. This means each test will run 200 times. This is a lot, so we recommend only running this cycle once, or on a subset of tests.

If a test fails on any of the runs, we will raise a special error called a ConcurrencyError that indicates the test partially failed, most likely due to a lack of thread-safety and/or a race condition.

Selecting specific tests

If, during this phase you decide to only run a subset of tests, you can use the -k flag to select tests by name.

Alternatively, you can set the number of threads and the executions using a marker:

import pytest

@pytest.mark.freethreaded(threads=10, executions=10)
def test_my_threaded_code():
    pass

Then on the command-line set the threads and executions to 1:

pytest --threads 1 --executions 1

Phase 3: Run your tests with the GIL disabled

If you passed phase 3; congratulations! You can now run your tests with the GIL disabled.

We recommend using the --require-gil-disabled flag in Pytest for this phase so that if you accidentally run Python 3.13 with the GIL on, it will fail the tests.

pytest --require-gil-disabled

During this final phase, if you see any new Concurrency Errors or even worse you see a crash in Python 3.13, you will need to debug it.

Development

Setup:

python3.13t -m venv .venv
source .venv/bin/activate
python -m pip install flit
python -m flit install -s

Testing

To run the tests

pytest -vvvv --log-level=DEBUG

Credits

This extension was created at PyCon JP sprints with the following team members:

  • Anthony Shaw @tonybaloney
  • Mike Hommey @glandium
  • Oliver Basset @obassett
  • Maryanne Wachter @m-clare
  • Toshihiko Yanase @toshihikoyanase
  • Heejun Shin @abel9851
  • Otake Katsuaki
  • Rei Suyama @rhoboro

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pytest_freethreaded-0.1.0.tar.gz (6.4 kB view details)

Uploaded Source

Built Distribution

pytest_freethreaded-0.1.0-py3-none-any.whl (5.3 kB view details)

Uploaded Python 3

File details

Details for the file pytest_freethreaded-0.1.0.tar.gz.

File metadata

  • Download URL: pytest_freethreaded-0.1.0.tar.gz
  • Upload date:
  • Size: 6.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-requests/2.32.3

File hashes

Hashes for pytest_freethreaded-0.1.0.tar.gz
Algorithm Hash digest
SHA256 9cb4f412db3abb5a8ffdcb29a2d42e058c41f16437c0a1815101651b485dcf45
MD5 2b7729b05ba617b6ae703cc12dd65314
BLAKE2b-256 20093bf5fb5321463551787c41bd1af128c1eff018a55e1bb91af0501df54773

See more details on using hashes here.

File details

Details for the file pytest_freethreaded-0.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for pytest_freethreaded-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 5f2beb6026857f274162d0aa131fe605b8c6d2a3b0c2d35a2af14721959398f1
MD5 128ba4243a332812b9d6f459a8e05ab3
BLAKE2b-256 a6e0f379af92804357ac186d2d90ec73a1534183477de1e32e1a4f0bb86b85da

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page