Skip to content

Commit

Permalink
Fix double compress when compression enabled and compressed file exists
Browse files Browse the repository at this point in the history
If the handler has `.enable_compression` set and we use the pre-compressed
.gz we would compress twice. The caller may not know if the `.gz` file
exists or not or have a mixed use case where some files are
pre-compressed and some are compressed on the fly based if they have
limited storage

fixes aio-libs#8011
  • Loading branch information
bdraco committed Jan 9, 2024
1 parent fd7dfe5 commit 86fe1f3
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 1 deletion.
4 changes: 4 additions & 0 deletions aiohttp/web_fileresponse.py
Original file line number Diff line number Diff line change
Expand Up @@ -267,6 +267,10 @@ async def prepare(self, request: "BaseRequest") -> Optional[AbstractStreamWriter
self.headers[hdrs.CONTENT_ENCODING] = encoding
if gzip:
self.headers[hdrs.VARY] = hdrs.ACCEPT_ENCODING
# Disable compression if we are already sending
# a compressed file since we don't want to double
# compress.
self._compression = False

self.etag = etag_value # type: ignore[assignment]
self.last_modified = st.st_mtime # type: ignore[assignment]
Expand Down
30 changes: 29 additions & 1 deletion tests/test_web_sendfile_functional.py
Original file line number Diff line number Diff line change
Expand Up @@ -250,7 +250,35 @@ async def handler(request):
await client.close()


async def test_static_file_with_content_encoding(aiohttp_client, sender) -> None:
async def test_static_file_with_gziped_counter_part_enable_compression(
aiohttp_client: Any, sender: Any
):
"""Test that enable_compression does not double compress when a .gz file is also present."""
filepath = pathlib.Path(__file__).parent / "hello.txt"

async def handler(request):
resp = sender(filepath)
resp.enable_compression()
return resp

app = web.Application()
app.router.add_get("/", handler)
client = await aiohttp_client(app)

resp = await client.get("/")
assert resp.status == 200
body = await resp.read()
assert body == b"hello aiohttp\n"
assert resp.headers["Content-Type"] == "text/plain"
assert resp.headers.get("Content-Encoding") == "gzip"
resp.close()
await resp.release()
await client.close()


async def test_static_file_with_content_encoding(
aiohttp_client: Any, sender: Any
) -> None:
filepath = pathlib.Path(__file__).parent / "hello.txt.gz"

async def handler(request):
Expand Down

0 comments on commit 86fe1f3

Please sign in to comment.