From 33b22d7c4fc8e45872f28f88fbfbbb16b14f517c Mon Sep 17 00:00:00 2001 From: unknown Date: Fri, 22 May 2020 13:22:09 -0400 Subject: [PATCH] zlib: add `maxOutputLength` option Fixes: https://github.com/nodejs/node/issues/27253 PR-URL: https://github.com/nodejs/node/pull/33516 Reviewed-By: Anna Henningsen --- doc/api/zlib.md | 13 ++++++++++ lib/internal/errors.js | 2 +- lib/zlib.js | 25 +++++++++++++------ .../test-zlib-brotli-kmaxlength-rangeerror.js | 2 +- .../test-zlib-kmaxlength-rangeerror.js | 2 +- test/parallel/test-zlib-maxOutputLength.js | 25 +++++++++++++++++++ 6 files changed, 59 insertions(+), 10 deletions(-) create mode 100644 test/parallel/test-zlib-maxOutputLength.js diff --git a/doc/api/zlib.md b/doc/api/zlib.md index 759ad4f58194c2..72b9452ca0971a 100644 --- a/doc/api/zlib.md +++ b/doc/api/zlib.md @@ -486,6 +486,9 @@ These advanced options are available for controlling decompression: @@ -531,6 +540,8 @@ Each Brotli-based class takes an `options` object. All options are optional. * `finishFlush` {integer} **Default:** `zlib.constants.BROTLI_OPERATION_FINISH` * `chunkSize` {integer} **Default:** `16 * 1024` * `params` {Object} Key-value object containing indexed [Brotli parameters][]. +* `maxOutputLength` {integer} Limits output size when using + [convenience methods][]. **Default:** [`buffer.kMaxLength`][] For example: @@ -1142,6 +1153,7 @@ Decompress a chunk of data with [`Unzip`][]. [`BrotliCompress`]: #zlib_class_zlib_brotlicompress [`BrotliDecompress`]: #zlib_class_zlib_brotlidecompress [`Buffer`]: buffer.html#buffer_class_buffer +[`buffer.kMaxLength`]: buffer.html#buffer_buffer_kmaxlength [`Content-Encoding`]: https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.11 [`DataView`]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DataView [`DeflateRaw`]: #zlib_class_zlib_deflateraw @@ -1159,5 +1171,6 @@ Decompress a chunk of data with [`Unzip`][]. [Memory usage tuning]: #zlib_memory_usage_tuning [RFC 7932]: https://www.rfc-editor.org/rfc/rfc7932.txt [Streams API]: stream.md +[convenience methods]: #zlib_convenience_methods [zlib documentation]: https://zlib.net/manual.html#Constants [zlib.createGzip example]: #zlib_zlib diff --git a/lib/internal/errors.js b/lib/internal/errors.js index 507fde4352d39e..921bcce2878706 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -763,7 +763,7 @@ E('ERR_BUFFER_OUT_OF_BOUNDS', return 'Attempt to access memory outside buffer bounds'; }, RangeError); E('ERR_BUFFER_TOO_LARGE', - `Cannot create a Buffer larger than 0x${kMaxLength.toString(16)} bytes`, + 'Cannot create a Buffer larger than %s bytes', RangeError); E('ERR_CANNOT_WATCH_SIGINT', 'Cannot watch for SIGINT signals', Error); E('ERR_CHILD_CLOSED_BEFORE_REPLY', diff --git a/lib/zlib.js b/lib/zlib.js index 8168aeea09f86f..8a81856949b18c 100644 --- a/lib/zlib.js +++ b/lib/zlib.js @@ -122,6 +122,11 @@ function zlibBufferOnData(chunk) { else this.buffers.push(chunk); this.nread += chunk.length; + if (this.nread > this._maxOutputLength) { + this.close(); + this.removeAllListeners('end'); + this.cb(new ERR_BUFFER_TOO_LARGE(this._maxOutputLength)); + } } function zlibBufferOnError(err) { @@ -132,9 +137,7 @@ function zlibBufferOnError(err) { function zlibBufferOnEnd() { let buf; let err; - if (this.nread >= kMaxLength) { - err = new ERR_BUFFER_TOO_LARGE(); - } else if (this.nread === 0) { + if (this.nread === 0) { buf = Buffer.alloc(0); } else { const bufs = this.buffers; @@ -230,6 +233,7 @@ const checkRangesOrGetDefault = hideStackFrames( // The base class for all Zlib-style streams. function ZlibBase(opts, mode, handle, { flush, finishFlush, fullFlush }) { let chunkSize = Z_DEFAULT_CHUNK; + let maxOutputLength = kMaxLength; // The ZlibBase class is not exported to user land, the mode should only be // passed in by us. assert(typeof mode === 'number'); @@ -252,6 +256,10 @@ function ZlibBase(opts, mode, handle, { flush, finishFlush, fullFlush }) { opts.finishFlush, 'options.finishFlush', Z_NO_FLUSH, Z_BLOCK, finishFlush); + maxOutputLength = checkRangesOrGetDefault( + opts.maxOutputLength, 'options.maxOutputLength', + 1, kMaxLength, kMaxLength); + if (opts.encoding || opts.objectMode || opts.writableObjectMode) { opts = { ...opts }; opts.encoding = null; @@ -276,6 +284,7 @@ function ZlibBase(opts, mode, handle, { flush, finishFlush, fullFlush }) { this._defaultFullFlushFlag = fullFlush; this.once('end', _close.bind(null, this)); this._info = opts && opts.info; + this._maxOutputLength = maxOutputLength; } ObjectSetPrototypeOf(ZlibBase.prototype, Transform.prototype); ObjectSetPrototypeOf(ZlibBase, Transform); @@ -445,6 +454,12 @@ function processChunkSync(self, chunk, flushFlag) { else buffers.push(out); nread += out.byteLength; + + if (nread > self._maxOutputLength) { + _close(self); + throw new ERR_BUFFER_TOO_LARGE(self._maxOutputLength); + } + } else { assert(have === 0, 'have should not go down'); } @@ -471,10 +486,6 @@ function processChunkSync(self, chunk, flushFlag) { self.bytesWritten = inputRead; _close(self); - if (nread >= kMaxLength) { - throw new ERR_BUFFER_TOO_LARGE(); - } - if (nread === 0) return Buffer.alloc(0); diff --git a/test/parallel/test-zlib-brotli-kmaxlength-rangeerror.js b/test/parallel/test-zlib-brotli-kmaxlength-rangeerror.js index c1765eec2983ee..6a59ad34b0a174 100644 --- a/test/parallel/test-zlib-brotli-kmaxlength-rangeerror.js +++ b/test/parallel/test-zlib-brotli-kmaxlength-rangeerror.js @@ -11,7 +11,7 @@ const assert = require('assert'); // large Buffers. const buffer = require('buffer'); const oldkMaxLength = buffer.kMaxLength; -buffer.kMaxLength = 128; +buffer.kMaxLength = 64; const zlib = require('zlib'); buffer.kMaxLength = oldkMaxLength; diff --git a/test/parallel/test-zlib-kmaxlength-rangeerror.js b/test/parallel/test-zlib-kmaxlength-rangeerror.js index e8e47865f79604..9803630214eb36 100644 --- a/test/parallel/test-zlib-kmaxlength-rangeerror.js +++ b/test/parallel/test-zlib-kmaxlength-rangeerror.js @@ -11,7 +11,7 @@ const assert = require('assert'); // large Buffers. const buffer = require('buffer'); const oldkMaxLength = buffer.kMaxLength; -buffer.kMaxLength = 128; +buffer.kMaxLength = 64; const zlib = require('zlib'); buffer.kMaxLength = oldkMaxLength; diff --git a/test/parallel/test-zlib-maxOutputLength.js b/test/parallel/test-zlib-maxOutputLength.js new file mode 100644 index 00000000000000..9af0b3736f8815 --- /dev/null +++ b/test/parallel/test-zlib-maxOutputLength.js @@ -0,0 +1,25 @@ +'use strict'; +const common = require('../common'); +const assert = require('assert'); +const zlib = require('zlib'); + +const encoded = Buffer.from('G38A+CXCIrFAIAM=', 'base64'); + +// Async +zlib.brotliDecompress(encoded, { maxOutputLength: 64 }, common.expectsError({ + code: 'ERR_BUFFER_TOO_LARGE', + message: 'Cannot create a Buffer larger than 64 bytes' +})); + +// Sync +assert.throws(function() { + zlib.brotliDecompressSync(encoded, { maxOutputLength: 64 }); +}, RangeError); + +// Async +zlib.brotliDecompress(encoded, { maxOutputLength: 256 }, function(err) { + assert.strictEqual(err, null); +}); + +// Sync +zlib.brotliDecompressSync(encoded, { maxOutputLength: 256 });