2025-04-02 06:50:39 -04:00

214 lines
4.2 KiB
JavaScript

/*!
* destroy
* Copyright(c) 2014 Jonathan Ong
* Copyright(c) 2015-2022 Douglas Christopher Wilson
* MIT Licensed
*/
'use strict';
/**
* Module dependencies.
* @private
*/
var EventEmitter = require('events').EventEmitter;
var ReadStream = require('fs').ReadStream;
var Stream = require('stream');
var Zlib = require('zlib');
/**
* Module exports.
* @public
*/
module.exports = destroy;
/**
* Destroy the given stream, and optionally suppress any future `error` events.
*
* @param {object} stream
* @param {boolean} suppress
* @public
*/
function destroy(stream, suppress) {
if (isFsReadStream(stream)) {
destroyReadStream(stream);
} else if (isZlibStream(stream)) {
destroyZlibStream(stream);
} else if (hasDestroy(stream)) {
stream.destroy();
}
if (isEventEmitter(stream) && suppress) {
stream.removeAllListeners('error');
stream.addListener('error', noop);
}
return stream;
}
/**
* Destroy a ReadStream.
*
* @param {object} stream
* @private
*/
function destroyReadStream(stream) {
stream.destroy();
if (typeof stream.close === 'function') {
// node.js core bug work-around
stream.on('open', onOpenClose);
}
}
/**
* Close a Zlib stream.
*
* Zlib streams below Node.js 4.5.5 have a buggy implementation
* of .close() when zlib encountered an error.
*
* @param {object} stream
* @private
*/
function closeZlibStream(stream) {
if (stream._hadError === true) {
var prop = stream._binding === null ? '_binding' : '_handle';
stream[prop] = {
close: function () {
this[prop] = null;
},
};
}
stream.close();
}
/**
* Destroy a Zlib stream.
*
* Zlib streams don't have a destroy function in Node.js 6. On top of that
* simply calling destroy on a zlib stream in Node.js 8+ will result in a
* memory leak. So until that is fixed, we need to call both close AND destroy.
*
* PR to fix memory leak: https://github.com/nodejs/node/pull/23734
*
* In Node.js 6+8, it's important that destroy is called before close as the
* stream would otherwise emit the error 'zlib binding closed'.
*
* @param {object} stream
* @private
*/
function destroyZlibStream(stream) {
if (typeof stream.destroy === 'function') {
// node.js core bug work-around
// istanbul ignore if: node.js 0.8
if (stream._binding) {
// node.js < 0.10.0
stream.destroy();
if (stream._processing) {
stream._needDrain = true;
stream.once('drain', onDrainClearBinding);
} else {
stream._binding.clear();
}
} else if (
stream._destroy &&
stream._destroy !== Stream.Transform.prototype._destroy
) {
// node.js >= 12, ^11.1.0, ^10.15.1
stream.destroy();
} else if (stream._destroy && typeof stream.close === 'function') {
// node.js 7, 8
stream.destroyed = true;
stream.close();
} else {
// fallback
// istanbul ignore next
stream.destroy();
}
} else if (typeof stream.close === 'function') {
// node.js < 8 fallback
closeZlibStream(stream);
}
}
/**
* Determine if stream has destroy.
* @private
*/
function hasDestroy(stream) {
return stream instanceof Stream && typeof stream.destroy === 'function';
}
/**
* Determine if val is EventEmitter.
* @private
*/
function isEventEmitter(val) {
return val instanceof EventEmitter;
}
/**
* Determine if stream is fs.ReadStream stream.
* @private
*/
function isFsReadStream(stream) {
return stream instanceof ReadStream;
}
/**
* Determine if stream is Zlib stream.
* @private
*/
function isZlibStream(stream) {
return (
stream instanceof Zlib.Gzip ||
stream instanceof Zlib.Gunzip ||
stream instanceof Zlib.Deflate ||
stream instanceof Zlib.DeflateRaw ||
stream instanceof Zlib.Inflate ||
stream instanceof Zlib.InflateRaw ||
stream instanceof Zlib.Unzip
);
}
/**
* No-op function.
* @private
*/
function noop() {}
/**
* On drain handler to clear binding.
* @private
*/
// istanbul ignore next: node.js 0.8
function onDrainClearBinding() {
this._binding.clear();
}
/**
* On open handler to close stream.
* @private
*/
function onOpenClose() {
if (typeof this.fd === 'number') {
// actually close down the fd
this.close();
}
}