Parametrize tests within unittest TestCases.
Project description
Parametrize tests within unittest TestCases.
Testing a Django project? Check out my book Speed Up Your Django Tests which covers loads of recommendations to write faster, more accurate tests.
Installation
Install with:
python -m pip install unittest-parametrize
Python 3.9 to 3.14 supported.
Usage
The API mirrors @pytest.mark.parametrize as much as possible. (Even the name parametrize over the slightly more common parameterize with an extra “e”. Don’t get caught out by that…)
There are two steps to parametrize a test case:
Use ParametrizedTestCase in the base classes for your test case.
Apply @parametrize to any test methods for parametrization. This decorator takes (at least):
the argument names to parametrize, as comma-separated string or sequence of strings.
a list of parameters to create individual tests for, which may be tuples, param objects, or single values (for one argument).
Here’s a basic example:
from unittest_parametrize import ParametrizedTestCase, parametrize
class SquareTests(ParametrizedTestCase):
@parametrize(
"x,expected",
[
(1, 1),
(2, 4),
],
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
@parametrize modifies the class at definition time with Python’s __init_subclass__ hook. It removes the original test method and creates wrapped copies with individual names. Thus the parametrization should work regardless of the test runner you use (be it unittest, Django’s test runner, pytest, etc.). It supports both synchronous and asynchronous test methods.
Provide a single parameter without a wrapping tuple
If you only need a single parameter, you can provide values without wrapping them in tuples:
from unittest_parametrize import ParametrizedTestCase, parametrize
class EqualTests(ParametrizedTestCase):
@parametrize(
"x",
[1, 2, 3],
)
def test_equal(self, x: int) -> None:
self.assertEqual(x, x)
Provide argument names as separate strings
You can provide argument names as a sequence of strings instead:
from unittest_parametrize import ParametrizedTestCase, parametrize
class SquareTests(ParametrizedTestCase):
@parametrize(
("x", "expected"),
[
(1, 1),
(2, 4),
],
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
Use ParametrizedTestCase in your base test case class
ParametrizedTestCase does nothing if there aren’t any @parametrize-decorated tests within a class. Therefore you can include it in your project’s base test case class so that @parametrize works immediately in all test cases.
For example, within a Django project, you can create a set of project-specific base test case classes extending those provided by Django. You can do this in a module like example.test, and use the base classes throughout your test suite. To add ParametrizedTestCase to all your copies, use it in a custom SimpleTestCase and then mixin to others using multiple inheritance like so:
from django import test
from unittest_parametrize import ParametrizedTestCase
class SimpleTestCase(ParametrizedTestCase, test.SimpleTestCase):
pass
class TestCase(SimpleTestCase, test.TestCase):
pass
class TransactionTestCase(SimpleTestCase, test.TransactionTestCase):
pass
class LiveServerTestCase(SimpleTestCase, test.LiveServerTestCase):
pass
Custom test name suffixes
By default, test names are extended with an index, starting at zero. You can see these names when running the tests:
$ python -m unittest t.py -v
test_square_0 (t.SquareTests.test_square_0) ... ok
test_square_1 (t.SquareTests.test_square_1) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK
You can customize these names in several ways:
Using param objects with IDs.
Passing a sequence of strings as the ids argument.
Passing a callable as the ids argument.
Passing param objects with IDs
Pass a param object for each parameter set, setting the test ID suffix with the optional id argument:
from unittest_parametrize import ParametrizedTestCase, param, parametrize
class SquareTests(ParametrizedTestCase):
@parametrize(
"x,expected",
[
param(1, 1, id="one"),
param(2, 4, id="two"),
],
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
Yielding more natural names:
$ python -m unittest t.py -v
test_square_one (t.SquareTests.test_square_one) ... ok
test_square_two (t.SquareTests.test_square_two) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK
Parameter IDs should be valid Python identifier suffixes.
Since parameter IDs are optional, you can provide them only for some tests:
from unittest_parametrize import ParametrizedTestCase, param, parametrize
class SquareTests(ParametrizedTestCase):
@parametrize(
"x,expected",
[
param(1, 1),
param(20, 400, id="large"),
],
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
The ID-free params fall back to the default index suffixes:
$ python -m unittest t.py -v
test_square_0 (example.SquareTests.test_square_0) ... ok
test_square_large (example.SquareTests.test_square_large) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK
Passing a sequence of strings as the ids argument
Another option is to provide the IDs in the separate ids argument:
from unittest_parametrize import ParametrizedTestCase, parametrize
class SquareTests(ParametrizedTestCase):
@parametrize(
"x,expected",
[
(1, 1),
(2, 4),
],
ids=["one", "two"],
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
This option sets the full suffixes to the provided strings:
$ python -m unittest t.py -v
test_square_one (example.SquareTests.test_square_one) ... ok
test_square_two (example.SquareTests.test_square_two) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK
Passing a callable as the ids argument
The ids argument can also be a callable, which unittest-parametrize calls once per parameter value. The callable can return a string for that value, or None to use the default index suffix. The values are then joined with underscores to form the full suffix.
For example:
from unittest_parametrize import ParametrizedTestCase, parametrize
def make_id(value):
if isinstance(value, int):
return f"num{value}"
return None
class SquareTests(ParametrizedTestCase):
@parametrize(
"x,expected",
[
(1, 1),
(2, 4),
],
ids=make_id,
)
def test_square(self, x: int, expected: int) -> None:
self.assertEqual(x**2, expected)
…yields:
$ python -m unittest t.py -v
test_square_num1_num1 (example.SquareTests.test_square_num1_num1) ... ok
test_square_num2_num4 (example.SquareTests.test_square_num2_num4) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK
Use with other test decorators
@parametrize tries to ensure it is the top-most (outermost) decorator. This limitation exists to ensure that other decorators apply to each parametrized test. So decorators like @mock.patch need be beneath @parametrize:
from unittest import mock
from unittest_parametrize import ParametrizedTestCase, parametrize
class CarpentryTests(ParametrizedTestCase):
@parametrize(
"nails",
[(11,), (17,)],
)
@mock.patch("example.hammer", autospec=True)
def test_nail_a_board(self, mock_hammer, nails): ...
Also note that due to how mock.patch always adds positional arguments at the start, the parametrized arguments must come last. @parametrize always adds parameters as keyword arguments, so you can also use keyword-only syntax for parametrized arguments:
# ...
def test_nail_a_board(self, mock_hammer, *, nails): ...
Multiple @parametrize decorators
@parametrize is not stackable. To create a cross-product of tests, you can use nested list comprehensions:
from unittest_parametrize import ParametrizedTestCase, parametrize
class RocketTests(ParametrizedTestCase):
@parametrize(
"use_ions,hyperdrive_level",
[
(use_ions, hyperdrive_level)
for use_ions in [True, False]
for hyperdrive_level in [0, 1, 2]
],
)
def test_takeoff(self, use_ions, hyperdrive_level) -> None: ...
The above creates 2 * 3 = 6 versions of test_takeoff.
For larger combinations, itertools.product() may be more readable:
from itertools import product
from unittest_parametrize import ParametrizedTestCase, parametrize
class RocketTests(ParametrizedTestCase):
@parametrize(
"use_ions,hyperdrive_level,nose_colour",
list(
product(
[True, False],
[0, 1, 2],
["red", "yellow"],
)
),
)
def test_takeoff(self, use_ions, hyperdrive_level, nose_colour) -> None: ...
The above creates 2 * 3 * 2 = 12 versions of test_takeoff.
Parametrizing multiple tests in a test case
@parametrize only works as a function decorator, not a class decorator. To parametrize all tests within a test case, create a separate decorator and apply it to each method:
from unittest_parametrize import parametrize
from unittest_parametrize import ParametrizedTestCase
parametrize_race = parametrize(
"race",
[("Human",), ("Halfling",), ("Dwarf",), ("Elf",)],
)
class StatsTests(ParametrizedTestCase):
@parametrize_race
def test_strength(self, race: str) -> None: ...
@parametrize_race
def test_dexterity(self, race: str) -> None: ...
...
Pass parameters in a dataclass
Thanks to Florian Bruhin for this tip, from his pytest tips and tricks presentation.
If your test uses many parameters or cases, the parametrization may become unwieldy, as cases don’t name the arguments. In this case, try using a dataclass to hold the arguments:
from dataclasses import dataclass
from unittest_parametrize import ParametrizedTestCase, parametrize
@dataclass
class SquareParams:
x: int
expected: int
class SquareTests(ParametrizedTestCase):
@parametrize(
"sp",
[
(SquareParams(x=1, expected=1),),
(SquareParams(x=2, expected=4),),
],
)
def test_square(self, sp: SquareParams) -> None:
self.assertEqual(sp.x**2, sp.expected)
This way, each parameter is type-checked and named, improving safety and readability.
History
When I started writing unit tests, I learned to use DDT (Data-Driven Tests) for parametrizing tests. It works, but the docs are a bit thin, and the API a little obscure (what does @ddt stand for again?).
Later when picking up pytest, I learned to use its parametrization API. It’s legible and flexible, but it doesn’t work with unittest test cases, which Django’s test tooling provides.
So, until the creation of this package, I was using parameterized on my (Django) test cases. This package supports parametrization across multiple test runners, though most of them are “legacy” by now.
I created unittest-parametrize as a smaller alternative to parameterized, with these goals:
Only support unittest test cases. For other types of test, you can use pytest’s parametrization.
Avoid any custom test runner support. Modifying the class at definition time means that all test runners will see the tests the same.
Use modern Python features like __init_subclass__.
Have full type hint coverage. You shouldn’t find unittest-parametrize a blocker when adopting Mypy with strict mode on.
Use the name “parametrize” rather than “parameterize”. This unification of spelling with pytest should help reduce confusion around the extra “e”.
Thanks to the creators and maintainers of ddt, parameterized, and pytest for their hard work.
Why not subtests?
TestCase.subTest() is unittest’s built-in “parametrization” solution. You use it in a loop within a single test method:
from unittest import TestCase
class SquareTests(TestCase):
def test_square(self):
tests = [
(1, 1),
(2, 4),
]
for x, expected in tests:
with self.subTest(x=x):
self.assertEqual(x**2, expected)
This approach crams multiple actual tests into one test method, with several consequences:
If a subtest fails, it prevents the next subtests from running. Thus, failures are harder to debug, since each test run can only give you partial information.
Subtests can leak state. Without correct isolation, they may not test what they appear to.
Subtests cannot be reordered by tools that detect state leakage, like pytest-randomly.
Subtests skew test timings, since the test method runs multiple tests.
Everything is indented two extra levels for the loop and context manager.
Parametrization avoids all these issues by creating individual test methods.
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 unittest_parametrize-1.8.0.tar.gz.
File metadata
- Download URL: unittest_parametrize-1.8.0.tar.gz
- Upload date:
- Size: 15.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
90818ddeb28648e795eb53983f42d44e273b26ab03a1db49c27f6830ed419236
|
|
| MD5 |
44453f53e733a7f8e925ad97464f9cce
|
|
| BLAKE2b-256 |
d05a32bb766239d938f6030dd3920387dedc30b1c89aca01cb5281c3075bdf87
|
Provenance
The following attestation bundles were made for unittest_parametrize-1.8.0.tar.gz:
Publisher:
main.yml on adamchainz/unittest-parametrize
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
unittest_parametrize-1.8.0.tar.gz -
Subject digest:
90818ddeb28648e795eb53983f42d44e273b26ab03a1db49c27f6830ed419236 - Sigstore transparency entry: 487859974
- Sigstore integration time:
-
Permalink:
adamchainz/unittest-parametrize@cdcaa7da11ec0f43bbbac25a25d3bb6b1201c8ab -
Branch / Tag:
refs/tags/1.8.0 - Owner: https://github.com/adamchainz
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
main.yml@cdcaa7da11ec0f43bbbac25a25d3bb6b1201c8ab -
Trigger Event:
push
-
Statement type:
File details
Details for the file unittest_parametrize-1.8.0-py3-none-any.whl.
File metadata
- Download URL: unittest_parametrize-1.8.0-py3-none-any.whl
- Upload date:
- Size: 9.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
10a580aa04d13cf4f1c8e5e07be5694392b7ecef3b4d585bfc2bddce65bc3275
|
|
| MD5 |
7b6ac10ac1f40fce08e287d95605d647
|
|
| BLAKE2b-256 |
ab62193690ddc422c7af943b3f371ef34742dbb6a061e0b348e356399a52004d
|
Provenance
The following attestation bundles were made for unittest_parametrize-1.8.0-py3-none-any.whl:
Publisher:
main.yml on adamchainz/unittest-parametrize
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
unittest_parametrize-1.8.0-py3-none-any.whl -
Subject digest:
10a580aa04d13cf4f1c8e5e07be5694392b7ecef3b4d585bfc2bddce65bc3275 - Sigstore transparency entry: 487860006
- Sigstore integration time:
-
Permalink:
adamchainz/unittest-parametrize@cdcaa7da11ec0f43bbbac25a25d3bb6b1201c8ab -
Branch / Tag:
refs/tags/1.8.0 - Owner: https://github.com/adamchainz
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
main.yml@cdcaa7da11ec0f43bbbac25a25d3bb6b1201c8ab -
Trigger Event:
push
-
Statement type: