diff --git a/sanic/asgi.py b/sanic/asgi.py index 27285c353a..ad2f9d2d42 100644 --- a/sanic/asgi.py +++ b/sanic/asgi.py @@ -3,7 +3,6 @@ import warnings from typing import TYPE_CHECKING, Optional -from urllib.parse import quote from sanic.compat import Header from sanic.exceptions import BadRequest, ServerError @@ -146,14 +145,6 @@ async def create( raise BadRequest( "Header names can only contain US-ASCII characters" ) - path = ( - scope["path"][1:] - if scope["path"].startswith("/") - else scope["path"] - ) - url = "/".join([scope.get("root_path", ""), quote(path)]) - url_bytes = url.encode("latin-1") - url_bytes += b"?" + scope["query_string"] if scope["type"] == "http": version = scope["http_version"] @@ -168,6 +159,13 @@ async def create( else: raise ServerError("Received unknown ASGI scope") + url_bytes, query = scope["raw_path"], scope["query_string"] + if query: + # httpx ASGI client sends query string as part of raw_path + url_bytes = url_bytes.split(b"?", 1)[0] + # All servers send them separately + url_bytes = b"%b?%b" % (url_bytes, query) + request_class = sanic_app.request_class or Request instance.request = request_class( url_bytes, diff --git a/sanic/http/http1.py b/sanic/http/http1.py index 5884d0b4c3..bb317a92ec 100644 --- a/sanic/http/http1.py +++ b/sanic/http/http1.py @@ -240,9 +240,14 @@ async def http1_request_header(self): # no cov headers_instance.getone("upgrade", "").lower() == "websocket" ) + try: + url_bytes = self.url.encode("ASCII") + except UnicodeEncodeError: + raise BadRequest("URL may only contain US-ASCII characters.") + # Prepare a Request object request = self.protocol.request_class( - url_bytes=self.url.encode(), + url_bytes=url_bytes, headers=headers_instance, head=bytes(head), version=protocol[5:], @@ -445,9 +450,18 @@ def create_empty_request(self) -> None: bogus response for error handling use. """ + # Reformat any URL already received with \xHH escapes for better logs + url_bytes = ( + self.url.encode(errors="surrogateescape") + .decode("ASCII", errors="backslashreplace") + .encode("ASCII") + if self.url + else b"*" + ) + # FIXME: Avoid this by refactoring error handling and response code self.request = self.protocol.request_class( - url_bytes=self.url.encode() if self.url else b"*", + url_bytes=url_bytes, headers=Header({}), version="1.1", method="NONE", diff --git a/sanic/http/http3.py b/sanic/http/http3.py index 13884c5611..91f343fe35 100644 --- a/sanic/http/http3.py +++ b/sanic/http/http3.py @@ -18,7 +18,12 @@ from sanic.compat import Header from sanic.constants import LocalCertCreator -from sanic.exceptions import PayloadTooLarge, SanicException, ServerError +from sanic.exceptions import ( + BadRequest, + PayloadTooLarge, + SanicException, + ServerError, +) from sanic.helpers import has_message_body from sanic.http.constants import Stage from sanic.http.stream import Stream @@ -333,7 +338,17 @@ def get_receiver_by_stream_id(self, stream_id: int) -> Receiver: return self.receivers[stream_id] def _make_request(self, event: HeadersReceived) -> Request: - headers = Header(((k.decode(), v.decode()) for k, v in event.headers)) + try: + headers = Header( + ( + (k.decode("ASCII"), v.decode(errors="surrogateescape")) + for k, v in event.headers + ) + ) + except UnicodeDecodeError: + raise BadRequest( + "Header names may only contain US-ASCII characters." + ) method = headers[":method"] path = headers[":path"] scheme = headers.pop(":scheme", "") @@ -342,9 +357,14 @@ def _make_request(self, event: HeadersReceived) -> Request: if authority: headers["host"] = authority + try: + url_bytes = path.encode("ASCII") + except UnicodeEncodeError: + raise BadRequest("URL may only contain US-ASCII characters.") + transport = HTTP3Transport(self.protocol) request = self.protocol.request_class( - path.encode(), + url_bytes, headers, "3", method, diff --git a/sanic/request/types.py b/sanic/request/types.py index 97ed7ff8bd..106ba5102e 100644 --- a/sanic/request/types.py +++ b/sanic/request/types.py @@ -133,7 +133,8 @@ def __init__( try: self._parsed_url = parse_url(url_bytes) except HttpParserInvalidURLError: - raise BadURL(f"Bad URL: {url_bytes.decode()}") + url = url_bytes.decode(errors="backslashreplace") + raise BadURL(f"Bad URL: {url}") self._id: Optional[Union[uuid.UUID, str, int]] = None self._name: Optional[str] = None self._stream_id = stream_id diff --git a/setup.py b/setup.py index de14206828..82e351fc92 100644 --- a/setup.py +++ b/setup.py @@ -116,7 +116,7 @@ def str_to_bool(val: str) -> bool: ] tests_require = [ - "sanic-testing>=22.9.0", + "sanic-testing@git+https://github.com/sanic-org/sanic-testing.git@main#egg=sanic-testing>=22.12.0", "pytest==7.1.*", "coverage", "beautifulsoup4", diff --git a/tests/http3/test_http_receiver.py b/tests/http3/test_http_receiver.py index 2784e87b30..d996c4843e 100644 --- a/tests/http3/test_http_receiver.py +++ b/tests/http3/test_http_receiver.py @@ -11,7 +11,7 @@ from sanic import Request, Sanic from sanic.compat import Header from sanic.config import DEFAULT_CONFIG -from sanic.exceptions import PayloadTooLarge +from sanic.exceptions import BadRequest, PayloadTooLarge from sanic.http.constants import Stage from sanic.http.http3 import Http3, HTTPReceiver from sanic.models.server_types import ConnInfo @@ -292,3 +292,48 @@ def test_request_conn_info(app): receiver = http3.get_receiver_by_stream_id(1) assert isinstance(receiver.request.conn_info, ConnInfo) + + +def test_request_header_encoding(app): + protocol = generate_protocol(app) + http3 = Http3(protocol, protocol.transmit) + with pytest.raises(BadRequest) as exc_info: + http3.http_event_received( + HeadersReceived( + [ + (b":method", b"GET"), + (b":path", b"/location"), + (b":scheme", b"https"), + (b":authority", b"localhost:8443"), + ("foo\u00A0".encode(), b"bar"), + ], + 1, + False, + ) + ) + assert exc_info.value.status_code == 400 + assert ( + str(exc_info.value) + == "Header names may only contain US-ASCII characters." + ) + + +def test_request_url_encoding(app): + protocol = generate_protocol(app) + http3 = Http3(protocol, protocol.transmit) + with pytest.raises(BadRequest) as exc_info: + http3.http_event_received( + HeadersReceived( + [ + (b":method", b"GET"), + (b":path", b"/location\xA0"), + (b":scheme", b"https"), + (b":authority", b"localhost:8443"), + (b"foo", b"bar"), + ], + 1, + False, + ) + ) + assert exc_info.value.status_code == 400 + assert str(exc_info.value) == "URL may only contain US-ASCII characters." diff --git a/tests/test_asgi.py b/tests/test_asgi.py index af44ead38f..8fce84af3c 100644 --- a/tests/test_asgi.py +++ b/tests/test_asgi.py @@ -652,3 +652,17 @@ def mocked_headers_init(self, *args, **kwargs): _, response = await app.asgi_client.get("/", headers={"Test-Header": "😅"}) assert response.status_code == 200 + + +@pytest.mark.asyncio +async def test_asgi_url_decoding(app): + @app.get("/dir/", unquote=True) + def _request(request: Request, name): + return text(name) + + # 2F should not become a path separator (unquoted later) + _, response = await app.asgi_client.get("/dir/some%2Fpath") + assert response.text == "some/path" + + _, response = await app.asgi_client.get("/dir/some%F0%9F%98%80path") + assert response.text == "some😀path" diff --git a/tests/test_http.py b/tests/test_http.py index f4e11e0ee8..ca2e86c03e 100644 --- a/tests/test_http.py +++ b/tests/test_http.py @@ -98,3 +98,17 @@ def test_transfer_chunked(client): data = stdjson.loads(body) assert data == ["foo", "bar"] + + +def test_url_encoding(client): + client.send( + """ + GET /invalid\xA0url HTTP/1.1 + + """ + ) + response = client.recv() + headers, body = response.rsplit(b"\r\n\r\n", 1) + + assert b"400 Bad Request" in headers + assert b"URL may only contain US-ASCII characters." in body