Skip to content

Commit d272132

Browse files
bjohansebasdanielgindinicksrandallUlisesGasconwesleytodd
authored
feat: support for brotli (#194)
* Added support for brotli ('br') content-encoding * Update README.md * Update README.md * Update README.md * Apply default value also when params is specified * Increase coverage for specifying params * Updated brotli detection method * Prefer br over gzip and deflate * feat: use "koa-compress" logic to determine the preferred encoding * test: adding one more test case br/gzip with quality params * chore: fix linting errors * fix: hand write encodings lib to be compatible with node 0.8 * Fix: fixing lint errors in new lib * Fix: fixing lint errors in new lib * implemented required encoding negotiator without 3rd party dependency * fix * use negotiator * improve negotiateEnconding * fix support * update history * add new test * add new test * Update test/compression.js Co-authored-by: Wes Todd <[email protected]> * improve parse options * don't directly manipulate the object. * remove .npmrc * use object assign in params * test: add test for enforceEnconding * deps: remove object-assign --------- Co-authored-by: Daniel Cohen Gindi <[email protected]> Co-authored-by: Nick Randall <[email protected]> Co-authored-by: Ulises Gascón <[email protected]> Co-authored-by: Wes Todd <[email protected]>
1 parent 96df7c5 commit d272132

File tree

4 files changed

+229
-5
lines changed

4 files changed

+229
-5
lines changed

HISTORY.md

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
unreleased
22
==========
33
* Use `res.headersSent` when available
4+
* add brotli support for versions of node that support it
45
* Add the enforceEncoding option for requests without `Accept-Encoding` header
56

67
1.7.5 / 2024-10-31

README.md

+10-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@ The following compression codings are supported:
1111

1212
- deflate
1313
- gzip
14+
- br (brotli)
15+
16+
**Note** Brotli is supported only since Node.js versions v11.7.0 and v10.16.0.
1417

1518
## Install
1619

@@ -42,7 +45,8 @@ as compressing will transform the body.
4245

4346
`compression()` accepts these properties in the options object. In addition to
4447
those listed below, [zlib](http://nodejs.org/api/zlib.html) options may be
45-
passed in to the options object.
48+
passed in to the options object or
49+
[brotli](https://nodejs.org/api/zlib.html#zlib_class_brotlioptions) options.
4650

4751
##### chunkSize
4852

@@ -99,6 +103,11 @@ The default value is `zlib.Z_DEFAULT_MEMLEVEL`, or `8`.
99103
See [Node.js documentation](http://nodejs.org/api/zlib.html#zlib_memory_usage_tuning)
100104
regarding the usage.
101105

106+
##### brotli
107+
108+
This specifies the options for configuring Brotli. See [Node.js documentation](https://nodejs.org/api/zlib.html#class-brotlioptions) for a complete list of available options.
109+
110+
102111
##### strategy
103112

104113
This is used to tune the compression algorithm. This value only affects the

index.js

+24-4
Original file line numberDiff line numberDiff line change
@@ -30,14 +30,21 @@ var zlib = require('zlib')
3030
module.exports = compression
3131
module.exports.filter = shouldCompress
3232

33+
/**
34+
* @const
35+
* whether current node version has brotli support
36+
*/
37+
var hasBrotliSupport = 'createBrotliCompress' in zlib
38+
3339
/**
3440
* Module variables.
3541
* @private
3642
*/
37-
3843
var cacheControlNoTransformRegExp = /(?:^|,)\s*?no-transform\s*?(?:,|$)/
44+
var SUPPORTED_ENCODING = hasBrotliSupport ? ['br', 'gzip', 'deflate', 'identity'] : ['gzip', 'deflate', 'identity']
45+
var PREFERRED_ENCODING = hasBrotliSupport ? ['br', 'gzip'] : ['gzip']
3946

40-
var encodingSupported = ['*', 'gzip', 'deflate', 'identity']
47+
var encodingSupported = ['*', 'gzip', 'deflate', 'identity', 'br']
4148

4249
/**
4350
* Compress response data with gzip / deflate.
@@ -49,6 +56,17 @@ var encodingSupported = ['*', 'gzip', 'deflate', 'identity']
4956

5057
function compression (options) {
5158
var opts = options || {}
59+
var optsBrotli = {}
60+
61+
if (hasBrotliSupport) {
62+
Object.assign(optsBrotli, opts.brotli)
63+
64+
var brotliParams = {}
65+
brotliParams[zlib.constants.BROTLI_PARAM_QUALITY] = 4
66+
67+
// set the default level to a reasonable value with balanced speed/ratio
68+
optsBrotli.params = Object.assign(brotliParams, optsBrotli.params)
69+
}
5270

5371
// options
5472
var filter = opts.filter || shouldCompress
@@ -178,7 +196,7 @@ function compression (options) {
178196

179197
// compression method
180198
var negotiator = new Negotiator(req)
181-
var method = negotiator.encoding(['gzip', 'deflate', 'identity'], ['gzip'])
199+
var method = negotiator.encoding(SUPPORTED_ENCODING, PREFERRED_ENCODING)
182200

183201
// if no method is found, use the default encoding
184202
if (!req.headers['accept-encoding'] && encodingSupported.indexOf(enforceEncoding) !== -1) {
@@ -195,7 +213,9 @@ function compression (options) {
195213
debug('%s compression', method)
196214
stream = method === 'gzip'
197215
? zlib.createGzip(opts)
198-
: zlib.createDeflate(opts)
216+
: method === 'br'
217+
? zlib.createBrotliCompress(optsBrotli)
218+
: zlib.createDeflate(opts)
199219

200220
// add buffered listeners to stream
201221
addListeners(stream, stream.on, listeners)

test/compression.js

+194
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,13 @@ try {
1919

2020
var compression = require('..')
2121

22+
/**
23+
* @const
24+
* whether current node version has brotli support
25+
*/
26+
var hasBrotliSupport = 'createBrotliCompress' in zlib
27+
var brotli = hasBrotliSupport ? it : it.skip
28+
2229
describe('compression()', function () {
2330
it('should skip HEAD', function (done) {
2431
var server = createServer({ threshold: 0 }, function (req, res) {
@@ -510,6 +517,52 @@ describe('compression()', function () {
510517
})
511518
})
512519

520+
describe('when "Accept-Encoding: br"', function () {
521+
brotli('should respond with br', function (done) {
522+
var server = createServer({ threshold: 0 }, function (req, res) {
523+
res.setHeader('Content-Type', 'text/plain')
524+
res.end('hello, world')
525+
})
526+
527+
request(server)
528+
.get('/')
529+
.set('Accept-Encoding', 'br')
530+
.expect('Content-Encoding', 'br', done)
531+
})
532+
})
533+
534+
describe('when "Accept-Encoding: br" and passing compression level', function () {
535+
brotli('should respond with br', function (done) {
536+
var params = {}
537+
params[zlib.constants.BROTLI_PARAM_QUALITY] = 11
538+
539+
var server = createServer({ threshold: 0, brotli: { params: params } }, function (req, res) {
540+
res.setHeader('Content-Type', 'text/plain')
541+
res.end('hello, world')
542+
})
543+
544+
request(server)
545+
.get('/')
546+
.set('Accept-Encoding', 'br')
547+
.expect('Content-Encoding', 'br', done)
548+
})
549+
550+
brotli('shouldn\'t break compression when gzip is requested', function (done) {
551+
var params = {}
552+
params[zlib.constants.BROTLI_PARAM_QUALITY] = 8
553+
554+
var server = createServer({ threshold: 0, brotli: { params: params } }, function (req, res) {
555+
res.setHeader('Content-Type', 'text/plain')
556+
res.end('hello, world')
557+
})
558+
559+
request(server)
560+
.get('/')
561+
.set('Accept-Encoding', 'gzip')
562+
.expect('Content-Encoding', 'gzip', done)
563+
})
564+
})
565+
513566
describe('when "Accept-Encoding: gzip, deflate"', function () {
514567
it('should respond with gzip', function (done) {
515568
var server = createServer({ threshold: 0 }, function (req, res) {
@@ -538,6 +591,105 @@ describe('compression()', function () {
538591
})
539592
})
540593

594+
describe('when "Accept-Encoding: gzip, br"', function () {
595+
var brotli = hasBrotliSupport ? it : it.skip
596+
brotli('should respond with br', function (done) {
597+
var server = createServer({ threshold: 0 }, function (req, res) {
598+
res.setHeader('Content-Type', 'text/plain')
599+
res.end('hello, world')
600+
})
601+
602+
request(server)
603+
.get('/')
604+
.set('Accept-Encoding', 'gzip, br')
605+
.expect('Content-Encoding', 'br', done)
606+
})
607+
608+
brotli = hasBrotliSupport ? it.skip : it
609+
610+
brotli('should respond with gzip', function (done) {
611+
var server = createServer({ threshold: 0 }, function (req, res) {
612+
res.setHeader('Content-Type', 'text/plain')
613+
res.end('hello, world')
614+
})
615+
616+
request(server)
617+
.get('/')
618+
.set('Accept-Encoding', 'br, gzip')
619+
.expect('Content-Encoding', 'gzip', done)
620+
})
621+
})
622+
623+
describe('when "Accept-Encoding: deflate, gzip, br"', function () {
624+
brotli('should respond with br', function (done) {
625+
var server = createServer({ threshold: 0 }, function (req, res) {
626+
res.setHeader('Content-Type', 'text/plain')
627+
res.end('hello, world')
628+
})
629+
630+
request(server)
631+
.get('/')
632+
.set('Accept-Encoding', 'deflate, gzip, br')
633+
.expect('Content-Encoding', 'br', done)
634+
})
635+
})
636+
637+
describe('when "Accept-Encoding: gzip;q=1, br;q=0.3"', function () {
638+
brotli('should respond with gzip', function (done) {
639+
var server = createServer({ threshold: 0 }, function (req, res) {
640+
res.setHeader('Content-Type', 'text/plain')
641+
res.end('hello, world')
642+
})
643+
644+
request(server)
645+
.get('/')
646+
.set('Accept-Encoding', 'gzip;q=1, br;q=0.3')
647+
.expect('Content-Encoding', 'gzip', done)
648+
})
649+
})
650+
651+
describe('when "Accept-Encoding: gzip, br;q=0.8"', function () {
652+
brotli('should respond with gzip', function (done) {
653+
var server = createServer({ threshold: 0 }, function (req, res) {
654+
res.setHeader('Content-Type', 'text/plain')
655+
res.end('hello, world')
656+
})
657+
658+
request(server)
659+
.get('/')
660+
.set('Accept-Encoding', 'gzip, br;q=0.8')
661+
.expect('Content-Encoding', 'gzip', done)
662+
})
663+
})
664+
665+
describe('when "Accept-Encoding: gzip;q=0.001"', function () {
666+
brotli('should respond with gzip', function (done) {
667+
var server = createServer({ threshold: 0 }, function (req, res) {
668+
res.setHeader('Content-Type', 'text/plain')
669+
res.end('hello, world')
670+
})
671+
672+
request(server)
673+
.get('/')
674+
.set('Accept-Encoding', 'gzip;q=0.001')
675+
.expect('Content-Encoding', 'gzip', done)
676+
})
677+
})
678+
679+
describe('when "Accept-Encoding: deflate, br"', function () {
680+
brotli('should respond with br', function (done) {
681+
var server = createServer({ threshold: 0 }, function (req, res) {
682+
res.setHeader('Content-Type', 'text/plain')
683+
res.end('hello, world')
684+
})
685+
686+
request(server)
687+
.get('/')
688+
.set('Accept-Encoding', 'deflate, br')
689+
.expect('Content-Encoding', 'br', done)
690+
})
691+
})
692+
541693
describe('when "Cache-Control: no-transform" response header', function () {
542694
it('should not compress response', function (done) {
543695
var server = createServer({ threshold: 0 }, function (req, res) {
@@ -676,6 +828,32 @@ describe('compression()', function () {
676828
.end()
677829
})
678830

831+
brotli('should flush small chunks for brotli', function (done) {
832+
var chunks = 0
833+
var next
834+
var server = createServer({ threshold: 0 }, function (req, res) {
835+
next = writeAndFlush(res, 2, Buffer.from('..'))
836+
res.setHeader('Content-Type', 'text/plain')
837+
next()
838+
})
839+
840+
function onchunk (chunk) {
841+
assert.ok(chunks++ < 20)
842+
assert.strictEqual(chunk.toString(), '..')
843+
next()
844+
}
845+
846+
request(server)
847+
.get('/')
848+
.set('Accept-Encoding', 'br')
849+
.request()
850+
.on('response', unchunk('br', onchunk, function (err) {
851+
if (err) return done(err)
852+
server.close(done)
853+
}))
854+
.end()
855+
})
856+
679857
it('should flush small chunks for deflate', function (done) {
680858
var chunks = 0
681859
var next
@@ -756,6 +934,19 @@ describe('compression()', function () {
756934
.expect(200, 'hello, world', done)
757935
})
758936

937+
brotli('should compress when enforceEncoding is brotli', function (done) {
938+
var server = createServer({ threshold: 0, enforceEncoding: 'br' }, function (req, res) {
939+
res.setHeader('Content-Type', 'text/plain')
940+
res.end('hello, world')
941+
})
942+
943+
request(server)
944+
.get('/')
945+
.set('Accept-Encoding', '')
946+
.expect('Content-Encoding', 'br')
947+
.expect(200, done)
948+
})
949+
759950
it('should not compress when enforceEncoding is unknown', function (done) {
760951
var server = createServer({ threshold: 0, enforceEncoding: 'bogus' }, function (req, res) {
761952
res.setHeader('Content-Type', 'text/plain')
@@ -876,6 +1067,9 @@ function unchunk (encoding, onchunk, onend) {
8761067
case 'gzip':
8771068
stream = res.pipe(zlib.createGunzip())
8781069
break
1070+
case 'br':
1071+
stream = res.pipe(zlib.createBrotliDecompress())
1072+
break
8791073
}
8801074

8811075
stream.on('data', onchunk)

0 commit comments

Comments
 (0)