
* Streaming request by async for. * Make all requests streaming and preload body for non-streaming handlers. * Cleanup of code and avoid mixing streaming responses. * Async http protocol loop. * Change of test: don't require early bad request error but only after CRLF-CRLF. * Add back streaming requests. * Rewritten request body parser. * Misc. cleanup, down to 4 failing tests. * All tests OK. * Entirely remove request body queue. * Let black f*ckup the layout * Better testing error messages on protocol errors. * Remove StreamBuffer tests because the type is about to be removed. * Remove tests using the deprecated get_headers function that can no longer be supported. Chunked mode is now autodetected, so do not put content-length header if chunked mode is preferred. * Major refactoring of HTTP protocol handling (new module http.py added), all requests made streaming. A few compatibility issues and a lot of cleanup to be done remain, 16 tests failing. * Terminate check_timeouts once connection_task finishes. * Code cleanup, 14 tests failing. * Much cleanup, 12 failing... * Even more cleanup and error checking, 8 failing tests. * Remove keep-alive header from responses. First of all, it should say timeout=<value> which wasn't the case with existing implementation, and secondly none of the other web servers I tried include this header. * Everything but CustomServer OK. * Linter * Disable custom protocol test * Remove unnecessary variables, optimise performance. * A test was missing that body_init/body_push/body_finish are never called. Rewritten using receive_body and case switching to make it fail if bypassed. * Minor fixes. * Remove unused code. * Py 3.8 check for deprecated loop argument. * Fix a middleware cancellation handling test with py38. * Linter 'n fixes * Typing * Stricter handling of request header size * More specific error messages on Payload Too Large. * Init http.response = None * Messages further tuned. * Always try to consume request body, plus minor cleanup. * Add a missing check in case of close_if_idle on a dead connection. * Avoid error messages on PayloadTooLarge. * Add test for new API. * json takes str, not bytes * Default to no maximum request size for streaming handlers. * Fix chunked mode crash. * Header values should be strictly ASCII but both UTF-8 and Latin-1 exist. Use UTF-8B to cope with all. * Refactoring and cleanup. * Unify response header processing of ASGI and asyncio modes. * Avoid special handling of StreamingHTTPResponse. * 35 % speedup in HTTP/1.1 response formatting (not so much overall effect). * Duplicate set-cookie headers were being produced. * Cleanup processed_headers some more. * Linting * Import ordering * Response middleware ran by async request.respond(). * Need to check if transport is closing to avoid getting stuck in sending loops after peer has disconnected. * Middleware and error handling refactoring. * Linter * Fix tracking of HTTP stage when writing to transport fails. * Add clarifying comment * Add a check for request body functions and a test for NotImplementedError. * Linter and typing * These must be tuples + hack mypy warnings away. * New streaming test and minor fixes. * Constant receive buffer size. * 256 KiB send and receive buffers. * Revert "256 KiB send and receive buffers." This reverts commit abc1e3edb21a5e6925fa4c856657559608a8d65b. * app.handle_exception already sends the response. * Improved handling of errors during request. * An odd hack to avoid an httpx limitation that causes test failures. * Limit request header size to 8 KiB at most. * Remove unnecessary use of format string. * Cleanup tests * Remove artifact * Fix type checking * Mark test for skipping * Cleanup some edge cases * Add ignore_body flag to safe methods * Add unit tests for timeout logic * Add unit tests for timeout logic * Fix Mock usage in timeout test * Change logging test to only logger in handler * Windows py3.8 logging issue with current testing client * Add test_header_size_exceeded * Resolve merge conflicts * Add request middleware to hard exception handling * Add request middleware to hard exception handling * Request middleware on exception handlers * Linting * Cleanup deprecations Co-authored-by: L. Kärkkäinen <tronic@users.noreply.github.com> Co-authored-by: Adam Hopkins <admhpkns@gmail.com>
285 lines
9.1 KiB
Python
285 lines
9.1 KiB
Python
from json import JSONDecodeError
|
|
from socket import socket
|
|
|
|
import httpx
|
|
import websockets
|
|
|
|
from sanic.asgi import ASGIApp
|
|
from sanic.exceptions import MethodNotSupported
|
|
from sanic.log import logger
|
|
from sanic.response import text
|
|
|
|
|
|
ASGI_HOST = "mockserver"
|
|
ASGI_PORT = 1234
|
|
ASGI_BASE_URL = f"http://{ASGI_HOST}:{ASGI_PORT}"
|
|
HOST = "127.0.0.1"
|
|
PORT = None
|
|
|
|
|
|
class SanicTestClient:
|
|
def __init__(self, app, port=PORT, host=HOST):
|
|
"""Use port=None to bind to a random port"""
|
|
self.app = app
|
|
self.port = port
|
|
self.host = host
|
|
|
|
@app.listener("after_server_start")
|
|
def _start_test_mode(sanic, *args, **kwargs):
|
|
sanic.test_mode = True
|
|
|
|
@app.listener("before_server_end")
|
|
def _end_test_mode(sanic, *args, **kwargs):
|
|
sanic.test_mode = False
|
|
|
|
def get_new_session(self):
|
|
return httpx.AsyncClient(verify=False)
|
|
|
|
async def _local_request(self, method, url, *args, **kwargs):
|
|
logger.info(url)
|
|
raw_cookies = kwargs.pop("raw_cookies", None)
|
|
|
|
if method == "websocket":
|
|
async with websockets.connect(url, *args, **kwargs) as websocket:
|
|
websocket.opened = websocket.open
|
|
return websocket
|
|
else:
|
|
async with self.get_new_session() as session:
|
|
|
|
try:
|
|
if method == "request":
|
|
args = [url] + list(args)
|
|
url = kwargs.pop("http_method", "GET").upper()
|
|
response = await getattr(session, method.lower())(
|
|
url, *args, **kwargs
|
|
)
|
|
except httpx.HTTPError as e:
|
|
if hasattr(e, "response"):
|
|
response = e.response
|
|
else:
|
|
logger.error(
|
|
f"{method.upper()} {url} received no response!",
|
|
exc_info=True,
|
|
)
|
|
return None
|
|
|
|
response.body = await response.aread()
|
|
response.status = response.status_code
|
|
response.content_type = response.headers.get("content-type")
|
|
|
|
# response can be decoded as json after response._content
|
|
# is set by response.aread()
|
|
try:
|
|
response.json = response.json()
|
|
except (JSONDecodeError, UnicodeDecodeError):
|
|
response.json = None
|
|
|
|
if raw_cookies:
|
|
response.raw_cookies = {}
|
|
|
|
for cookie in response.cookies.jar:
|
|
response.raw_cookies[cookie.name] = cookie
|
|
|
|
return response
|
|
|
|
def _sanic_endpoint_test(
|
|
self,
|
|
method="get",
|
|
uri="/",
|
|
gather_request=True,
|
|
debug=False,
|
|
server_kwargs={"auto_reload": False},
|
|
host=None,
|
|
*request_args,
|
|
**request_kwargs,
|
|
):
|
|
results = [None, None]
|
|
exceptions = []
|
|
if gather_request:
|
|
|
|
def _collect_request(request):
|
|
if results[0] is None:
|
|
results[0] = request
|
|
|
|
self.app.request_middleware.appendleft(_collect_request)
|
|
|
|
@self.app.exception(MethodNotSupported)
|
|
async def error_handler(request, exception):
|
|
if request.method in ["HEAD", "PATCH", "PUT", "DELETE"]:
|
|
return text(
|
|
"", exception.status_code, headers=exception.headers
|
|
)
|
|
else:
|
|
return self.app.error_handler.default(request, exception)
|
|
|
|
if self.port:
|
|
server_kwargs = dict(
|
|
host=host or self.host,
|
|
port=self.port,
|
|
**server_kwargs,
|
|
)
|
|
host, port = host or self.host, self.port
|
|
else:
|
|
sock = socket()
|
|
sock.bind((host or self.host, 0))
|
|
server_kwargs = dict(sock=sock, **server_kwargs)
|
|
host, port = sock.getsockname()
|
|
self.port = port
|
|
|
|
if uri.startswith(
|
|
("http:", "https:", "ftp:", "ftps://", "//", "ws:", "wss:")
|
|
):
|
|
url = uri
|
|
else:
|
|
uri = uri if uri.startswith("/") else f"/{uri}"
|
|
scheme = "ws" if method == "websocket" else "http"
|
|
url = f"{scheme}://{host}:{port}{uri}"
|
|
# Tests construct URLs using PORT = None, which means random port not
|
|
# known until this function is called, so fix that here
|
|
url = url.replace(":None/", f":{port}/")
|
|
|
|
@self.app.listener("after_server_start")
|
|
async def _collect_response(sanic, loop):
|
|
try:
|
|
response = await self._local_request(
|
|
method, url, *request_args, **request_kwargs
|
|
)
|
|
results[-1] = response
|
|
except Exception as e:
|
|
logger.exception("Exception")
|
|
exceptions.append(e)
|
|
self.app.stop()
|
|
|
|
self.app.run(debug=debug, **server_kwargs)
|
|
self.app.listeners["after_server_start"].pop()
|
|
|
|
if exceptions:
|
|
raise ValueError(f"Exception during request: {exceptions}")
|
|
|
|
if gather_request:
|
|
try:
|
|
request, response = results
|
|
return request, response
|
|
except BaseException: # noqa
|
|
raise ValueError(
|
|
f"Request and response object expected, got ({results})"
|
|
)
|
|
else:
|
|
try:
|
|
return results[-1]
|
|
except BaseException: # noqa
|
|
raise ValueError(f"Request object expected, got ({results})")
|
|
|
|
def request(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("request", *args, **kwargs)
|
|
|
|
def get(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("get", *args, **kwargs)
|
|
|
|
def post(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("post", *args, **kwargs)
|
|
|
|
def put(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("put", *args, **kwargs)
|
|
|
|
def delete(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("delete", *args, **kwargs)
|
|
|
|
def patch(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("patch", *args, **kwargs)
|
|
|
|
def options(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("options", *args, **kwargs)
|
|
|
|
def head(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("head", *args, **kwargs)
|
|
|
|
def websocket(self, *args, **kwargs):
|
|
return self._sanic_endpoint_test("websocket", *args, **kwargs)
|
|
|
|
|
|
class TestASGIApp(ASGIApp):
|
|
async def __call__(self):
|
|
await super().__call__()
|
|
return self.request
|
|
|
|
|
|
async def app_call_with_return(self, scope, receive, send):
|
|
asgi_app = await TestASGIApp.create(self, scope, receive, send)
|
|
return await asgi_app()
|
|
|
|
|
|
class SanicASGITestClient(httpx.AsyncClient):
|
|
def __init__(
|
|
self,
|
|
app,
|
|
base_url: str = ASGI_BASE_URL,
|
|
suppress_exceptions: bool = False,
|
|
) -> None:
|
|
app.__class__.__call__ = app_call_with_return
|
|
app.asgi = True
|
|
|
|
self.app = app
|
|
transport = httpx.ASGITransport(app=app, client=(ASGI_HOST, ASGI_PORT))
|
|
super().__init__(transport=transport, base_url=base_url)
|
|
|
|
self.last_request = None
|
|
|
|
def _collect_request(request):
|
|
self.last_request = request
|
|
|
|
@app.listener("after_server_start")
|
|
def _start_test_mode(sanic, *args, **kwargs):
|
|
sanic.test_mode = True
|
|
|
|
@app.listener("before_server_end")
|
|
def _end_test_mode(sanic, *args, **kwargs):
|
|
sanic.test_mode = False
|
|
|
|
app.request_middleware.appendleft(_collect_request)
|
|
|
|
async def request(self, method, url, gather_request=True, *args, **kwargs):
|
|
|
|
self.gather_request = gather_request
|
|
response = await super().request(method, url, *args, **kwargs)
|
|
response.status = response.status_code
|
|
response.body = response.content
|
|
response.content_type = response.headers.get("content-type")
|
|
|
|
return self.last_request, response
|
|
|
|
async def websocket(self, uri, subprotocols=None, *args, **kwargs):
|
|
scheme = "ws"
|
|
path = uri
|
|
root_path = f"{scheme}://{ASGI_HOST}"
|
|
|
|
headers = kwargs.get("headers", {})
|
|
headers.setdefault("connection", "upgrade")
|
|
headers.setdefault("sec-websocket-key", "testserver==")
|
|
headers.setdefault("sec-websocket-version", "13")
|
|
if subprotocols is not None:
|
|
headers.setdefault(
|
|
"sec-websocket-protocol", ", ".join(subprotocols)
|
|
)
|
|
|
|
scope = {
|
|
"type": "websocket",
|
|
"asgi": {"version": "3.0"},
|
|
"http_version": "1.1",
|
|
"headers": [map(lambda y: y.encode(), x) for x in headers.items()],
|
|
"scheme": scheme,
|
|
"root_path": root_path,
|
|
"path": path,
|
|
"query_string": b"",
|
|
}
|
|
|
|
async def receive():
|
|
return {}
|
|
|
|
async def send(message):
|
|
pass
|
|
|
|
await self.app(scope, receive, send)
|
|
|
|
return None, {}
|