Spaces:
Running
Running
| ; | |
| const zlib = require('zlib'); | |
| const bufferUtil = require('./buffer-util'); | |
| const Limiter = require('./limiter'); | |
| const { kStatusCode } = require('./constants'); | |
| const FastBuffer = Buffer[Symbol.species]; | |
| const TRAILER = Buffer.from([0x00, 0x00, 0xff, 0xff]); | |
| const kPerMessageDeflate = Symbol('permessage-deflate'); | |
| const kTotalLength = Symbol('total-length'); | |
| const kCallback = Symbol('callback'); | |
| const kBuffers = Symbol('buffers'); | |
| const kError = Symbol('error'); | |
| // | |
| // We limit zlib concurrency, which prevents severe memory fragmentation | |
| // as documented in https://github.com/nodejs/node/issues/8871#issuecomment-250915913 | |
| // and https://github.com/websockets/ws/issues/1202 | |
| // | |
| // Intentionally global; it's the global thread pool that's an issue. | |
| // | |
| let zlibLimiter; | |
| /** | |
| * permessage-deflate implementation. | |
| */ | |
| class PerMessageDeflate { | |
| /** | |
| * Creates a PerMessageDeflate instance. | |
| * | |
| * @param {Object} [options] Configuration options | |
| * @param {(Boolean|Number)} [options.clientMaxWindowBits] Advertise support | |
| * for, or request, a custom client window size | |
| * @param {Boolean} [options.clientNoContextTakeover=false] Advertise/ | |
| * acknowledge disabling of client context takeover | |
| * @param {Number} [options.concurrencyLimit=10] The number of concurrent | |
| * calls to zlib | |
| * @param {(Boolean|Number)} [options.serverMaxWindowBits] Request/confirm the | |
| * use of a custom server window size | |
| * @param {Boolean} [options.serverNoContextTakeover=false] Request/accept | |
| * disabling of server context takeover | |
| * @param {Number} [options.threshold=1024] Size (in bytes) below which | |
| * messages should not be compressed if context takeover is disabled | |
| * @param {Object} [options.zlibDeflateOptions] Options to pass to zlib on | |
| * deflate | |
| * @param {Object} [options.zlibInflateOptions] Options to pass to zlib on | |
| * inflate | |
| * @param {Boolean} [isServer=false] Create the instance in either server or | |
| * client mode | |
| * @param {Number} [maxPayload=0] The maximum allowed message length | |
| */ | |
| constructor(options, isServer, maxPayload) { | |
| this._maxPayload = maxPayload | 0; | |
| this._options = options || {}; | |
| this._threshold = | |
| this._options.threshold !== undefined ? this._options.threshold : 1024; | |
| this._isServer = !!isServer; | |
| this._deflate = null; | |
| this._inflate = null; | |
| this.params = null; | |
| if (!zlibLimiter) { | |
| const concurrency = | |
| this._options.concurrencyLimit !== undefined | |
| ? this._options.concurrencyLimit | |
| : 10; | |
| zlibLimiter = new Limiter(concurrency); | |
| } | |
| } | |
| /** | |
| * @type {String} | |
| */ | |
| static get extensionName() { | |
| return 'permessage-deflate'; | |
| } | |
| /** | |
| * Create an extension negotiation offer. | |
| * | |
| * @return {Object} Extension parameters | |
| * @public | |
| */ | |
| offer() { | |
| const params = {}; | |
| if (this._options.serverNoContextTakeover) { | |
| params.server_no_context_takeover = true; | |
| } | |
| if (this._options.clientNoContextTakeover) { | |
| params.client_no_context_takeover = true; | |
| } | |
| if (this._options.serverMaxWindowBits) { | |
| params.server_max_window_bits = this._options.serverMaxWindowBits; | |
| } | |
| if (this._options.clientMaxWindowBits) { | |
| params.client_max_window_bits = this._options.clientMaxWindowBits; | |
| } else if (this._options.clientMaxWindowBits == null) { | |
| params.client_max_window_bits = true; | |
| } | |
| return params; | |
| } | |
| /** | |
| * Accept an extension negotiation offer/response. | |
| * | |
| * @param {Array} configurations The extension negotiation offers/reponse | |
| * @return {Object} Accepted configuration | |
| * @public | |
| */ | |
| accept(configurations) { | |
| configurations = this.normalizeParams(configurations); | |
| this.params = this._isServer | |
| ? this.acceptAsServer(configurations) | |
| : this.acceptAsClient(configurations); | |
| return this.params; | |
| } | |
| /** | |
| * Releases all resources used by the extension. | |
| * | |
| * @public | |
| */ | |
| cleanup() { | |
| if (this._inflate) { | |
| this._inflate.close(); | |
| this._inflate = null; | |
| } | |
| if (this._deflate) { | |
| const callback = this._deflate[kCallback]; | |
| this._deflate.close(); | |
| this._deflate = null; | |
| if (callback) { | |
| callback( | |
| new Error( | |
| 'The deflate stream was closed while data was being processed' | |
| ) | |
| ); | |
| } | |
| } | |
| } | |
| /** | |
| * Accept an extension negotiation offer. | |
| * | |
| * @param {Array} offers The extension negotiation offers | |
| * @return {Object} Accepted configuration | |
| * @private | |
| */ | |
| acceptAsServer(offers) { | |
| const opts = this._options; | |
| const accepted = offers.find((params) => { | |
| if ( | |
| (opts.serverNoContextTakeover === false && | |
| params.server_no_context_takeover) || | |
| (params.server_max_window_bits && | |
| (opts.serverMaxWindowBits === false || | |
| (typeof opts.serverMaxWindowBits === 'number' && | |
| opts.serverMaxWindowBits > params.server_max_window_bits))) || | |
| (typeof opts.clientMaxWindowBits === 'number' && | |
| !params.client_max_window_bits) | |
| ) { | |
| return false; | |
| } | |
| return true; | |
| }); | |
| if (!accepted) { | |
| throw new Error('None of the extension offers can be accepted'); | |
| } | |
| if (opts.serverNoContextTakeover) { | |
| accepted.server_no_context_takeover = true; | |
| } | |
| if (opts.clientNoContextTakeover) { | |
| accepted.client_no_context_takeover = true; | |
| } | |
| if (typeof opts.serverMaxWindowBits === 'number') { | |
| accepted.server_max_window_bits = opts.serverMaxWindowBits; | |
| } | |
| if (typeof opts.clientMaxWindowBits === 'number') { | |
| accepted.client_max_window_bits = opts.clientMaxWindowBits; | |
| } else if ( | |
| accepted.client_max_window_bits === true || | |
| opts.clientMaxWindowBits === false | |
| ) { | |
| delete accepted.client_max_window_bits; | |
| } | |
| return accepted; | |
| } | |
| /** | |
| * Accept the extension negotiation response. | |
| * | |
| * @param {Array} response The extension negotiation response | |
| * @return {Object} Accepted configuration | |
| * @private | |
| */ | |
| acceptAsClient(response) { | |
| const params = response[0]; | |
| if ( | |
| this._options.clientNoContextTakeover === false && | |
| params.client_no_context_takeover | |
| ) { | |
| throw new Error('Unexpected parameter "client_no_context_takeover"'); | |
| } | |
| if (!params.client_max_window_bits) { | |
| if (typeof this._options.clientMaxWindowBits === 'number') { | |
| params.client_max_window_bits = this._options.clientMaxWindowBits; | |
| } | |
| } else if ( | |
| this._options.clientMaxWindowBits === false || | |
| (typeof this._options.clientMaxWindowBits === 'number' && | |
| params.client_max_window_bits > this._options.clientMaxWindowBits) | |
| ) { | |
| throw new Error( | |
| 'Unexpected or invalid parameter "client_max_window_bits"' | |
| ); | |
| } | |
| return params; | |
| } | |
| /** | |
| * Normalize parameters. | |
| * | |
| * @param {Array} configurations The extension negotiation offers/reponse | |
| * @return {Array} The offers/response with normalized parameters | |
| * @private | |
| */ | |
| normalizeParams(configurations) { | |
| configurations.forEach((params) => { | |
| Object.keys(params).forEach((key) => { | |
| let value = params[key]; | |
| if (value.length > 1) { | |
| throw new Error(`Parameter "${key}" must have only a single value`); | |
| } | |
| value = value[0]; | |
| if (key === 'client_max_window_bits') { | |
| if (value !== true) { | |
| const num = +value; | |
| if (!Number.isInteger(num) || num < 8 || num > 15) { | |
| throw new TypeError( | |
| `Invalid value for parameter "${key}": ${value}` | |
| ); | |
| } | |
| value = num; | |
| } else if (!this._isServer) { | |
| throw new TypeError( | |
| `Invalid value for parameter "${key}": ${value}` | |
| ); | |
| } | |
| } else if (key === 'server_max_window_bits') { | |
| const num = +value; | |
| if (!Number.isInteger(num) || num < 8 || num > 15) { | |
| throw new TypeError( | |
| `Invalid value for parameter "${key}": ${value}` | |
| ); | |
| } | |
| value = num; | |
| } else if ( | |
| key === 'client_no_context_takeover' || | |
| key === 'server_no_context_takeover' | |
| ) { | |
| if (value !== true) { | |
| throw new TypeError( | |
| `Invalid value for parameter "${key}": ${value}` | |
| ); | |
| } | |
| } else { | |
| throw new Error(`Unknown parameter "${key}"`); | |
| } | |
| params[key] = value; | |
| }); | |
| }); | |
| return configurations; | |
| } | |
| /** | |
| * Decompress data. Concurrency limited. | |
| * | |
| * @param {Buffer} data Compressed data | |
| * @param {Boolean} fin Specifies whether or not this is the last fragment | |
| * @param {Function} callback Callback | |
| * @public | |
| */ | |
| decompress(data, fin, callback) { | |
| zlibLimiter.add((done) => { | |
| this._decompress(data, fin, (err, result) => { | |
| done(); | |
| callback(err, result); | |
| }); | |
| }); | |
| } | |
| /** | |
| * Compress data. Concurrency limited. | |
| * | |
| * @param {(Buffer|String)} data Data to compress | |
| * @param {Boolean} fin Specifies whether or not this is the last fragment | |
| * @param {Function} callback Callback | |
| * @public | |
| */ | |
| compress(data, fin, callback) { | |
| zlibLimiter.add((done) => { | |
| this._compress(data, fin, (err, result) => { | |
| done(); | |
| callback(err, result); | |
| }); | |
| }); | |
| } | |
| /** | |
| * Decompress data. | |
| * | |
| * @param {Buffer} data Compressed data | |
| * @param {Boolean} fin Specifies whether or not this is the last fragment | |
| * @param {Function} callback Callback | |
| * @private | |
| */ | |
| _decompress(data, fin, callback) { | |
| const endpoint = this._isServer ? 'client' : 'server'; | |
| if (!this._inflate) { | |
| const key = `${endpoint}_max_window_bits`; | |
| const windowBits = | |
| typeof this.params[key] !== 'number' | |
| ? zlib.Z_DEFAULT_WINDOWBITS | |
| : this.params[key]; | |
| this._inflate = zlib.createInflateRaw({ | |
| ...this._options.zlibInflateOptions, | |
| windowBits | |
| }); | |
| this._inflate[kPerMessageDeflate] = this; | |
| this._inflate[kTotalLength] = 0; | |
| this._inflate[kBuffers] = []; | |
| this._inflate.on('error', inflateOnError); | |
| this._inflate.on('data', inflateOnData); | |
| } | |
| this._inflate[kCallback] = callback; | |
| this._inflate.write(data); | |
| if (fin) this._inflate.write(TRAILER); | |
| this._inflate.flush(() => { | |
| const err = this._inflate[kError]; | |
| if (err) { | |
| this._inflate.close(); | |
| this._inflate = null; | |
| callback(err); | |
| return; | |
| } | |
| const data = bufferUtil.concat( | |
| this._inflate[kBuffers], | |
| this._inflate[kTotalLength] | |
| ); | |
| if (this._inflate._readableState.endEmitted) { | |
| this._inflate.close(); | |
| this._inflate = null; | |
| } else { | |
| this._inflate[kTotalLength] = 0; | |
| this._inflate[kBuffers] = []; | |
| if (fin && this.params[`${endpoint}_no_context_takeover`]) { | |
| this._inflate.reset(); | |
| } | |
| } | |
| callback(null, data); | |
| }); | |
| } | |
| /** | |
| * Compress data. | |
| * | |
| * @param {(Buffer|String)} data Data to compress | |
| * @param {Boolean} fin Specifies whether or not this is the last fragment | |
| * @param {Function} callback Callback | |
| * @private | |
| */ | |
| _compress(data, fin, callback) { | |
| const endpoint = this._isServer ? 'server' : 'client'; | |
| if (!this._deflate) { | |
| const key = `${endpoint}_max_window_bits`; | |
| const windowBits = | |
| typeof this.params[key] !== 'number' | |
| ? zlib.Z_DEFAULT_WINDOWBITS | |
| : this.params[key]; | |
| this._deflate = zlib.createDeflateRaw({ | |
| ...this._options.zlibDeflateOptions, | |
| windowBits | |
| }); | |
| this._deflate[kTotalLength] = 0; | |
| this._deflate[kBuffers] = []; | |
| this._deflate.on('data', deflateOnData); | |
| } | |
| this._deflate[kCallback] = callback; | |
| this._deflate.write(data); | |
| this._deflate.flush(zlib.Z_SYNC_FLUSH, () => { | |
| if (!this._deflate) { | |
| // | |
| // The deflate stream was closed while data was being processed. | |
| // | |
| return; | |
| } | |
| let data = bufferUtil.concat( | |
| this._deflate[kBuffers], | |
| this._deflate[kTotalLength] | |
| ); | |
| if (fin) { | |
| data = new FastBuffer(data.buffer, data.byteOffset, data.length - 4); | |
| } | |
| // | |
| // Ensure that the callback will not be called again in | |
| // `PerMessageDeflate#cleanup()`. | |
| // | |
| this._deflate[kCallback] = null; | |
| this._deflate[kTotalLength] = 0; | |
| this._deflate[kBuffers] = []; | |
| if (fin && this.params[`${endpoint}_no_context_takeover`]) { | |
| this._deflate.reset(); | |
| } | |
| callback(null, data); | |
| }); | |
| } | |
| } | |
| module.exports = PerMessageDeflate; | |
| /** | |
| * The listener of the `zlib.DeflateRaw` stream `'data'` event. | |
| * | |
| * @param {Buffer} chunk A chunk of data | |
| * @private | |
| */ | |
| function deflateOnData(chunk) { | |
| this[kBuffers].push(chunk); | |
| this[kTotalLength] += chunk.length; | |
| } | |
| /** | |
| * The listener of the `zlib.InflateRaw` stream `'data'` event. | |
| * | |
| * @param {Buffer} chunk A chunk of data | |
| * @private | |
| */ | |
| function inflateOnData(chunk) { | |
| this[kTotalLength] += chunk.length; | |
| if ( | |
| this[kPerMessageDeflate]._maxPayload < 1 || | |
| this[kTotalLength] <= this[kPerMessageDeflate]._maxPayload | |
| ) { | |
| this[kBuffers].push(chunk); | |
| return; | |
| } | |
| this[kError] = new RangeError('Max payload size exceeded'); | |
| this[kError].code = 'WS_ERR_UNSUPPORTED_MESSAGE_LENGTH'; | |
| this[kError][kStatusCode] = 1009; | |
| this.removeListener('data', inflateOnData); | |
| this.reset(); | |
| } | |
| /** | |
| * The listener of the `zlib.InflateRaw` stream `'error'` event. | |
| * | |
| * @param {Error} err The emitted error | |
| * @private | |
| */ | |
| function inflateOnError(err) { | |
| // | |
| // There is no need to call `Zlib#close()` as the handle is automatically | |
| // closed when an error is emitted. | |
| // | |
| this[kPerMessageDeflate]._inflate = null; | |
| err[kStatusCode] = 1007; | |
| this[kCallback](err); | |
| } | |