254 lines
7.6 KiB
JavaScript
254 lines
7.6 KiB
JavaScript
'use strict'
|
|
|
|
const { createInflate, createGunzip, createBrotliDecompress, createZstdDecompress } = require('node:zlib')
|
|
const { pipeline } = require('node:stream')
|
|
const DecoratorHandler = require('../handler/decorator-handler')
|
|
|
|
/** @typedef {import('node:stream').Transform} Transform */
|
|
/** @typedef {import('node:stream').Transform} Controller */
|
|
/** @typedef {Transform&import('node:zlib').Zlib} DecompressorStream */
|
|
|
|
/** @type {Record<string, () => DecompressorStream>} */
|
|
const supportedEncodings = {
|
|
gzip: createGunzip,
|
|
'x-gzip': createGunzip,
|
|
br: createBrotliDecompress,
|
|
deflate: createInflate,
|
|
compress: createInflate,
|
|
'x-compress': createInflate,
|
|
...(createZstdDecompress ? { zstd: createZstdDecompress } : {})
|
|
}
|
|
|
|
const defaultSkipStatusCodes = /** @type {const} */ ([204, 304])
|
|
|
|
let warningEmitted = /** @type {boolean} */ (false)
|
|
|
|
/**
|
|
* @typedef {Object} DecompressHandlerOptions
|
|
* @property {number[]|Readonly<number[]>} [skipStatusCodes=[204, 304]] - List of status codes to skip decompression for
|
|
* @property {boolean} [skipErrorResponses] - Whether to skip decompression for error responses (status codes >= 400)
|
|
*/
|
|
|
|
class DecompressHandler extends DecoratorHandler {
|
|
/** @type {Transform[]} */
|
|
#decompressors = []
|
|
/** @type {NodeJS.WritableStream&NodeJS.ReadableStream|null} */
|
|
#pipelineStream
|
|
/** @type {Readonly<number[]>} */
|
|
#skipStatusCodes
|
|
/** @type {boolean} */
|
|
#skipErrorResponses
|
|
|
|
constructor (handler, { skipStatusCodes = defaultSkipStatusCodes, skipErrorResponses = true } = {}) {
|
|
super(handler)
|
|
this.#skipStatusCodes = skipStatusCodes
|
|
this.#skipErrorResponses = skipErrorResponses
|
|
}
|
|
|
|
/**
|
|
* Determines if decompression should be skipped based on encoding and status code
|
|
* @param {string} contentEncoding - Content-Encoding header value
|
|
* @param {number} statusCode - HTTP status code of the response
|
|
* @returns {boolean} - True if decompression should be skipped
|
|
*/
|
|
#shouldSkipDecompression (contentEncoding, statusCode) {
|
|
if (!contentEncoding || statusCode < 200) return true
|
|
if (this.#skipStatusCodes.includes(statusCode)) return true
|
|
if (this.#skipErrorResponses && statusCode >= 400) return true
|
|
return false
|
|
}
|
|
|
|
/**
|
|
* Creates a chain of decompressors for multiple content encodings
|
|
*
|
|
* @param {string} encodings - Comma-separated list of content encodings
|
|
* @returns {Array<DecompressorStream>} - Array of decompressor streams
|
|
*/
|
|
#createDecompressionChain (encodings) {
|
|
const parts = encodings.split(',')
|
|
|
|
/** @type {DecompressorStream[]} */
|
|
const decompressors = []
|
|
|
|
for (let i = parts.length - 1; i >= 0; i--) {
|
|
const encoding = parts[i].trim()
|
|
if (!encoding) continue
|
|
|
|
if (!supportedEncodings[encoding]) {
|
|
decompressors.length = 0 // Clear if unsupported encoding
|
|
return decompressors // Unsupported encoding
|
|
}
|
|
|
|
decompressors.push(supportedEncodings[encoding]())
|
|
}
|
|
|
|
return decompressors
|
|
}
|
|
|
|
/**
|
|
* Sets up event handlers for a decompressor stream using readable events
|
|
* @param {DecompressorStream} decompressor - The decompressor stream
|
|
* @param {Controller} controller - The controller to coordinate with
|
|
* @returns {void}
|
|
*/
|
|
#setupDecompressorEvents (decompressor, controller) {
|
|
decompressor.on('readable', () => {
|
|
let chunk
|
|
while ((chunk = decompressor.read()) !== null) {
|
|
const result = super.onResponseData(controller, chunk)
|
|
if (result === false) {
|
|
break
|
|
}
|
|
}
|
|
})
|
|
|
|
decompressor.on('error', (error) => {
|
|
super.onResponseError(controller, error)
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Sets up event handling for a single decompressor
|
|
* @param {Controller} controller - The controller to handle events
|
|
* @returns {void}
|
|
*/
|
|
#setupSingleDecompressor (controller) {
|
|
const decompressor = this.#decompressors[0]
|
|
this.#setupDecompressorEvents(decompressor, controller)
|
|
|
|
decompressor.on('end', () => {
|
|
super.onResponseEnd(controller, {})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Sets up event handling for multiple chained decompressors using pipeline
|
|
* @param {Controller} controller - The controller to handle events
|
|
* @returns {void}
|
|
*/
|
|
#setupMultipleDecompressors (controller) {
|
|
const lastDecompressor = this.#decompressors[this.#decompressors.length - 1]
|
|
this.#setupDecompressorEvents(lastDecompressor, controller)
|
|
|
|
this.#pipelineStream = pipeline(this.#decompressors, (err) => {
|
|
if (err) {
|
|
super.onResponseError(controller, err)
|
|
return
|
|
}
|
|
super.onResponseEnd(controller, {})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Cleans up decompressor references to prevent memory leaks
|
|
* @returns {void}
|
|
*/
|
|
#cleanupDecompressors () {
|
|
this.#decompressors.length = 0
|
|
this.#pipelineStream = null
|
|
}
|
|
|
|
/**
|
|
* @param {Controller} controller
|
|
* @param {number} statusCode
|
|
* @param {Record<string, string | string[] | undefined>} headers
|
|
* @param {string} statusMessage
|
|
* @returns {void}
|
|
*/
|
|
onResponseStart (controller, statusCode, headers, statusMessage) {
|
|
const contentEncoding = headers['content-encoding']
|
|
|
|
// If content encoding is not supported or status code is in skip list
|
|
if (this.#shouldSkipDecompression(contentEncoding, statusCode)) {
|
|
return super.onResponseStart(controller, statusCode, headers, statusMessage)
|
|
}
|
|
|
|
const decompressors = this.#createDecompressionChain(contentEncoding.toLowerCase())
|
|
|
|
if (decompressors.length === 0) {
|
|
this.#cleanupDecompressors()
|
|
return super.onResponseStart(controller, statusCode, headers, statusMessage)
|
|
}
|
|
|
|
this.#decompressors = decompressors
|
|
|
|
// Remove compression headers since we're decompressing
|
|
const { 'content-encoding': _, 'content-length': __, ...newHeaders } = headers
|
|
|
|
if (this.#decompressors.length === 1) {
|
|
this.#setupSingleDecompressor(controller)
|
|
} else {
|
|
this.#setupMultipleDecompressors(controller)
|
|
}
|
|
|
|
super.onResponseStart(controller, statusCode, newHeaders, statusMessage)
|
|
}
|
|
|
|
/**
|
|
* @param {Controller} controller
|
|
* @param {Buffer} chunk
|
|
* @returns {void}
|
|
*/
|
|
onResponseData (controller, chunk) {
|
|
if (this.#decompressors.length > 0) {
|
|
this.#decompressors[0].write(chunk)
|
|
return
|
|
}
|
|
super.onResponseData(controller, chunk)
|
|
}
|
|
|
|
/**
|
|
* @param {Controller} controller
|
|
* @param {Record<string, string | string[]> | undefined} trailers
|
|
* @returns {void}
|
|
*/
|
|
onResponseEnd (controller, trailers) {
|
|
if (this.#decompressors.length > 0) {
|
|
this.#decompressors[0].end()
|
|
this.#cleanupDecompressors()
|
|
return
|
|
}
|
|
super.onResponseEnd(controller, trailers)
|
|
}
|
|
|
|
/**
|
|
* @param {Controller} controller
|
|
* @param {Error} err
|
|
* @returns {void}
|
|
*/
|
|
onResponseError (controller, err) {
|
|
if (this.#decompressors.length > 0) {
|
|
for (const decompressor of this.#decompressors) {
|
|
decompressor.destroy(err)
|
|
}
|
|
this.#cleanupDecompressors()
|
|
}
|
|
super.onResponseError(controller, err)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a decompression interceptor for HTTP responses
|
|
* @param {DecompressHandlerOptions} [options] - Options for the interceptor
|
|
* @returns {Function} - Interceptor function
|
|
*/
|
|
function createDecompressInterceptor (options = {}) {
|
|
// Emit experimental warning only once
|
|
if (!warningEmitted) {
|
|
process.emitWarning(
|
|
'DecompressInterceptor is experimental and subject to change',
|
|
'ExperimentalWarning'
|
|
)
|
|
warningEmitted = true
|
|
}
|
|
|
|
return (dispatch) => {
|
|
return (opts, handler) => {
|
|
const decompressHandler = new DecompressHandler(handler, options)
|
|
return dispatch(opts, decompressHandler)
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = createDecompressInterceptor
|