|
| 1 | +# pytest-param-scope |
| 2 | + |
| 3 | +The pytest parametrize scope fixture workaround. |
| 4 | + |
| 5 | +---- |
| 6 | + |
| 7 | +`pytest-param-scope` provides a `param_scope` marker to pass setup and teardown functions to a parametrized function. |
| 8 | + |
| 9 | +There's also a `param_scope` fixture to allow the return value of the setup function to be passed to the test. |
| 10 | + |
| 11 | +## Installation |
| 12 | + |
| 13 | +From PyPI: |
| 14 | + |
| 15 | +``` |
| 16 | +$ pip install pytest-param-scope |
| 17 | +``` |
| 18 | + |
| 19 | +## Example |
| 20 | + |
| 21 | + |
| 22 | +```python |
| 23 | +import pytest |
| 24 | + |
| 25 | + |
| 26 | +def param_setup(): |
| 27 | + print('\nparam setup') |
| 28 | + return 'some data from setup' |
| 29 | + |
| 30 | + |
| 31 | +def param_teardown(): |
| 32 | + print('\nparam teardown') |
| 33 | + |
| 34 | + |
| 35 | +@pytest.mark.param_scope(param_setup, param_teardown) |
| 36 | +@pytest.mark.parametrize('x', ['a', 'b', 'c']) |
| 37 | +def test_param(x, param_scope): |
| 38 | + """ |
| 39 | + mark.param_scope is used to pass |
| 40 | + setup and teardown functions to fixture |
| 41 | +
|
| 42 | + param_scope fixture is necessary if you |
| 43 | + want to use the value from setup |
| 44 | + """ |
| 45 | + assert param_scope == 'some data from setup' |
| 46 | +``` |
| 47 | + |
| 48 | +**Let's see it run:** |
| 49 | + |
| 50 | +```shell |
| 51 | +(venv) $ pytest -s -v test_param_scope.py::test_param |
| 52 | +================== test session starts =================== |
| 53 | +collected 3 items |
| 54 | + |
| 55 | +test_param_scope.py::test_param[a] |
| 56 | +param setup |
| 57 | +PASSED |
| 58 | +test_param_scope.py::test_param[b] PASSED |
| 59 | +test_param_scope.py::test_param[c] PASSED |
| 60 | +param teardown |
| 61 | + |
| 62 | +=================== 3 passed in 0.01s ==================== |
| 63 | + |
| 64 | +``` |
| 65 | + |
| 66 | +**What are we seeing?** |
| 67 | + |
| 68 | +1. Setup is run before the first parameter in a set. |
| 69 | +2. Teardown is run after the last parameter. |
| 70 | +3. The `param_scope` fixture holds the return value of the setup function. |
| 71 | + |
| 72 | + |
| 73 | +## Similarities to Fixtures |
| 74 | + |
| 75 | +* Teardown is not run if the setup fails. |
| 76 | +* Setup is run once, even though the value can be retrieved by all parametrized test cases. |
| 77 | +* If an exception occurs in setup, the test will report Error and not run. The teardown will also not run. |
| 78 | +* If an exception occurs in teardown, the LAST parametrized test case to run results in BOTH PASS and Error. This is weird, but consistent with pytest fixtures. |
| 79 | + |
| 80 | +## More examples |
| 81 | + |
| 82 | +Please see `examples` directory in the repo. |
| 83 | + |
| 84 | + |
| 85 | +## Limitations |
| 86 | + |
| 87 | +The implementation is a total hack and relies on global variables and looking up the next test to see when to run the teardown. There is undoubtedly room for improvement. |
| 88 | + |
| 89 | +* With `pytest-xdist`: I haven't come up with a failing case yet, but it seems like this will be sketchy with tests running in parallel. |
| 90 | + |
| 91 | +* With `pytest-repeat`: This actually works great with `pytest-repeat`, as repeat works by generating parametrized tests. |
| 92 | + |
| 93 | + |
| 94 | +## Contributing |
| 95 | + |
| 96 | +Contributions are very welcome. Tests can be run with [tox](https://tox.readthedocs.io/en/latest/). |
| 97 | + |
| 98 | +## License |
| 99 | + |
| 100 | +Distributed under the terms of the [MIT](http://opensource.org/licenses/MIT) license, "pytest-param-scope" is free and open source software |
| 101 | + |
| 102 | +## Issues |
| 103 | + |
| 104 | +If you encounter any problems, please [file an issue](https://github.com/okken/pytest-param-scope/issues) along with a detailed description. |
| 105 | + |
| 106 | +## Changelog |
| 107 | + |
| 108 | +See [changelog.md](https://github.com/okken/pytest-param-scope/blob/main/changelog.md) |
0 commit comments