Skip to content

Commit 24203e0

Browse files
committed
Merge branch 'master' into feature/brotli
2 parents 5516b87 + f4e596c commit 24203e0

File tree

4 files changed

+183
-3
lines changed

4 files changed

+183
-3
lines changed

HISTORY.md

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

67
1.7.5 / 2024-10-31
78
==========
8-
99
* deps: Replace accepts with negotiator@~0.6.4
1010
- Add preference option
1111

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -148,6 +148,12 @@ The default value is `zlib.Z_DEFAULT_WINDOWBITS`, or `15`.
148148
See [Node.js documentation](http://nodejs.org/api/zlib.html#zlib_memory_usage_tuning)
149149
regarding the usage.
150150

151+
##### enforceEncoding
152+
153+
This is the default encoding to use when the client does not specify an encoding in the request's [Accept-Encoding](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Accept-Encoding) header.
154+
155+
The default value is `identity`.
156+
151157
#### .filter
152158

153159
The default `filter` function. This is used to construct a custom filter

index.js

+10-2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@ var cacheControlNoTransformRegExp = /(?:^|,)\s*?no-transform\s*?(?:,|$)/
4545
var SUPPORTED_ENCODING = hasBrotliSupport ? ['br', 'gzip', 'deflate', 'identity'] : ['gzip', 'deflate', 'identity']
4646
var PREFERRED_ENCODING = hasBrotliSupport ? ['br', 'gzip'] : ['gzip']
4747

48+
var encodingSupported = ['*', 'gzip', 'deflate', 'identity']
49+
4850
/**
4951
* Compress response data with gzip / deflate.
5052
*
@@ -68,6 +70,7 @@ function compression (options) {
6870
// options
6971
var filter = opts.filter || shouldCompress
7072
var threshold = bytes.parse(opts.threshold)
73+
var enforceEncoding = opts.enforceEncoding || 'identity'
7174

7275
if (threshold == null) {
7376
threshold = 1024
@@ -98,7 +101,7 @@ function compression (options) {
98101
}
99102

100103
if (!headersSent(res)) {
101-
this._implicitHeader()
104+
this.writeHead(this.statusCode)
102105
}
103106

104107
return stream
@@ -117,7 +120,7 @@ function compression (options) {
117120
length = chunkLength(chunk, encoding)
118121
}
119122

120-
this._implicitHeader()
123+
this.writeHead(this.statusCode)
121124
}
122125

123126
if (!stream) {
@@ -194,6 +197,11 @@ function compression (options) {
194197
var negotiator = new Negotiator(req)
195198
var method = negotiator.encoding(SUPPORTED_ENCODING, PREFERRED_ENCODING)
196199

200+
// if no method is found, use the default encoding
201+
if (!req.headers['accept-encoding'] && encodingSupported.indexOf(enforceEncoding) !== -1) {
202+
method = enforceEncoding === '*' ? 'gzip' : enforceEncoding
203+
}
204+
197205
// negotiation failed
198206
if (!method || method === 'identity') {
199207
nocompress('not acceptable')

test/compression.js

+166
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,16 @@ var http = require('http')
77
var request = require('supertest')
88
var zlib = require('zlib')
99

10+
var describeHttp2 = describe.skip
11+
try {
12+
var http2 = require('http2')
13+
describeHttp2 = describe
14+
} catch (err) {
15+
if (err) {
16+
console.log('http2 tests disabled.')
17+
}
18+
}
19+
1020
var compression = require('..')
1121

1222
/**
@@ -312,6 +322,41 @@ describe('compression()', function () {
312322
.expect(200, done)
313323
})
314324

325+
describeHttp2('http2', function () {
326+
it('should work with http2 server', function (done) {
327+
var server = createHttp2Server({ threshold: 0 }, function (req, res) {
328+
res.setHeader(http2.constants.HTTP2_HEADER_CONTENT_TYPE, 'text/plain')
329+
res.end('hello, world')
330+
})
331+
server.on('listening', function () {
332+
var client = createHttp2Client(server.address().port)
333+
// using ES5 as Node.js <=4.0.0 does not have Computed Property Names
334+
var reqHeaders = {}
335+
reqHeaders[http2.constants.HTTP2_HEADER_ACCEPT_ENCODING] = 'gzip'
336+
var request = client.request(reqHeaders)
337+
request.on('response', function (headers) {
338+
assert.strictEqual(headers[http2.constants.HTTP2_HEADER_STATUS], 200)
339+
assert.strictEqual(headers[http2.constants.HTTP2_HEADER_CONTENT_TYPE], 'text/plain')
340+
assert.strictEqual(headers[http2.constants.HTTP2_HEADER_CONTENT_ENCODING], 'gzip')
341+
})
342+
var chunks = []
343+
request.on('data', function (chunk) {
344+
chunks.push(chunk)
345+
})
346+
request.on('end', function () {
347+
closeHttp2(client, server, function () {
348+
zlib.gunzip(Buffer.concat(chunks), function (err, data) {
349+
assert.ok(!err)
350+
assert.strictEqual(data.toString(), 'hello, world')
351+
done()
352+
})
353+
})
354+
})
355+
request.end()
356+
})
357+
})
358+
})
359+
315360
describe('threshold', function () {
316361
it('should not compress responses below the threshold size', function (done) {
317362
var server = createServer({ threshold: '1kb' }, function (req, res) {
@@ -835,6 +880,86 @@ describe('compression()', function () {
835880
.end()
836881
})
837882
})
883+
884+
describe('enforceEncoding', function () {
885+
it('should compress the provided encoding and not the default encoding', function (done) {
886+
var server = createServer({ threshold: 0, enforceEncoding: 'deflate' }, function (req, res) {
887+
res.setHeader('Content-Type', 'text/plain')
888+
res.end('hello, world')
889+
})
890+
891+
request(server)
892+
.get('/')
893+
.set('Accept-Encoding', 'gzip')
894+
.expect('Content-Encoding', 'gzip')
895+
.expect(200, 'hello, world', done)
896+
})
897+
898+
it('should not compress when enforceEncoding is identity', function (done) {
899+
var server = createServer({ threshold: 0, enforceEncoding: 'identity' }, function (req, res) {
900+
res.setHeader('Content-Type', 'text/plain')
901+
res.end('hello, world')
902+
})
903+
904+
request(server)
905+
.get('/')
906+
.set('Accept-Encoding', '')
907+
.expect(shouldNotHaveHeader('Content-Encoding'))
908+
.expect(200, 'hello, world', done)
909+
})
910+
911+
it('should compress when enforceEncoding is gzip', function (done) {
912+
var server = createServer({ threshold: 0, enforceEncoding: 'gzip' }, function (req, res) {
913+
res.setHeader('Content-Type', 'text/plain')
914+
res.end('hello, world')
915+
})
916+
917+
request(server)
918+
.get('/')
919+
.set('Accept-Encoding', '')
920+
.expect('Content-Encoding', 'gzip')
921+
.expect(200, 'hello, world', done)
922+
})
923+
924+
it('should compress when enforceEncoding is deflate', function (done) {
925+
var server = createServer({ threshold: 0, enforceEncoding: 'deflate' }, function (req, res) {
926+
res.setHeader('Content-Type', 'text/plain')
927+
res.end('hello, world')
928+
})
929+
930+
request(server)
931+
.get('/')
932+
.set('Accept-Encoding', '')
933+
.expect('Content-Encoding', 'deflate')
934+
.expect(200, 'hello, world', done)
935+
})
936+
937+
it('should not compress when enforceEncoding is unknown', function (done) {
938+
var server = createServer({ threshold: 0, enforceEncoding: 'bogus' }, 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(shouldNotHaveHeader('Content-Encoding'))
947+
.expect(200, 'hello, world', done)
948+
})
949+
950+
it('should be gzip if no accept-encoding is sent when enforceEncoding is *', function (done) {
951+
var server = createServer({ threshold: 0, enforceEncoding: '*' }, function (req, res) {
952+
res.setHeader('Content-Type', 'text/plain')
953+
res.end('hello, world')
954+
})
955+
956+
request(server)
957+
.get('/')
958+
.set('Accept-Encoding', '')
959+
.expect('Content-Encoding', 'gzip')
960+
.expect(200, 'hello, world', done)
961+
})
962+
})
838963
})
839964

840965
function createServer (opts, fn) {
@@ -852,6 +977,47 @@ function createServer (opts, fn) {
852977
})
853978
}
854979

980+
function createHttp2Server (opts, fn) {
981+
var _compression = compression(opts)
982+
var server = http2.createServer(function (req, res) {
983+
_compression(req, res, function (err) {
984+
if (err) {
985+
res.statusCode = err.status || 500
986+
res.end(err.message)
987+
return
988+
}
989+
990+
fn(req, res)
991+
})
992+
})
993+
server.listen(0, '127.0.0.1')
994+
return server
995+
}
996+
997+
function createHttp2Client (port) {
998+
return http2.connect('http://127.0.0.1:' + port)
999+
}
1000+
1001+
function closeHttp2 (client, server, callback) {
1002+
if (typeof client.shutdown === 'function') {
1003+
// this is the node v8.x way of closing the connections
1004+
client.shutdown({}, function () {
1005+
server.close(function () {
1006+
callback()
1007+
})
1008+
})
1009+
} else {
1010+
// this is the node v9.x onwards way of closing the connections
1011+
client.close(function () {
1012+
// force existing connections to time out after 1ms.
1013+
// this is done to force the server to close in some cases where it wouldn't do it otherwise.
1014+
server.close(function () {
1015+
callback()
1016+
})
1017+
})
1018+
}
1019+
}
1020+
8551021
function shouldHaveBodyLength (length) {
8561022
return function (res) {
8571023
assert.strictEqual(res.text.length, length, 'should have body length of ' + length)

0 commit comments

Comments
 (0)