forked from pytest-dev/pytest-asyncio
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest_function_scope.py
230 lines (189 loc) · 6.27 KB
/
test_function_scope.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
from textwrap import dedent
from pytest import Pytester
def test_asyncio_mark_provides_function_scoped_loop_strict_mode(pytester: Pytester):
pytester.makepyfile(
dedent(
"""\
import asyncio
import pytest
pytestmark = pytest.mark.asyncio
loop: asyncio.AbstractEventLoop
async def test_remember_loop():
global loop
loop = asyncio.get_running_loop()
async def test_does_not_run_in_same_loop():
global loop
assert asyncio.get_running_loop() is not loop
"""
)
)
result = pytester.runpytest("--asyncio-mode=strict")
result.assert_outcomes(passed=2)
def test_function_scope_supports_explicit_event_loop_fixture_request(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import pytest
pytestmark = pytest.mark.asyncio
async def test_remember_loop(event_loop):
pass
"""
)
)
result = pytester.runpytest_subprocess("--asyncio-mode=strict", "-W default")
result.assert_outcomes(passed=1, warnings=1)
result.stdout.fnmatch_lines(
'*is asynchronous and explicitly requests the "event_loop" fixture*'
)
def test_asyncio_mark_respects_the_loop_policy(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import asyncio
import pytest
pytestmark = pytest.mark.asyncio
class CustomEventLoopPolicy(asyncio.DefaultEventLoopPolicy):
pass
@pytest.fixture(scope="function")
def event_loop_policy():
return CustomEventLoopPolicy()
async def test_uses_custom_event_loop_policy():
assert isinstance(
asyncio.get_event_loop_policy(),
CustomEventLoopPolicy,
)
"""
),
)
result = pytester.runpytest("--asyncio-mode=strict")
result.assert_outcomes(passed=1)
def test_asyncio_mark_respects_parametrized_loop_policies(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import asyncio
import pytest
pytestmark = pytest.mark.asyncio
class CustomEventLoopPolicy(asyncio.DefaultEventLoopPolicy):
pass
@pytest.fixture(
scope="module",
params=[
CustomEventLoopPolicy(),
CustomEventLoopPolicy(),
],
)
def event_loop_policy(request):
return request.param
async def test_parametrized_loop():
assert isinstance(
asyncio.get_event_loop_policy(),
CustomEventLoopPolicy,
)
"""
)
)
result = pytester.runpytest_subprocess("--asyncio-mode=strict")
result.assert_outcomes(passed=2)
def test_asyncio_mark_provides_function_scoped_loop_to_fixtures(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import asyncio
import pytest
import pytest_asyncio
pytestmark = pytest.mark.asyncio
loop: asyncio.AbstractEventLoop
@pytest_asyncio.fixture
async def my_fixture():
global loop
loop = asyncio.get_running_loop()
async def test_runs_is_same_loop_as_fixture(my_fixture):
global loop
assert asyncio.get_running_loop() is loop
"""
)
)
result = pytester.runpytest_subprocess("--asyncio-mode=strict")
result.assert_outcomes(passed=1)
def test_asyncio_mark_handles_missing_event_loop_triggered_by_fixture(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import pytest
import asyncio
@pytest.fixture
def sets_event_loop_to_none():
# asyncio.run() creates a new event loop without closing the existing
# one. For any test, but the first one, this leads to a ResourceWarning
# when the discarded loop is destroyed by the garbage collector.
# We close the current loop to avoid this
try:
asyncio.get_event_loop().close()
except RuntimeError:
pass
return asyncio.run(asyncio.sleep(0))
# asyncio.run() sets the current event loop to None when finished
@pytest.mark.asyncio
# parametrization may impact fixture ordering
@pytest.mark.parametrize("n", (0, 1))
async def test_does_not_fail(sets_event_loop_to_none, n):
pass
"""
)
)
result = pytester.runpytest("--asyncio-mode=strict")
result.assert_outcomes(passed=2)
def test_standalone_test_does_not_trigger_warning_about_no_current_event_loop_being_set(
pytester: Pytester,
):
pytester.makepyfile(
dedent(
"""\
import pytest
@pytest.mark.asyncio
async def test_anything():
pass
"""
)
)
result = pytester.runpytest_subprocess("--asyncio-mode=strict")
result.assert_outcomes(warnings=0, passed=1)
def test_asyncio_mark_does_not_duplicate_other_marks_in_auto_mode(
pytester: Pytester,
):
pytester.makeconftest(
dedent(
"""\
def pytest_configure(config):
config.addinivalue_line(
"markers", "dummy_marker: mark used for testing purposes"
)
"""
)
)
pytester.makepyfile(
dedent(
"""\
import pytest
@pytest.mark.dummy_marker
async def test_markers_not_duplicated(request):
markers = []
for node, marker in request.node.iter_markers_with_node():
markers.append(marker)
assert len(markers) == 2
"""
)
)
result = pytester.runpytest_subprocess("--asyncio-mode=auto")
result.assert_outcomes(warnings=0, passed=1)