Skip to content
This repository was archived by the owner on Jan 13, 2021. It is now read-only.

Commit 4b522f5

Browse files
authored
Merge pull request #368 from viranch/brotli
Add support for brotli compression
2 parents 98d9e5f + 1026313 commit 4b522f5

File tree

5 files changed

+27
-1
lines changed

5 files changed

+27
-1
lines changed

hyper/http11/response.py

+3
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
import logging
1010
import weakref
1111
import zlib
12+
import brotli
1213

1314
from ..common.decoder import DeflateDecoder
1415
from ..common.exceptions import ChunkedDecodeError, InvalidResponseError
@@ -88,6 +89,8 @@ def __init__(self, code, reason, headers, sock, connection=None,
8889
# http://stackoverflow.com/a/2695466/1401686
8990
if b'gzip' in self.headers.get(b'content-encoding', []):
9091
self._decompressobj = zlib.decompressobj(16 + zlib.MAX_WBITS)
92+
elif b'br' in self.headers.get(b'content-encoding', []):
93+
self._decompressobj = brotli.Decompressor()
9194
elif b'deflate' in self.headers.get(b'content-encoding', []):
9295
self._decompressobj = DeflateDecoder()
9396
else:

hyper/http20/response.py

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
"""
99
import logging
1010
import zlib
11+
import brotli
1112

1213
from ..common.decoder import DeflateDecoder
1314
from ..common.headers import HTTPHeaderMap
@@ -31,6 +32,7 @@ def strip_headers(headers):
3132

3233
decompressors = {
3334
b'gzip': lambda: zlib.decompressobj(16 + zlib.MAX_WBITS),
35+
b'br': brotli.Decompressor,
3436
b'deflate': DeflateDecoder
3537
}
3638

setup.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ def run_tests(self):
7777
'Programming Language :: Python :: Implementation :: CPython',
7878
],
7979
install_requires=[
80-
'h2>=2.4,<3.0,!=2.5.0', 'hyperframe>=3.2,<4.0', 'rfc3986>=1.1.0,<2.0'
80+
'h2>=2.4,<3.0,!=2.5.0', 'hyperframe>=3.2,<4.0', 'rfc3986>=1.1.0,<2.0', 'brotlipy>=0.7.0,<1.0'
8181
],
8282
tests_require=['pytest', 'requests', 'mock'],
8383
cmdclass={'test': PyTest},

test/test_http11.py

+11
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
"""
88
import os
99
import zlib
10+
import brotli
1011

1112
from collections import namedtuple
1213
from io import BytesIO, StringIO
@@ -644,6 +645,16 @@ def test_response_transparently_decrypts_gzip(self):
644645

645646
assert r.read() == b'this is test data'
646647

648+
def test_response_transparently_decrypts_brotli(self):
649+
d = DummySocket()
650+
headers = {b'content-encoding': [b'br'], b'connection': [b'close']}
651+
r = HTTP11Response(200, 'OK', headers, d, None)
652+
653+
body = brotli.compress(b'this is test data')
654+
d._buffer = BytesIO(body)
655+
656+
assert r.read() == b'this is test data'
657+
647658
def test_response_transparently_decrypts_real_deflate(self):
648659
d = DummySocket()
649660
headers = {

test/test_hyper.py

+10
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@
2626
import pytest
2727
import socket
2828
import zlib
29+
import brotli
2930
from io import BytesIO
3031

3132
TEST_DIR = os.path.abspath(os.path.dirname(__file__))
@@ -1088,6 +1089,15 @@ def test_response_transparently_decrypts_gzip(self):
10881089

10891090
assert resp.read() == b'this is test data'
10901091

1092+
def test_response_transparently_decrypts_brotli(self):
1093+
headers = HTTPHeaderMap(
1094+
[(':status', '200'), ('content-encoding', 'br')]
1095+
)
1096+
body = brotli.compress(b'this is test data')
1097+
resp = HTTP20Response(headers, DummyStream(body))
1098+
1099+
assert resp.read() == b'this is test data'
1100+
10911101
def test_response_transparently_decrypts_real_deflate(self):
10921102
headers = HTTPHeaderMap(
10931103
[(':status', '200'), ('content-encoding', 'deflate')]

0 commit comments

Comments
 (0)