2019-04-23 22:44:42 +01:00
|
|
|
import asyncio
|
|
|
|
|
|
|
|
import httpcore
|
|
|
|
import requests_async as requests
|
2017-09-13 01:18:36 +01:00
|
|
|
|
2016-11-26 04:55:45 +00:00
|
|
|
from sanic import Sanic
|
|
|
|
from sanic.response import text
|
2019-04-23 22:44:42 +01:00
|
|
|
from sanic.testing import SanicTestClient
|
2017-09-13 01:18:36 +01:00
|
|
|
|
2018-08-16 06:00:23 +01:00
|
|
|
|
2019-04-23 22:44:42 +01:00
|
|
|
class DelayableSanicConnectionPool(httpcore.ConnectionPool):
|
|
|
|
def __init__(self, request_delay=None, *args, **kwargs):
|
|
|
|
self._request_delay = request_delay
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
async def request(
|
|
|
|
self,
|
|
|
|
method,
|
|
|
|
url,
|
|
|
|
headers=(),
|
|
|
|
body=b"",
|
|
|
|
stream=False,
|
|
|
|
ssl=None,
|
|
|
|
timeout=None,
|
|
|
|
):
|
|
|
|
if ssl is None:
|
|
|
|
ssl = self.ssl_config
|
|
|
|
if timeout is None:
|
|
|
|
timeout = self.timeout
|
|
|
|
|
|
|
|
parsed_url = httpcore.URL(url)
|
|
|
|
request = httpcore.Request(
|
|
|
|
method, parsed_url, headers=headers, body=body
|
2018-12-30 11:18:06 +00:00
|
|
|
)
|
2019-04-23 22:44:42 +01:00
|
|
|
connection = await self.acquire_connection(
|
|
|
|
parsed_url, ssl=ssl, timeout=timeout
|
2018-12-30 11:18:06 +00:00
|
|
|
)
|
2019-04-23 22:44:42 +01:00
|
|
|
if self._request_delay:
|
|
|
|
print(f"\t>> Sleeping ({self._request_delay})")
|
|
|
|
await asyncio.sleep(self._request_delay)
|
|
|
|
response = await connection.send(request)
|
|
|
|
if not stream:
|
|
|
|
try:
|
|
|
|
await response.read()
|
|
|
|
finally:
|
|
|
|
await response.close()
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
class DelayableSanicAdapter(requests.adapters.HTTPAdapter):
|
|
|
|
def __init__(self, request_delay=None):
|
|
|
|
self.pool = DelayableSanicConnectionPool(request_delay=request_delay)
|
|
|
|
|
|
|
|
|
|
|
|
class DelayableSanicSession(requests.Session):
|
|
|
|
def __init__(self, request_delay=None, *args, **kwargs) -> None:
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
adapter = DelayableSanicAdapter(request_delay=request_delay)
|
|
|
|
self.mount("http://", adapter)
|
|
|
|
self.mount("https://", adapter)
|
2016-11-26 04:55:45 +00:00
|
|
|
|
2017-09-11 08:17:33 +01:00
|
|
|
|
|
|
|
class DelayableSanicTestClient(SanicTestClient):
|
2019-04-23 22:44:42 +01:00
|
|
|
def __init__(self, app, request_delay=None):
|
|
|
|
super().__init__(app)
|
2017-09-11 08:17:33 +01:00
|
|
|
self._request_delay = request_delay
|
2017-09-13 01:18:36 +01:00
|
|
|
self._loop = None
|
2017-09-11 08:17:33 +01:00
|
|
|
|
2019-04-23 22:44:42 +01:00
|
|
|
def get_new_session(self):
|
|
|
|
return DelayableSanicSession(request_delay=self._request_delay)
|
2017-09-11 08:17:33 +01:00
|
|
|
|
|
|
|
|
2018-12-30 11:18:06 +00:00
|
|
|
request_timeout_default_app = Sanic("test_request_timeout_default")
|
|
|
|
request_no_timeout_app = Sanic("test_request_no_timeout")
|
2018-12-30 18:37:30 +00:00
|
|
|
request_timeout_default_app.config.REQUEST_TIMEOUT = 0.6
|
|
|
|
request_no_timeout_app.config.REQUEST_TIMEOUT = 0.6
|
2016-11-26 04:55:45 +00:00
|
|
|
|
|
|
|
|
2018-12-30 11:18:06 +00:00
|
|
|
@request_timeout_default_app.route("/1")
|
2017-09-13 01:18:36 +01:00
|
|
|
async def handler1(request):
|
2018-12-30 11:18:06 +00:00
|
|
|
return text("OK")
|
2017-09-13 01:18:36 +01:00
|
|
|
|
|
|
|
|
2018-12-30 11:18:06 +00:00
|
|
|
@request_no_timeout_app.route("/1")
|
2017-09-13 01:18:36 +01:00
|
|
|
async def handler2(request):
|
2018-12-30 11:18:06 +00:00
|
|
|
return text("OK")
|
2016-11-26 04:55:45 +00:00
|
|
|
|
|
|
|
|
2018-12-30 11:18:06 +00:00
|
|
|
@request_timeout_default_app.websocket("/ws1")
|
2018-12-22 15:21:45 +00:00
|
|
|
async def ws_handler1(request, ws):
|
2018-12-30 11:18:06 +00:00
|
|
|
await ws.send("OK")
|
2018-12-22 15:21:45 +00:00
|
|
|
|
|
|
|
|
2016-11-26 04:55:45 +00:00
|
|
|
def test_default_server_error_request_timeout():
|
2019-04-23 22:44:42 +01:00
|
|
|
client = DelayableSanicTestClient(request_timeout_default_app, 2)
|
2018-12-30 11:18:06 +00:00
|
|
|
request, response = client.get("/1")
|
2016-11-26 04:55:45 +00:00
|
|
|
assert response.status == 408
|
2018-12-30 11:18:06 +00:00
|
|
|
assert response.text == "Error: Request Timeout"
|
2017-09-13 01:18:36 +01:00
|
|
|
|
|
|
|
|
|
|
|
def test_default_server_error_request_dont_timeout():
|
2019-04-23 22:44:42 +01:00
|
|
|
client = DelayableSanicTestClient(request_no_timeout_app, 0.2)
|
2018-12-30 11:18:06 +00:00
|
|
|
request, response = client.get("/1")
|
2017-09-13 01:18:36 +01:00
|
|
|
assert response.status == 200
|
2018-12-30 11:18:06 +00:00
|
|
|
assert response.text == "OK"
|
2018-12-22 15:21:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_default_server_error_websocket_request_timeout():
|
|
|
|
|
2018-12-30 11:18:06 +00:00
|
|
|
headers = {
|
|
|
|
"Upgrade": "websocket",
|
|
|
|
"Connection": "upgrade",
|
|
|
|
"Sec-WebSocket-Key": "dGhlIHNhbXBsZSBub25jZQ==",
|
|
|
|
"Sec-WebSocket-Version": "13",
|
2018-12-22 15:21:45 +00:00
|
|
|
}
|
|
|
|
|
2019-04-23 22:44:42 +01:00
|
|
|
client = DelayableSanicTestClient(request_timeout_default_app, 2)
|
2018-12-30 11:18:06 +00:00
|
|
|
request, response = client.get("/ws1", headers=headers)
|
2018-12-22 15:21:45 +00:00
|
|
|
|
|
|
|
assert response.status == 408
|
2018-12-30 11:18:06 +00:00
|
|
|
assert response.text == "Error: Request Timeout"
|