Run unittest test suites concurrently
Project description
concurrencytest
Python - Run unittest test suites concurrently
- Development: GitHub
- Download/Install: PyPI
- License: GPLv2+
- Copyright (c) 2013-2026 Corey Goldberg
| Type | Status |
|---|---|
| Latest Version | |
| Supported Python Versions | |
| Build/Tests (CI) |
About
concurrencytest allows parallel execution of unittest tests across multiple
worker processes.
- Default: 1 process per CPU core using round-robin test distribution.
- Optional: specify number of processes and partition strategy.
Components
ConcurrentTestSuiteclass: unittest-compatibleTestSuitefor running parallel tests.fork_for_testsfunction: fork-basedmake_testsimplementation.partition_testsfunction: round-robin test distribution.partition_tests_by_class: class-local test distribution.
Installation
Install from PyPI:
pip install concurrencytest
Requirements
- Python 3.10+
- Unix-like OS with
os.fork() - Dependencies:
Usage
Basic steps:
- write your tests in normal
unitteststyle (test methods inside aunittest.TestCaseclass) - load a suite of tests using
unittest.TestLoader(orunittest.defaultTestLoader):
suite = unittest.TestLoader().discover("tests")suite = unittest.TestLoader().loadTestsFromModule(my_tests)suite = unittest.TestLoader().loadTestsFromTestCase(MyTests)suite = unittest.TestLoader().loadTestsFromName("MyTests.test_1")suite = unittest.TestLoader().loadTestsFromNames("MyTests.test_1", "MyTests.test_2")
- Wrap with
ConcurrentTestSuite:
concurrent_suite = ConcurrentTestSuite(suite)concurrent_suite = ConcurrentTestSuite(suite, fork_for_tests(4))
- Run the suite using a unittest-compatible runner:
unittest.TextTestRunner().run(concurrent_suite)
Configuring number of processes and partition strategy:
The concurrencytest module provides a make_tests implementation
(fork_for_tests). This allows you to specify the number of worker processes
to use and a partition strategy for specifying how tests are distributed to
workers. If ConcurrentTestSuite is instantiated without a make_tests
argument, it defaults to forking one process per available CPU core, and
distributing tests using a round-robin strategy.
The fork_for_tests function is called with positional or keyword arguments
like this:
fork_for_tests(num_processes, partition_func)
num_processes(optional): Number of worker processes to spawn- Defaults to the number of CPUs on the system.
partition_func(optional): Function used to partition tests across workers.- Defaults to
partition_tests(round-robin partition strategy).
- Defaults to
Available partition functions:
-
partition_tests(round-robin):This is the default strategy.
This function splits a test suite into its individual test cases and assigns them in a round-robin fashion to distribute load evenly across workers. This helps avoid situations where one worker gets all slow tests while others finish quickly. One potential drawback is that if you have a
setUpClass/tearDownClassdefined in aTestCase, it may be run multiple times if tests from the same class are run on different workers. -
partition_tests_by_class(class-local):This function groups all tests belonging to the same test case class and assigns them as a block to the worker with the current smallest number of tests already assigned. This ensures that all tests from a single class run in the same worker, which preserves
setUpClass/tearDownClasslifecycle semantics.
Examples of creating a ConcurrentTestSuite:
-
default concurrency and round-robin partition strategy:
ConcurrentTestSuite(suite) -
4 worker processes and round-robin partition strategy:
ConcurrentTestSuite(suite, fork_for_tests(4)) -
default concurrency and class-local partition strategy:
ConcurrentTestSuite(suite, fork_for_tests(partition_func=partition_tests_by_class)) -
4 worker processes and class-local partition strategy:
ConcurrentTestSuite(suite, fork_for_tests(4, partition_tests_by_class))
Examples
Basic example:
import time
import unittest
from concurrencytest import ConcurrentTestSuite
"""Tests just sleep for demo."""
class ExampleTestCase(unittest.TestCase):
def test_1(self):
time.sleep(1)
def test_2(self):
time.sleep(1)
def test_3(self):
time.sleep(1)
def test_4(self):
time.sleep(1)
runner = unittest.TextTestRunner()
# Run the tests from above sequentially
suite = unittest.defaultTestLoader.loadTestsFromTestCase(ExampleTestCase)
print("running sequential (without concurrencytest):")
runner.run(suite)
print()
# Run same tests concurrently across multiple processes
# (1 process per available CPU core)
suite = unittest.defaultTestLoader.loadTestsFromTestCase(ExampleTestCase)
print("running parallel:")
concurrent_suite = ConcurrentTestSuite(suite)
runner.run(concurrent_suite)
Output:
running sequential (without concurrencytest):
....
----------------------------------------------------------------------
Ran 4 tests in 4.002s
OK
running parallel:
....
----------------------------------------------------------------------
Ran 4 tests in 1.009s
OK
Advanced example:
import time
import unittest
from concurrencytest import (
ConcurrentTestSuite,
fork_for_tests,
partition_tests_by_class,
)
"""Tests just sleep for demo."""
class ExampleTestCase1(unittest.TestCase):
def test_1(self):
time.sleep(1)
def test_2(self):
time.sleep(1)
class ExampleTestCase2(unittest.TestCase):
"""Dummy tests that sleep for demo."""
def test_3(self):
time.sleep(1)
def test_4(self):
time.sleep(1)
def load_test_suite(*test_cases):
suite = unittest.TestSuite()
for cls in test_cases:
suite.addTests(unittest.defaultTestLoader.loadTestsFromTestCase(cls))
return suite
runner = unittest.TextTestRunner()
# Run the tests from above sequentially
suite = load_test_suite(ExampleTestCase1, ExampleTestCase2)
print("running sequential (without concurrencytest):")
runner.run(suite)
print()
# Run same tests concurrently across multiple processes
# (1 process per available CPU core)
suite = load_test_suite(ExampleTestCase1, ExampleTestCase2)
concurrent_suite = ConcurrentTestSuite(suite)
print("running parallel:")
runner.run(concurrent_suite)
print()
# Run same tests concurrently across 4 processes
suite = load_test_suite(ExampleTestCase1, ExampleTestCase2)
concurrent_suite = ConcurrentTestSuite(suite, fork_for_tests(2))
print("running parallel (2 processes):")
runner.run(concurrent_suite)
print()
# Run same tests concurrently across multiple processes
# (1 process per available CPU core), keeping tests class-local
suite = load_test_suite(ExampleTestCase1, ExampleTestCase2)
concurrent_suite = ConcurrentTestSuite(
suite, fork_for_tests(partition_func=partition_tests_by_class)
)
print("running parallel (grouped by class):")
runner.run(concurrent_suite)
Output:
running sequential (without concurrencytest):
....
----------------------------------------------------------------------
Ran 4 tests in 4.002s
OK
running parallel:
....
----------------------------------------------------------------------
Ran 4 tests in 1.010s
OK
running parallel (2 processes):
....
----------------------------------------------------------------------
Ran 4 tests in 2.006s
OK
running parallel (grouped by class):
....
----------------------------------------------------------------------
Ran 4 tests in 2.008s
OK
Notes
For more info about writing/running tests with the unittest testing
framework, see the
official documentation.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file concurrencytest-0.1.11.tar.gz.
File metadata
- Download URL: concurrencytest-0.1.11.tar.gz
- Upload date:
- Size: 13.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eec8bf3015d45850dce131894fa609d87b7eb75daa8a2f5a94dbbb0f7de89a10
|
|
| MD5 |
57c871718d3550b7b6687907e0493275
|
|
| BLAKE2b-256 |
15a07eee9d266d33216a73606269cb76cad5333a0afe2a6b69de1b299db1b98b
|
File details
Details for the file concurrencytest-0.1.11-py3-none-any.whl.
File metadata
- Download URL: concurrencytest-0.1.11-py3-none-any.whl
- Upload date:
- Size: 13.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ffb2ef292ad50b27507621e04580622700fbeab360b4136e468c1225326cf4a0
|
|
| MD5 |
47be9db8859bebaa563fac11ac91e1c9
|
|
| BLAKE2b-256 |
e9a90884c9d83437a320bc3d2a3e277046dd6a51385be066bffff9783c75f26c
|