Skip to content

Commit 32f9907

Browse files
Try supporting yield functions for setup/teardown. (#1)
* Support yield functions for setup/teardown. Co-authored-by: Blake Naccarato <[email protected]>
1 parent e05fcdd commit 32f9907

File tree

8 files changed

+213
-5
lines changed

8 files changed

+213
-5
lines changed

.coveragerc

+2-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,5 @@ branch = True
44
[paths]
55
source =
66
src
7-
.tox/*/site-packages
7+
.tox/*/lib/*/site-packages
8+

README.md

+46
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,52 @@ param teardown
8181
* If an exception occurs in setup, the test will report Error and not run. The teardown will also not run.
8282
* 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.
8383

84+
85+
## You can combine setup and teardown in one function
86+
87+
You can provide a function separated by a `yield` to put both setup and teardown in one function.
88+
89+
However, there's a trick to doing this:
90+
91+
* Either, pass `None` as the teardown.
92+
* Or use `with_args`, as in `@pytest.mark.param_scope.with_args(my_func)`
93+
94+
Here's a combo setup/teardown function:
95+
96+
```python
97+
def setup_and_teardown():
98+
print('\nsetup')
99+
yield 42
100+
print('\nteardown')
101+
102+
```
103+
104+
Calling it with `None` for teardown:
105+
106+
```python
107+
import pytest
108+
109+
@pytest.mark.param_scope(setup_and_teardown, None)
110+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
111+
def test_yield(x, param_scope):
112+
assert param_scope == 42
113+
114+
```
115+
116+
Or using `with_args`:
117+
118+
```python
119+
@pytest.mark.param_scope.with_args(setup_and_teardown)
120+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
121+
def test_just_one_func(x, param_scope):
122+
assert param_scope == 42
123+
124+
```
125+
126+
Both of these examples are in `examples/test_yield.py`.
127+
128+
129+
84130
## More examples
85131

86132
Please see `examples` directory in the repo.

examples/test_error.py

-1
Original file line numberDiff line numberDiff line change
@@ -36,4 +36,3 @@ def test_error_during_teardown(x):
3636
"""
3737
...
3838

39-

examples/test_marker_bad_params.py

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
import pytest
2+
3+
4+
def foo():
5+
...
6+
7+
@pytest.mark.param_scope(foo)
8+
def test_one_params_to_marker():
9+
"""
10+
This also blows up, with_args required.
11+
You gotta use `@pytest.mark.param_scope.with_args(foo)`
12+
"""
13+
...

examples/test_yield.py

+49
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
import pytest
2+
3+
4+
def setup_and_teardown():
5+
print('\nsetup')
6+
yield 42
7+
print('\nteardown')
8+
9+
10+
@pytest.mark.param_scope(setup_and_teardown, None)
11+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
12+
def test_yield(x, param_scope):
13+
assert param_scope == 42
14+
15+
16+
17+
def separate_teardown():
18+
print('separate teardown')
19+
20+
21+
@pytest.mark.param_scope(setup_and_teardown, separate_teardown)
22+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
23+
def test_two_teardowns(x, param_scope):
24+
"""
25+
For now, we'll allow this odd use model.
26+
Weird, but really, why not?
27+
"""
28+
assert param_scope == 42
29+
30+
31+
@pytest.mark.param_scope.with_args(setup_and_teardown)
32+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
33+
def test_just_one_func(x, param_scope):
34+
"""
35+
It's not pretty, but if you want to just pass in one,
36+
you gotta use "with_args".
37+
See "Passing a callable to custom markers" in pytest docs
38+
- https://docs.pytest.org/en/stable/example/markers.html#passing-a-callable-to-custom-markers
39+
"""
40+
assert param_scope == 42
41+
42+
43+
@pytest.mark.param_scope
44+
@pytest.mark.parametrize('x', ['a', 'b', 'c'])
45+
def test_no_param_scope_args(x):
46+
"""
47+
No point in this, but it doesn't blow up
48+
"""
49+
...

src/pytest_param_scope/plugin.py

+30-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
from __future__ import annotations
2+
import types
3+
from typing import Generator
24
import pytest
35
from dataclasses import dataclass
46
from typing import Callable, Any
@@ -14,6 +16,7 @@ def pytest_configure(config):
1416
class ParamScopeData():
1517
test_name: str | None = None
1618
teardown_func: Callable | None = None
19+
teardown_gen: Generator[Any, None, None] | None = None
1720
ready_for_teardown: bool = False
1821
setup_value: Any = None
1922
exception: Exception | None = None
@@ -39,12 +42,24 @@ def param_scope(request):
3942

4043
m = request.node.get_closest_marker("param_scope")
4144
if m:
42-
setup_func = m.args[0]
43-
__data.teardown_func = m.args[1]
45+
if len(m.args) >= 1:
46+
setup_func = m.args[0]
47+
if len(m.args) >= 2:
48+
__data.teardown_func = m.args[1]
4449

4550
if setup_func:
4651
try:
47-
__data.setup_value = setup_func()
52+
# setup could be a func, or could be a generator
53+
setup_value = setup_func()
54+
if isinstance(setup_value, types.GeneratorType):
55+
# if generator, call next() once for setup section
56+
new_value = next(setup_value)
57+
__data.setup_value = new_value
58+
# and save it for teardown
59+
__data.teardown_gen = setup_value
60+
else:
61+
# otherwise, just save the value
62+
__data.setup_value = setup_value
4863
except Exception as e:
4964
__data.exception = e
5065
raise e
@@ -58,8 +73,20 @@ def param_scope(request):
5873
yield __data.setup_value
5974

6075
if __data.ready_for_teardown:
76+
teardown_gen = __data.teardown_gen
6177
teardown_func = __data.teardown_func
78+
6279
__data = ParamScopeData() # reset for next one
80+
81+
if teardown_gen:
82+
try:
83+
next(teardown_gen)
84+
except StopIteration:
85+
pass # this is expected
86+
87+
88+
# should we disallow both a teardown from gen and a teardown?
89+
# for now, I'll allow it, as it's a bit messy to disallow it
6390
if teardown_func:
6491
teardown_func()
6592

tests/test_error.py

+17
Original file line numberDiff line numberDiff line change
@@ -27,3 +27,20 @@ def test_error_during_teardown(pytester):
2727
result = pytester.runpytest('test_error.py::test_error_during_teardown', '-v', '-s')
2828
result.assert_outcomes(passed=3, errors=1)
2929

30+
def test_error_marker_bad_params(pytester):
31+
"""
32+
Markers that accept functions have to accept 2 or more.
33+
34+
- all tests to pass
35+
- last test to error
36+
- yes, this is normal-ish for pytest with parametrized errors.
37+
"""
38+
pytester.copy_example("examples/test_marker_bad_params.py")
39+
result = pytester.runpytest('-v', '-s')
40+
result.assert_outcomes(errors=1)
41+
result.stdout.re_match_lines(
42+
[
43+
".*Interrupted: 1 error during collection.*"
44+
]
45+
)
46+

tests/test_yield.py

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
2+
def test_yield(pytester):
3+
pytester.copy_example("examples/test_yield.py")
4+
result = pytester.runpytest('test_yield.py::test_yield', '-v', '-s')
5+
result.assert_outcomes(passed=3)
6+
result.stdout.re_match_lines(
7+
[
8+
".*test_yield.a.*",
9+
"setup",
10+
".*test_yield.b.*",
11+
".*test_yield.c.*",
12+
"teardown",
13+
]
14+
)
15+
16+
def test_two_teardowns(pytester):
17+
pytester.copy_example("examples/test_yield.py")
18+
result = pytester.runpytest('test_yield.py::test_two_teardowns', '-v', '-s')
19+
result.assert_outcomes(passed=3)
20+
result.stdout.re_match_lines(
21+
[
22+
".*test_two_teardowns.a.*",
23+
"setup",
24+
".*test_two_teardowns.b.*",
25+
".*test_two_teardowns.c.*",
26+
"teardown",
27+
"separate teardown",
28+
]
29+
)
30+
31+
def test_one_param(pytester):
32+
pytester.copy_example("examples/test_yield.py")
33+
result = pytester.runpytest('test_yield.py::test_just_one_func', '-v', '-s')
34+
result.assert_outcomes(passed=3)
35+
result.stdout.re_match_lines(
36+
[
37+
".*test_just_one_func.a.*",
38+
"setup",
39+
".*test_just_one_func.b.*",
40+
".*test_just_one_func.c.*",
41+
"teardown",
42+
]
43+
)
44+
45+
46+
def test_no_params(pytester):
47+
pytester.copy_example("examples/test_yield.py")
48+
result = pytester.runpytest('test_yield.py::test_no_param_scope_args', '-v', '-s')
49+
result.assert_outcomes(passed=3)
50+
result.stdout.re_match_lines(
51+
[
52+
".*test_no_param_scope_args.a.*",
53+
".*test_no_param_scope_args.b.*",
54+
".*test_no_param_scope_args.c.*",
55+
]
56+
)

0 commit comments

Comments
 (0)