forked from aio-libs/aiohttp
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathtest_test_utils.py
More file actions
321 lines (238 loc) · 9.41 KB
/
test_test_utils.py
File metadata and controls
321 lines (238 loc) · 9.41 KB
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
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
import gzip
from unittest import mock
import pytest
from multidict import CIMultiDict, CIMultiDictProxy
from yarl import URL
import aiohttp
from aiohttp import web
from aiohttp.test_utils import AioHTTPTestCase
from aiohttp.test_utils import RawTestServer as _RawTestServer
from aiohttp.test_utils import TestClient as _TestClient
from aiohttp.test_utils import TestServer as _TestServer
from aiohttp.test_utils import (
loop_context,
make_mocked_request,
unittest_run_loop,
)
_hello_world_str = "Hello, world"
_hello_world_bytes = _hello_world_str.encode('utf-8')
_hello_world_gz = gzip.compress(_hello_world_bytes)
def _create_example_app():
async def hello(request):
return web.Response(body=_hello_world_bytes)
async def websocket_handler(request):
ws = web.WebSocketResponse()
await ws.prepare(request)
msg = await ws.receive()
if msg.type == aiohttp.WSMsgType.TEXT:
if msg.data == 'close':
await ws.close()
else:
await ws.send_str(msg.data + '/answer')
return ws
async def cookie_handler(request):
resp = web.Response(body=_hello_world_bytes)
resp.set_cookie('cookie', 'val')
return resp
app = web.Application()
app.router.add_route('*', '/', hello)
app.router.add_route('*', '/websocket', websocket_handler)
app.router.add_route('*', '/cookie', cookie_handler)
return app
# these exist to test the pytest scenario
@pytest.fixture
def loop():
with loop_context() as loop:
yield loop
@pytest.fixture
def app():
return _create_example_app()
@pytest.fixture
def test_client(loop, app) -> None:
async def make_client():
return _TestClient(_TestServer(app, loop=loop), loop=loop)
client = loop.run_until_complete(make_client())
loop.run_until_complete(client.start_server())
yield client
loop.run_until_complete(client.close())
def test_with_test_server_fails(loop) -> None:
app = _create_example_app()
with pytest.raises(TypeError):
with _TestServer(app, loop=loop):
pass
async def test_with_client_fails(loop) -> None:
app = _create_example_app()
with pytest.raises(TypeError):
with _TestClient(_TestServer(app, loop=loop), loop=loop):
pass
async def test_aiohttp_client_close_is_idempotent() -> None:
"""
a test client, called multiple times, should
not attempt to close the server again.
"""
app = _create_example_app()
client = _TestClient(_TestServer(app))
await client.close()
await client.close()
class TestAioHTTPTestCase(AioHTTPTestCase):
def get_app(self):
return _create_example_app()
@unittest_run_loop
async def test_example_with_loop(self) -> None:
request = await self.client.request("GET", "/")
assert request.status == 200
text = await request.text()
assert _hello_world_str == text
def test_example(self) -> None:
async def test_get_route() -> None:
resp = await self.client.request("GET", "/")
assert resp.status == 200
text = await resp.text()
assert _hello_world_str == text
self.loop.run_until_complete(test_get_route())
def test_get_route(loop, test_client) -> None:
async def test_get_route() -> None:
resp = await test_client.request("GET", "/")
assert resp.status == 200
text = await resp.text()
assert _hello_world_str == text
loop.run_until_complete(test_get_route())
async def test_client_websocket(loop, test_client) -> None:
resp = await test_client.ws_connect("/websocket")
await resp.send_str("foo")
msg = await resp.receive()
assert msg.type == aiohttp.WSMsgType.TEXT
assert "foo" in msg.data
await resp.send_str("close")
msg = await resp.receive()
assert msg.type == aiohttp.WSMsgType.CLOSE
async def test_client_cookie(loop, test_client) -> None:
assert not test_client.session.cookie_jar
await test_client.get("/cookie")
cookies = list(test_client.session.cookie_jar)
assert cookies[0].key == 'cookie'
assert cookies[0].value == 'val'
@pytest.mark.parametrize("method", [
"get", "post", "options", "post", "put", "patch", "delete"
])
async def test_test_client_methods(method, loop, test_client) -> None:
resp = await getattr(test_client, method)("/")
assert resp.status == 200
text = await resp.text()
assert _hello_world_str == text
async def test_test_client_head(loop, test_client) -> None:
resp = await test_client.head("/")
assert resp.status == 200
@pytest.mark.parametrize(
"headers", [{'token': 'x'}, CIMultiDict({'token': 'x'}), {}])
def test_make_mocked_request(headers) -> None:
req = make_mocked_request('GET', '/', headers=headers)
assert req.method == "GET"
assert req.path == "/"
assert isinstance(req, web.Request)
assert isinstance(req.headers, CIMultiDictProxy)
def test_make_mocked_request_sslcontext() -> None:
req = make_mocked_request('GET', '/')
assert req.transport.get_extra_info('sslcontext') is None
def test_make_mocked_request_unknown_extra_info() -> None:
req = make_mocked_request('GET', '/')
assert req.transport.get_extra_info('unknown_extra_info') is None
def test_make_mocked_request_app() -> None:
app = mock.Mock()
req = make_mocked_request('GET', '/', app=app)
assert req.app is app
def test_make_mocked_request_app_can_store_values() -> None:
req = make_mocked_request('GET', '/')
req.app['a_field'] = 'a_value'
assert req.app['a_field'] == 'a_value'
def test_make_mocked_request_match_info() -> None:
req = make_mocked_request('GET', '/', match_info={'a': '1', 'b': '2'})
assert req.match_info == {'a': '1', 'b': '2'}
def test_make_mocked_request_content() -> None:
payload = mock.Mock()
req = make_mocked_request('GET', '/', payload=payload)
assert req.content is payload
def test_make_mocked_request_transport() -> None:
transport = mock.Mock()
req = make_mocked_request('GET', '/', transport=transport)
assert req.transport is transport
async def test_test_client_props(loop) -> None:
app = _create_example_app()
client = _TestClient(_TestServer(app, host='127.0.0.1', loop=loop),
loop=loop)
assert client.host == '127.0.0.1'
assert client.port is None
async with client:
assert isinstance(client.port, int)
assert client.server is not None
assert client.app is not None
assert client.port is None
async def test_test_client_raw_server_props(loop) -> None:
async def hello(request):
return web.Response(body=_hello_world_bytes)
client = _TestClient(_RawTestServer(hello, host='127.0.0.1', loop=loop),
loop=loop)
assert client.host == '127.0.0.1'
assert client.port is None
async with client:
assert isinstance(client.port, int)
assert client.server is not None
assert client.app is None
assert client.port is None
async def test_test_server_context_manager(loop) -> None:
app = _create_example_app()
async with _TestServer(app, loop=loop) as server:
client = aiohttp.ClientSession(loop=loop)
resp = await client.head(server.make_url('/'))
assert resp.status == 200
resp.close()
await client.close()
def test_client_unsupported_arg() -> None:
with pytest.raises(TypeError) as e:
_TestClient('string')
assert str(e.value) == \
"server must be TestServer instance, found type: <class 'str'>"
async def test_server_make_url_yarl_compatibility(loop) -> None:
app = _create_example_app()
async with _TestServer(app, loop=loop) as server:
make_url = server.make_url
assert make_url(URL('/foo')) == make_url('/foo')
with pytest.raises(AssertionError):
make_url('http://foo.com')
with pytest.raises(AssertionError):
make_url(URL('http://foo.com'))
def test_testcase_no_app(testdir, loop) -> None:
testdir.makepyfile(
"""
from aiohttp.test_utils import AioHTTPTestCase
class InvalidTestCase(AioHTTPTestCase):
def test_noop(self) -> None:
pass
""")
result = testdir.runpytest()
result.stdout.fnmatch_lines(["*RuntimeError*"])
async def test_server_context_manager(app, loop) -> None:
async with _TestServer(app, loop=loop) as server:
async with aiohttp.ClientSession(loop=loop) as client:
async with client.head(server.make_url('/')) as resp:
assert resp.status == 200
@pytest.mark.parametrize("method", [
"head", "get", "post", "options", "post", "put", "patch", "delete"
])
async def test_client_context_manager_response(method, app, loop) -> None:
async with _TestClient(_TestServer(app), loop=loop) as client:
async with getattr(client, method)('/') as resp:
assert resp.status == 200
if method != 'head':
text = await resp.text()
assert "Hello, world" in text
async def test_custom_port(loop, app, aiohttp_unused_port) -> None:
port = aiohttp_unused_port()
client = _TestClient(_TestServer(app, loop=loop, port=port), loop=loop)
await client.start_server()
assert client.server.port == port
resp = await client.get('/')
assert resp.status == 200
text = await resp.text()
assert _hello_world_str == text
await client.close()