Skip to content

Commit

Permalink
Merge pull request #1965 from ashleysommer/asgs_chunk_length
Browse files Browse the repository at this point in the history
Fix Chunked Transport-Encoding in ASGI streaming response
  • Loading branch information
ahopkins authored Nov 5, 2020
2 parents 5961da3 + 75994cd commit 33da077
Show file tree
Hide file tree
Showing 4 changed files with 27 additions and 7 deletions.
2 changes: 2 additions & 0 deletions sanic/app.py
Original file line number Diff line number Diff line change
Expand Up @@ -1454,6 +1454,8 @@ async def __call__(self, scope, receive, send):
asgi_app = await ASGIApp.create(self, scope, receive, send)
await asgi_app()

_asgi_single_callable = True # We conform to ASGI 3.0 single-callable

# -------------------------------------------------------------------- #
# Configuration
# -------------------------------------------------------------------- #
Expand Down
28 changes: 22 additions & 6 deletions sanic/asgi.py
Original file line number Diff line number Diff line change
Expand Up @@ -312,13 +312,19 @@ async def __call__(self) -> None:
callback = None if self.ws else self.stream_callback
await handler(self.request, None, callback)

async def stream_callback(self, response: HTTPResponse) -> None:
_asgi_single_callable = True # We conform to ASGI 3.0 single-callable

async def stream_callback(
self, response: Union[HTTPResponse, StreamingHTTPResponse]
) -> None:
"""
Write the response.
"""
headers: List[Tuple[bytes, bytes]] = []
cookies: Dict[str, str] = {}
content_length: List[str] = []
try:
content_length = response.headers.popall("content-length", [])
cookies = {
v.key: v
for _, v in list(
Expand Down Expand Up @@ -351,12 +357,22 @@ async def stream_callback(self, response: HTTPResponse) -> None:
]

response.asgi = True

if "content-length" not in response.headers and not isinstance(
response, StreamingHTTPResponse
):
is_streaming = isinstance(response, StreamingHTTPResponse)
if is_streaming and getattr(response, "chunked", False):
# disable sanic chunking, this is done at the ASGI-server level
setattr(response, "chunked", False)
# content-length header is removed to signal to the ASGI-server
# to use automatic-chunking if it supports it
elif len(content_length) > 0:
headers += [
(b"content-length", str(len(response.body)).encode("latin-1"))
(b"content-length", str(content_length[0]).encode("latin-1"))
]
elif not is_streaming:
headers += [
(
b"content-length",
str(len(getattr(response, "body", b""))).encode("latin-1"),
)
]

if "content-type" not in response.headers:
Expand Down
2 changes: 2 additions & 0 deletions sanic/response.py
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,8 @@ async def write(self, data):
"""
data = self._encode_body(data)

# `chunked` will always be False in ASGI-mode, even if the underlying
# ASGI Transport implements Chunked transport. That does it itself.
if self.chunked:
await self.protocol.push_data(b"%x\r\n%b\r\n" % (len(data), data))
else:
Expand Down
2 changes: 1 addition & 1 deletion tests/test_response.py
Original file line number Diff line number Diff line change
Expand Up @@ -238,7 +238,7 @@ def test_chunked_streaming_returns_correct_content(streaming_app):
@pytest.mark.asyncio
async def test_chunked_streaming_returns_correct_content_asgi(streaming_app):
request, response = await streaming_app.asgi_client.get("/")
assert response.text == "4\r\nfoo,\r\n3\r\nbar\r\n0\r\n\r\n"
assert response.text == "foo,bar"


def test_non_chunked_streaming_adds_correct_headers(non_chunked_streaming_app):
Expand Down

0 comments on commit 33da077

Please sign in to comment.