Skip to content

Commit

Permalink
Enable brotli decompression if it is available
Browse files Browse the repository at this point in the history
  • Loading branch information
immerrr committed Nov 12, 2021
1 parent 8c0bb73 commit c7b3b93
Show file tree
Hide file tree
Showing 3 changed files with 45 additions and 7 deletions.
17 changes: 17 additions & 0 deletions tests/integration/test_filter.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
from urllib.parse import urlencode
from urllib.error import HTTPError
import vcr
from vcr.filters import brotli
import json
from assertions import assert_cassette_has_one_response, assert_is_json

Expand Down Expand Up @@ -118,6 +119,22 @@ def test_decompress_deflate(tmpdir, httpbin):
assert_is_json(decoded_response)


def test_decompress_brotli(tmpdir, httpbin):
if brotli is None:
# XXX: this is never true, because brotlipy is installed with "httpbin"
pytest.skip('Brotli is not installed')

url = httpbin.url + "/brotli"
request = Request(url, headers={"Accept-Encoding": ["gzip, deflate, br"]})
cass_file = str(tmpdir.join("brotli_response.yaml"))
with vcr.use_cassette(cass_file, decode_compressed_response=True):
urlopen(request)
with vcr.use_cassette(cass_file) as cass:
decoded_response = urlopen(url).read()
assert_cassette_has_one_response(cass)
assert_is_json(decoded_response)


def test_decompress_regular(tmpdir, httpbin):
"""Test that it doesn't try to decompress content that isn't compressed"""
url = httpbin.url + "/get"
Expand Down
8 changes: 6 additions & 2 deletions tox.ini
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@ skip_missing_interpreters=true
envlist =
cov-clean,
lint,
{py36,py37,py38,py39,py310}-{requests,httplib2,urllib3,tornado4,boto3,aiohttp,httpx},
{pypy3}-{requests,httplib2,urllib3,tornado4,boto3},
{py36,py37,py38,py39,py310}-{requests,httplib2,urllib3,tornado4,boto3,aiohttp,httpx,brotli,brotlipy,brotlicffi},
{pypy3}-{requests,httplib2,urllib3,tornado4,boto3,brotli,brotlipy,brotlicffi},
cov-report


Expand Down Expand Up @@ -91,6 +91,10 @@ deps =
httpx: httpx
{py36,py37,py38,py39,py310}-{httpx}: httpx
{py36,py37,py38,py39,py310}-{httpx}: pytest-asyncio
{py36,py37,py38,py39,py310}-{httpx}: httpx
brotli: brotli
brotlipy: brotlipy
brotlicffi: brotlicffi
depends =
lint,{py36,py37,py38,py39,py310,pypy3}-{requests,httplib2,urllib3,tornado4,boto3},{py36,py37,py38,py39,py310}-{aiohttp},{py36,py37,py38,py39,py310}-{httpx}: cov-clean
cov-report: lint,{py36,py37,py38,py39,py310,pypy3}-{requests,httplib2,urllib3,tornado4,boto3},{py36,py37,py38,py39,py310}-{aiohttp}
Expand Down
27 changes: 22 additions & 5 deletions vcr/filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,20 @@

from .util import CaseInsensitiveDict

try:
# This supports both brotli & brotlipy packages
import brotli
except ImportError:
try:
import brotlicffi as brotli
except ImportError:
brotli = None


AVAILABLE_DECOMPRESSORS = {'gzip', 'deflate'}
if brotli is not None:
AVAILABLE_DECOMPRESSORS.add('br')


def replace_headers(request, replacements):
"""Replace headers in request according to replacements.
Expand Down Expand Up @@ -136,30 +150,33 @@ def remove_post_data_parameters(request, post_data_parameters_to_remove):

def decode_response(response):
"""
If the response is compressed with gzip or deflate:
If the response is compressed with any supported compression (gzip,
deflate, br if available):
1. decompress the response body
2. delete the content-encoding header
3. update content-length header to decompressed length
"""

def is_compressed(headers):
def is_decompressable(headers):
encoding = headers.get("content-encoding", [])
return encoding and encoding[0] in ("gzip", "deflate")
return encoding and encoding[0] in AVAILABLE_DECOMPRESSORS

def decompress_body(body, encoding):
"""Returns decompressed body according to encoding using zlib.
to (de-)compress gzip format, use wbits = zlib.MAX_WBITS | 16
"""
if encoding == "gzip":
return zlib.decompress(body, zlib.MAX_WBITS | 16)
else: # encoding == 'deflate'
elif encoding == "deflate":
return zlib.decompress(body)
else: # encoding == 'br'
return brotli.decompress(body)

# Deepcopy here in case `headers` contain objects that could
# be mutated by a shallow copy and corrupt the real response.
response = copy.deepcopy(response)
headers = CaseInsensitiveDict(response["headers"])
if is_compressed(headers):
if is_decompressable(headers):
encoding = headers["content-encoding"][0]
headers["content-encoding"].remove(encoding)
if not headers["content-encoding"]:
Expand Down

0 comments on commit c7b3b93

Please sign in to comment.