from brotli import compress from flask import request, make_response from functools import wraps from threading import Thread from hashlib import sha256 import redis import os.path import mimetypes class BrotliCache(object): def __init__(self, redis_kwargs={}): self.redis = redis.StrictRedis(**redis_kwargs) def compress(self, cache_key, lock_key, body): encbody = compress(body) self.redis.set(cache_key, encbody, ex=3600) self.redis.delete(lock_key) def wrap_response(self, response): if 'br' not in request.accept_encodings or response.is_streamed: return response body = response.get_data() digest = sha256(body).hexdigest() cache_key = 'brotlicache:{}'.format(digest) encbody = self.redis.get(cache_key) if encbody: response.headers.set('content-encoding', 'br') response.headers.set('vary', 'accept-encoding') response.set_data(encbody) return response else: lock_key = 'brotlicache:lock:{}'.format(digest) if self.redis.set(lock_key, 1, nx=True, ex=10): t = Thread(target=self.compress, args=(cache_key, lock_key, body)) t.start() return response def brotli(app, static = True, dynamic = True): original_static = app.view_functions['static'] def static_maybe_gzip_brotli(filename=None): path = os.path.join(app.static_folder, filename) for encoding, extension in (('br', '.br'), ('gzip', '.gz')): if encoding not in request.accept_encodings: continue encpath = path + extension if os.path.isfile(encpath): resp = make_response(original_static(filename=filename + extension)) resp.headers.set('content-encoding', encoding) resp.headers.set('vary', 'accept-encoding') mimetype = mimetypes.guess_type(filename)[0] or 'application/octet-stream' resp.headers.set('content-type', mimetype) return resp return original_static(filename=filename) if static: app.view_functions['static'] = static_maybe_gzip_brotli if dynamic: cache = BrotliCache() app.after_request(cache.wrap_response)