1028 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			1028 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | "use strict"; | ||
|  | var __importDefault = (this && this.__importDefault) || function (mod) { | ||
|  |     return (mod && mod.__esModule) ? mod : { "default": mod }; | ||
|  | }; | ||
|  | Object.defineProperty(exports, "__esModule", { value: true }); | ||
|  | exports.Minipass = exports.isWritable = exports.isReadable = exports.isStream = void 0; | ||
|  | const proc = typeof process === 'object' && process | ||
|  |     ? process | ||
|  |     : { | ||
|  |         stdout: null, | ||
|  |         stderr: null, | ||
|  |     }; | ||
|  | const node_events_1 = require("node:events"); | ||
|  | const node_stream_1 = __importDefault(require("node:stream")); | ||
|  | const node_string_decoder_1 = require("node:string_decoder"); | ||
|  | /** | ||
|  |  * Return true if the argument is a Minipass stream, Node stream, or something | ||
|  |  * else that Minipass can interact with. | ||
|  |  */ | ||
|  | const isStream = (s) => !!s && | ||
|  |     typeof s === 'object' && | ||
|  |     (s instanceof Minipass || | ||
|  |         s instanceof node_stream_1.default || | ||
|  |         (0, exports.isReadable)(s) || | ||
|  |         (0, exports.isWritable)(s)); | ||
|  | exports.isStream = isStream; | ||
|  | /** | ||
|  |  * Return true if the argument is a valid {@link Minipass.Readable} | ||
|  |  */ | ||
|  | const isReadable = (s) => !!s && | ||
|  |     typeof s === 'object' && | ||
|  |     s instanceof node_events_1.EventEmitter && | ||
|  |     typeof s.pipe === 'function' && | ||
|  |     // node core Writable streams have a pipe() method, but it throws
 | ||
|  |     s.pipe !== node_stream_1.default.Writable.prototype.pipe; | ||
|  | exports.isReadable = isReadable; | ||
|  | /** | ||
|  |  * Return true if the argument is a valid {@link Minipass.Writable} | ||
|  |  */ | ||
|  | const isWritable = (s) => !!s && | ||
|  |     typeof s === 'object' && | ||
|  |     s instanceof node_events_1.EventEmitter && | ||
|  |     typeof s.write === 'function' && | ||
|  |     typeof s.end === 'function'; | ||
|  | exports.isWritable = isWritable; | ||
|  | const EOF = Symbol('EOF'); | ||
|  | const MAYBE_EMIT_END = Symbol('maybeEmitEnd'); | ||
|  | const EMITTED_END = Symbol('emittedEnd'); | ||
|  | const EMITTING_END = Symbol('emittingEnd'); | ||
|  | const EMITTED_ERROR = Symbol('emittedError'); | ||
|  | const CLOSED = Symbol('closed'); | ||
|  | const READ = Symbol('read'); | ||
|  | const FLUSH = Symbol('flush'); | ||
|  | const FLUSHCHUNK = Symbol('flushChunk'); | ||
|  | const ENCODING = Symbol('encoding'); | ||
|  | const DECODER = Symbol('decoder'); | ||
|  | const FLOWING = Symbol('flowing'); | ||
|  | const PAUSED = Symbol('paused'); | ||
|  | const RESUME = Symbol('resume'); | ||
|  | const BUFFER = Symbol('buffer'); | ||
|  | const PIPES = Symbol('pipes'); | ||
|  | const BUFFERLENGTH = Symbol('bufferLength'); | ||
|  | const BUFFERPUSH = Symbol('bufferPush'); | ||
|  | const BUFFERSHIFT = Symbol('bufferShift'); | ||
|  | const OBJECTMODE = Symbol('objectMode'); | ||
|  | // internal event when stream is destroyed
 | ||
|  | const DESTROYED = Symbol('destroyed'); | ||
|  | // internal event when stream has an error
 | ||
|  | const ERROR = Symbol('error'); | ||
|  | const EMITDATA = Symbol('emitData'); | ||
|  | const EMITEND = Symbol('emitEnd'); | ||
|  | const EMITEND2 = Symbol('emitEnd2'); | ||
|  | const ASYNC = Symbol('async'); | ||
|  | const ABORT = Symbol('abort'); | ||
|  | const ABORTED = Symbol('aborted'); | ||
|  | const SIGNAL = Symbol('signal'); | ||
|  | const DATALISTENERS = Symbol('dataListeners'); | ||
|  | const DISCARDED = Symbol('discarded'); | ||
|  | const defer = (fn) => Promise.resolve().then(fn); | ||
|  | const nodefer = (fn) => fn(); | ||
|  | const isEndish = (ev) => ev === 'end' || ev === 'finish' || ev === 'prefinish'; | ||
|  | const isArrayBufferLike = (b) => b instanceof ArrayBuffer || | ||
|  |     (!!b && | ||
|  |         typeof b === 'object' && | ||
|  |         b.constructor && | ||
|  |         b.constructor.name === 'ArrayBuffer' && | ||
|  |         b.byteLength >= 0); | ||
|  | const isArrayBufferView = (b) => !Buffer.isBuffer(b) && ArrayBuffer.isView(b); | ||
|  | /** | ||
|  |  * Internal class representing a pipe to a destination stream. | ||
|  |  * | ||
|  |  * @internal | ||
|  |  */ | ||
|  | class Pipe { | ||
|  |     src; | ||
|  |     dest; | ||
|  |     opts; | ||
|  |     ondrain; | ||
|  |     constructor(src, dest, opts) { | ||
|  |         this.src = src; | ||
|  |         this.dest = dest; | ||
|  |         this.opts = opts; | ||
|  |         this.ondrain = () => src[RESUME](); | ||
|  |         this.dest.on('drain', this.ondrain); | ||
|  |     } | ||
|  |     unpipe() { | ||
|  |         this.dest.removeListener('drain', this.ondrain); | ||
|  |     } | ||
|  |     // only here for the prototype
 | ||
|  |     /* c8 ignore start */ | ||
|  |     proxyErrors(_er) { } | ||
|  |     /* c8 ignore stop */ | ||
|  |     end() { | ||
|  |         this.unpipe(); | ||
|  |         if (this.opts.end) | ||
|  |             this.dest.end(); | ||
|  |     } | ||
|  | } | ||
|  | /** | ||
|  |  * Internal class representing a pipe to a destination stream where | ||
|  |  * errors are proxied. | ||
|  |  * | ||
|  |  * @internal | ||
|  |  */ | ||
|  | class PipeProxyErrors extends Pipe { | ||
|  |     unpipe() { | ||
|  |         this.src.removeListener('error', this.proxyErrors); | ||
|  |         super.unpipe(); | ||
|  |     } | ||
|  |     constructor(src, dest, opts) { | ||
|  |         super(src, dest, opts); | ||
|  |         this.proxyErrors = er => dest.emit('error', er); | ||
|  |         src.on('error', this.proxyErrors); | ||
|  |     } | ||
|  | } | ||
|  | const isObjectModeOptions = (o) => !!o.objectMode; | ||
|  | const isEncodingOptions = (o) => !o.objectMode && !!o.encoding && o.encoding !== 'buffer'; | ||
|  | /** | ||
|  |  * Main export, the Minipass class | ||
|  |  * | ||
|  |  * `RType` is the type of data emitted, defaults to Buffer | ||
|  |  * | ||
|  |  * `WType` is the type of data to be written, if RType is buffer or string, | ||
|  |  * then any {@link Minipass.ContiguousData} is allowed. | ||
|  |  * | ||
|  |  * `Events` is the set of event handler signatures that this object | ||
|  |  * will emit, see {@link Minipass.Events} | ||
|  |  */ | ||
|  | class Minipass extends node_events_1.EventEmitter { | ||
|  |     [FLOWING] = false; | ||
|  |     [PAUSED] = false; | ||
|  |     [PIPES] = []; | ||
|  |     [BUFFER] = []; | ||
|  |     [OBJECTMODE]; | ||
|  |     [ENCODING]; | ||
|  |     [ASYNC]; | ||
|  |     [DECODER]; | ||
|  |     [EOF] = false; | ||
|  |     [EMITTED_END] = false; | ||
|  |     [EMITTING_END] = false; | ||
|  |     [CLOSED] = false; | ||
|  |     [EMITTED_ERROR] = null; | ||
|  |     [BUFFERLENGTH] = 0; | ||
|  |     [DESTROYED] = false; | ||
|  |     [SIGNAL]; | ||
|  |     [ABORTED] = false; | ||
|  |     [DATALISTENERS] = 0; | ||
|  |     [DISCARDED] = false; | ||
|  |     /** | ||
|  |      * true if the stream can be written | ||
|  |      */ | ||
|  |     writable = true; | ||
|  |     /** | ||
|  |      * true if the stream can be read | ||
|  |      */ | ||
|  |     readable = true; | ||
|  |     /** | ||
|  |      * If `RType` is Buffer, then options do not need to be provided. | ||
|  |      * Otherwise, an options object must be provided to specify either | ||
|  |      * {@link Minipass.SharedOptions.objectMode} or | ||
|  |      * {@link Minipass.SharedOptions.encoding}, as appropriate. | ||
|  |      */ | ||
|  |     constructor(...args) { | ||
|  |         const options = (args[0] || | ||
|  |             {}); | ||
|  |         super(); | ||
|  |         if (options.objectMode && typeof options.encoding === 'string') { | ||
|  |             throw new TypeError('Encoding and objectMode may not be used together'); | ||
|  |         } | ||
|  |         if (isObjectModeOptions(options)) { | ||
|  |             this[OBJECTMODE] = true; | ||
|  |             this[ENCODING] = null; | ||
|  |         } | ||
|  |         else if (isEncodingOptions(options)) { | ||
|  |             this[ENCODING] = options.encoding; | ||
|  |             this[OBJECTMODE] = false; | ||
|  |         } | ||
|  |         else { | ||
|  |             this[OBJECTMODE] = false; | ||
|  |             this[ENCODING] = null; | ||
|  |         } | ||
|  |         this[ASYNC] = !!options.async; | ||
|  |         this[DECODER] = this[ENCODING] | ||
|  |             ? new node_string_decoder_1.StringDecoder(this[ENCODING]) | ||
|  |             : null; | ||
|  |         //@ts-ignore - private option for debugging and testing
 | ||
|  |         if (options && options.debugExposeBuffer === true) { | ||
|  |             Object.defineProperty(this, 'buffer', { get: () => this[BUFFER] }); | ||
|  |         } | ||
|  |         //@ts-ignore - private option for debugging and testing
 | ||
|  |         if (options && options.debugExposePipes === true) { | ||
|  |             Object.defineProperty(this, 'pipes', { get: () => this[PIPES] }); | ||
|  |         } | ||
|  |         const { signal } = options; | ||
|  |         if (signal) { | ||
|  |             this[SIGNAL] = signal; | ||
|  |             if (signal.aborted) { | ||
|  |                 this[ABORT](); | ||
|  |             } | ||
|  |             else { | ||
|  |                 signal.addEventListener('abort', () => this[ABORT]()); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  |     /** | ||
|  |      * The amount of data stored in the buffer waiting to be read. | ||
|  |      * | ||
|  |      * For Buffer strings, this will be the total byte length. | ||
|  |      * For string encoding streams, this will be the string character length, | ||
|  |      * according to JavaScript's `string.length` logic. | ||
|  |      * For objectMode streams, this is a count of the items waiting to be | ||
|  |      * emitted. | ||
|  |      */ | ||
|  |     get bufferLength() { | ||
|  |         return this[BUFFERLENGTH]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * The `BufferEncoding` currently in use, or `null` | ||
|  |      */ | ||
|  |     get encoding() { | ||
|  |         return this[ENCODING]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * @deprecated - This is a read only property | ||
|  |      */ | ||
|  |     set encoding(_enc) { | ||
|  |         throw new Error('Encoding must be set at instantiation time'); | ||
|  |     } | ||
|  |     /** | ||
|  |      * @deprecated - Encoding may only be set at instantiation time | ||
|  |      */ | ||
|  |     setEncoding(_enc) { | ||
|  |         throw new Error('Encoding must be set at instantiation time'); | ||
|  |     } | ||
|  |     /** | ||
|  |      * True if this is an objectMode stream | ||
|  |      */ | ||
|  |     get objectMode() { | ||
|  |         return this[OBJECTMODE]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * @deprecated - This is a read-only property | ||
|  |      */ | ||
|  |     set objectMode(_om) { | ||
|  |         throw new Error('objectMode must be set at instantiation time'); | ||
|  |     } | ||
|  |     /** | ||
|  |      * true if this is an async stream | ||
|  |      */ | ||
|  |     get ['async']() { | ||
|  |         return this[ASYNC]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Set to true to make this stream async. | ||
|  |      * | ||
|  |      * Once set, it cannot be unset, as this would potentially cause incorrect | ||
|  |      * behavior.  Ie, a sync stream can be made async, but an async stream | ||
|  |      * cannot be safely made sync. | ||
|  |      */ | ||
|  |     set ['async'](a) { | ||
|  |         this[ASYNC] = this[ASYNC] || !!a; | ||
|  |     } | ||
|  |     // drop everything and get out of the flow completely
 | ||
|  |     [ABORT]() { | ||
|  |         this[ABORTED] = true; | ||
|  |         this.emit('abort', this[SIGNAL]?.reason); | ||
|  |         this.destroy(this[SIGNAL]?.reason); | ||
|  |     } | ||
|  |     /** | ||
|  |      * True if the stream has been aborted. | ||
|  |      */ | ||
|  |     get aborted() { | ||
|  |         return this[ABORTED]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * No-op setter. Stream aborted status is set via the AbortSignal provided | ||
|  |      * in the constructor options. | ||
|  |      */ | ||
|  |     set aborted(_) { } | ||
|  |     write(chunk, encoding, cb) { | ||
|  |         if (this[ABORTED]) | ||
|  |             return false; | ||
|  |         if (this[EOF]) | ||
|  |             throw new Error('write after end'); | ||
|  |         if (this[DESTROYED]) { | ||
|  |             this.emit('error', Object.assign(new Error('Cannot call write after a stream was destroyed'), { code: 'ERR_STREAM_DESTROYED' })); | ||
|  |             return true; | ||
|  |         } | ||
|  |         if (typeof encoding === 'function') { | ||
|  |             cb = encoding; | ||
|  |             encoding = 'utf8'; | ||
|  |         } | ||
|  |         if (!encoding) | ||
|  |             encoding = 'utf8'; | ||
|  |         const fn = this[ASYNC] ? defer : nodefer; | ||
|  |         // convert array buffers and typed array views into buffers
 | ||
|  |         // at some point in the future, we may want to do the opposite!
 | ||
|  |         // leave strings and buffers as-is
 | ||
|  |         // anything is only allowed if in object mode, so throw
 | ||
|  |         if (!this[OBJECTMODE] && !Buffer.isBuffer(chunk)) { | ||
|  |             if (isArrayBufferView(chunk)) { | ||
|  |                 //@ts-ignore - sinful unsafe type changing
 | ||
|  |                 chunk = Buffer.from(chunk.buffer, chunk.byteOffset, chunk.byteLength); | ||
|  |             } | ||
|  |             else if (isArrayBufferLike(chunk)) { | ||
|  |                 //@ts-ignore - sinful unsafe type changing
 | ||
|  |                 chunk = Buffer.from(chunk); | ||
|  |             } | ||
|  |             else if (typeof chunk !== 'string') { | ||
|  |                 throw new Error('Non-contiguous data written to non-objectMode stream'); | ||
|  |             } | ||
|  |         } | ||
|  |         // handle object mode up front, since it's simpler
 | ||
|  |         // this yields better performance, fewer checks later.
 | ||
|  |         if (this[OBJECTMODE]) { | ||
|  |             // maybe impossible?
 | ||
|  |             /* c8 ignore start */ | ||
|  |             if (this[FLOWING] && this[BUFFERLENGTH] !== 0) | ||
|  |                 this[FLUSH](true); | ||
|  |             /* c8 ignore stop */ | ||
|  |             if (this[FLOWING]) | ||
|  |                 this.emit('data', chunk); | ||
|  |             else | ||
|  |                 this[BUFFERPUSH](chunk); | ||
|  |             if (this[BUFFERLENGTH] !== 0) | ||
|  |                 this.emit('readable'); | ||
|  |             if (cb) | ||
|  |                 fn(cb); | ||
|  |             return this[FLOWING]; | ||
|  |         } | ||
|  |         // at this point the chunk is a buffer or string
 | ||
|  |         // don't buffer it up or send it to the decoder
 | ||
|  |         if (!chunk.length) { | ||
|  |             if (this[BUFFERLENGTH] !== 0) | ||
|  |                 this.emit('readable'); | ||
|  |             if (cb) | ||
|  |                 fn(cb); | ||
|  |             return this[FLOWING]; | ||
|  |         } | ||
|  |         // fast-path writing strings of same encoding to a stream with
 | ||
|  |         // an empty buffer, skipping the buffer/decoder dance
 | ||
|  |         if (typeof chunk === 'string' && | ||
|  |             // unless it is a string already ready for us to use
 | ||
|  |             !(encoding === this[ENCODING] && !this[DECODER]?.lastNeed)) { | ||
|  |             //@ts-ignore - sinful unsafe type change
 | ||
|  |             chunk = Buffer.from(chunk, encoding); | ||
|  |         } | ||
|  |         if (Buffer.isBuffer(chunk) && this[ENCODING]) { | ||
|  |             //@ts-ignore - sinful unsafe type change
 | ||
|  |             chunk = this[DECODER].write(chunk); | ||
|  |         } | ||
|  |         // Note: flushing CAN potentially switch us into not-flowing mode
 | ||
|  |         if (this[FLOWING] && this[BUFFERLENGTH] !== 0) | ||
|  |             this[FLUSH](true); | ||
|  |         if (this[FLOWING]) | ||
|  |             this.emit('data', chunk); | ||
|  |         else | ||
|  |             this[BUFFERPUSH](chunk); | ||
|  |         if (this[BUFFERLENGTH] !== 0) | ||
|  |             this.emit('readable'); | ||
|  |         if (cb) | ||
|  |             fn(cb); | ||
|  |         return this[FLOWING]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Low-level explicit read method. | ||
|  |      * | ||
|  |      * In objectMode, the argument is ignored, and one item is returned if | ||
|  |      * available. | ||
|  |      * | ||
|  |      * `n` is the number of bytes (or in the case of encoding streams, | ||
|  |      * characters) to consume. If `n` is not provided, then the entire buffer | ||
|  |      * is returned, or `null` is returned if no data is available. | ||
|  |      * | ||
|  |      * If `n` is greater that the amount of data in the internal buffer, | ||
|  |      * then `null` is returned. | ||
|  |      */ | ||
|  |     read(n) { | ||
|  |         if (this[DESTROYED]) | ||
|  |             return null; | ||
|  |         this[DISCARDED] = false; | ||
|  |         if (this[BUFFERLENGTH] === 0 || | ||
|  |             n === 0 || | ||
|  |             (n && n > this[BUFFERLENGTH])) { | ||
|  |             this[MAYBE_EMIT_END](); | ||
|  |             return null; | ||
|  |         } | ||
|  |         if (this[OBJECTMODE]) | ||
|  |             n = null; | ||
|  |         if (this[BUFFER].length > 1 && !this[OBJECTMODE]) { | ||
|  |             // not object mode, so if we have an encoding, then RType is string
 | ||
|  |             // otherwise, must be Buffer
 | ||
|  |             this[BUFFER] = [ | ||
|  |                 (this[ENCODING] | ||
|  |                     ? this[BUFFER].join('') | ||
|  |                     : Buffer.concat(this[BUFFER], this[BUFFERLENGTH])), | ||
|  |             ]; | ||
|  |         } | ||
|  |         const ret = this[READ](n || null, this[BUFFER][0]); | ||
|  |         this[MAYBE_EMIT_END](); | ||
|  |         return ret; | ||
|  |     } | ||
|  |     [READ](n, chunk) { | ||
|  |         if (this[OBJECTMODE]) | ||
|  |             this[BUFFERSHIFT](); | ||
|  |         else { | ||
|  |             const c = chunk; | ||
|  |             if (n === c.length || n === null) | ||
|  |                 this[BUFFERSHIFT](); | ||
|  |             else if (typeof c === 'string') { | ||
|  |                 this[BUFFER][0] = c.slice(n); | ||
|  |                 chunk = c.slice(0, n); | ||
|  |                 this[BUFFERLENGTH] -= n; | ||
|  |             } | ||
|  |             else { | ||
|  |                 this[BUFFER][0] = c.subarray(n); | ||
|  |                 chunk = c.subarray(0, n); | ||
|  |                 this[BUFFERLENGTH] -= n; | ||
|  |             } | ||
|  |         } | ||
|  |         this.emit('data', chunk); | ||
|  |         if (!this[BUFFER].length && !this[EOF]) | ||
|  |             this.emit('drain'); | ||
|  |         return chunk; | ||
|  |     } | ||
|  |     end(chunk, encoding, cb) { | ||
|  |         if (typeof chunk === 'function') { | ||
|  |             cb = chunk; | ||
|  |             chunk = undefined; | ||
|  |         } | ||
|  |         if (typeof encoding === 'function') { | ||
|  |             cb = encoding; | ||
|  |             encoding = 'utf8'; | ||
|  |         } | ||
|  |         if (chunk !== undefined) | ||
|  |             this.write(chunk, encoding); | ||
|  |         if (cb) | ||
|  |             this.once('end', cb); | ||
|  |         this[EOF] = true; | ||
|  |         this.writable = false; | ||
|  |         // if we haven't written anything, then go ahead and emit,
 | ||
|  |         // even if we're not reading.
 | ||
|  |         // we'll re-emit if a new 'end' listener is added anyway.
 | ||
|  |         // This makes MP more suitable to write-only use cases.
 | ||
|  |         if (this[FLOWING] || !this[PAUSED]) | ||
|  |             this[MAYBE_EMIT_END](); | ||
|  |         return this; | ||
|  |     } | ||
|  |     // don't let the internal resume be overwritten
 | ||
|  |     [RESUME]() { | ||
|  |         if (this[DESTROYED]) | ||
|  |             return; | ||
|  |         if (!this[DATALISTENERS] && !this[PIPES].length) { | ||
|  |             this[DISCARDED] = true; | ||
|  |         } | ||
|  |         this[PAUSED] = false; | ||
|  |         this[FLOWING] = true; | ||
|  |         this.emit('resume'); | ||
|  |         if (this[BUFFER].length) | ||
|  |             this[FLUSH](); | ||
|  |         else if (this[EOF]) | ||
|  |             this[MAYBE_EMIT_END](); | ||
|  |         else | ||
|  |             this.emit('drain'); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Resume the stream if it is currently in a paused state | ||
|  |      * | ||
|  |      * If called when there are no pipe destinations or `data` event listeners, | ||
|  |      * this will place the stream in a "discarded" state, where all data will | ||
|  |      * be thrown away. The discarded state is removed if a pipe destination or | ||
|  |      * data handler is added, if pause() is called, or if any synchronous or | ||
|  |      * asynchronous iteration is started. | ||
|  |      */ | ||
|  |     resume() { | ||
|  |         return this[RESUME](); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Pause the stream | ||
|  |      */ | ||
|  |     pause() { | ||
|  |         this[FLOWING] = false; | ||
|  |         this[PAUSED] = true; | ||
|  |         this[DISCARDED] = false; | ||
|  |     } | ||
|  |     /** | ||
|  |      * true if the stream has been forcibly destroyed | ||
|  |      */ | ||
|  |     get destroyed() { | ||
|  |         return this[DESTROYED]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * true if the stream is currently in a flowing state, meaning that | ||
|  |      * any writes will be immediately emitted. | ||
|  |      */ | ||
|  |     get flowing() { | ||
|  |         return this[FLOWING]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * true if the stream is currently in a paused state | ||
|  |      */ | ||
|  |     get paused() { | ||
|  |         return this[PAUSED]; | ||
|  |     } | ||
|  |     [BUFFERPUSH](chunk) { | ||
|  |         if (this[OBJECTMODE]) | ||
|  |             this[BUFFERLENGTH] += 1; | ||
|  |         else | ||
|  |             this[BUFFERLENGTH] += chunk.length; | ||
|  |         this[BUFFER].push(chunk); | ||
|  |     } | ||
|  |     [BUFFERSHIFT]() { | ||
|  |         if (this[OBJECTMODE]) | ||
|  |             this[BUFFERLENGTH] -= 1; | ||
|  |         else | ||
|  |             this[BUFFERLENGTH] -= this[BUFFER][0].length; | ||
|  |         return this[BUFFER].shift(); | ||
|  |     } | ||
|  |     [FLUSH](noDrain = false) { | ||
|  |         do { } while (this[FLUSHCHUNK](this[BUFFERSHIFT]()) && | ||
|  |             this[BUFFER].length); | ||
|  |         if (!noDrain && !this[BUFFER].length && !this[EOF]) | ||
|  |             this.emit('drain'); | ||
|  |     } | ||
|  |     [FLUSHCHUNK](chunk) { | ||
|  |         this.emit('data', chunk); | ||
|  |         return this[FLOWING]; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Pipe all data emitted by this stream into the destination provided. | ||
|  |      * | ||
|  |      * Triggers the flow of data. | ||
|  |      */ | ||
|  |     pipe(dest, opts) { | ||
|  |         if (this[DESTROYED]) | ||
|  |             return dest; | ||
|  |         this[DISCARDED] = false; | ||
|  |         const ended = this[EMITTED_END]; | ||
|  |         opts = opts || {}; | ||
|  |         if (dest === proc.stdout || dest === proc.stderr) | ||
|  |             opts.end = false; | ||
|  |         else | ||
|  |             opts.end = opts.end !== false; | ||
|  |         opts.proxyErrors = !!opts.proxyErrors; | ||
|  |         // piping an ended stream ends immediately
 | ||
|  |         if (ended) { | ||
|  |             if (opts.end) | ||
|  |                 dest.end(); | ||
|  |         } | ||
|  |         else { | ||
|  |             // "as" here just ignores the WType, which pipes don't care about,
 | ||
|  |             // since they're only consuming from us, and writing to the dest
 | ||
|  |             this[PIPES].push(!opts.proxyErrors | ||
|  |                 ? new Pipe(this, dest, opts) | ||
|  |                 : new PipeProxyErrors(this, dest, opts)); | ||
|  |             if (this[ASYNC]) | ||
|  |                 defer(() => this[RESUME]()); | ||
|  |             else | ||
|  |                 this[RESUME](); | ||
|  |         } | ||
|  |         return dest; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Fully unhook a piped destination stream. | ||
|  |      * | ||
|  |      * If the destination stream was the only consumer of this stream (ie, | ||
|  |      * there are no other piped destinations or `'data'` event listeners) | ||
|  |      * then the flow of data will stop until there is another consumer or | ||
|  |      * {@link Minipass#resume} is explicitly called. | ||
|  |      */ | ||
|  |     unpipe(dest) { | ||
|  |         const p = this[PIPES].find(p => p.dest === dest); | ||
|  |         if (p) { | ||
|  |             if (this[PIPES].length === 1) { | ||
|  |                 if (this[FLOWING] && this[DATALISTENERS] === 0) { | ||
|  |                     this[FLOWING] = false; | ||
|  |                 } | ||
|  |                 this[PIPES] = []; | ||
|  |             } | ||
|  |             else | ||
|  |                 this[PIPES].splice(this[PIPES].indexOf(p), 1); | ||
|  |             p.unpipe(); | ||
|  |         } | ||
|  |     } | ||
|  |     /** | ||
|  |      * Alias for {@link Minipass#on} | ||
|  |      */ | ||
|  |     addListener(ev, handler) { | ||
|  |         return this.on(ev, handler); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Mostly identical to `EventEmitter.on`, with the following | ||
|  |      * behavior differences to prevent data loss and unnecessary hangs: | ||
|  |      * | ||
|  |      * - Adding a 'data' event handler will trigger the flow of data | ||
|  |      * | ||
|  |      * - Adding a 'readable' event handler when there is data waiting to be read | ||
|  |      *   will cause 'readable' to be emitted immediately. | ||
|  |      * | ||
|  |      * - Adding an 'endish' event handler ('end', 'finish', etc.) which has | ||
|  |      *   already passed will cause the event to be emitted immediately and all | ||
|  |      *   handlers removed. | ||
|  |      * | ||
|  |      * - Adding an 'error' event handler after an error has been emitted will | ||
|  |      *   cause the event to be re-emitted immediately with the error previously | ||
|  |      *   raised. | ||
|  |      */ | ||
|  |     on(ev, handler) { | ||
|  |         const ret = super.on(ev, handler); | ||
|  |         if (ev === 'data') { | ||
|  |             this[DISCARDED] = false; | ||
|  |             this[DATALISTENERS]++; | ||
|  |             if (!this[PIPES].length && !this[FLOWING]) { | ||
|  |                 this[RESUME](); | ||
|  |             } | ||
|  |         } | ||
|  |         else if (ev === 'readable' && this[BUFFERLENGTH] !== 0) { | ||
|  |             super.emit('readable'); | ||
|  |         } | ||
|  |         else if (isEndish(ev) && this[EMITTED_END]) { | ||
|  |             super.emit(ev); | ||
|  |             this.removeAllListeners(ev); | ||
|  |         } | ||
|  |         else if (ev === 'error' && this[EMITTED_ERROR]) { | ||
|  |             const h = handler; | ||
|  |             if (this[ASYNC]) | ||
|  |                 defer(() => h.call(this, this[EMITTED_ERROR])); | ||
|  |             else | ||
|  |                 h.call(this, this[EMITTED_ERROR]); | ||
|  |         } | ||
|  |         return ret; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Alias for {@link Minipass#off} | ||
|  |      */ | ||
|  |     removeListener(ev, handler) { | ||
|  |         return this.off(ev, handler); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Mostly identical to `EventEmitter.off` | ||
|  |      * | ||
|  |      * If a 'data' event handler is removed, and it was the last consumer | ||
|  |      * (ie, there are no pipe destinations or other 'data' event listeners), | ||
|  |      * then the flow of data will stop until there is another consumer or | ||
|  |      * {@link Minipass#resume} is explicitly called. | ||
|  |      */ | ||
|  |     off(ev, handler) { | ||
|  |         const ret = super.off(ev, handler); | ||
|  |         // if we previously had listeners, and now we don't, and we don't
 | ||
|  |         // have any pipes, then stop the flow, unless it's been explicitly
 | ||
|  |         // put in a discarded flowing state via stream.resume().
 | ||
|  |         if (ev === 'data') { | ||
|  |             this[DATALISTENERS] = this.listeners('data').length; | ||
|  |             if (this[DATALISTENERS] === 0 && | ||
|  |                 !this[DISCARDED] && | ||
|  |                 !this[PIPES].length) { | ||
|  |                 this[FLOWING] = false; | ||
|  |             } | ||
|  |         } | ||
|  |         return ret; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Mostly identical to `EventEmitter.removeAllListeners` | ||
|  |      * | ||
|  |      * If all 'data' event handlers are removed, and they were the last consumer | ||
|  |      * (ie, there are no pipe destinations), then the flow of data will stop | ||
|  |      * until there is another consumer or {@link Minipass#resume} is explicitly | ||
|  |      * called. | ||
|  |      */ | ||
|  |     removeAllListeners(ev) { | ||
|  |         const ret = super.removeAllListeners(ev); | ||
|  |         if (ev === 'data' || ev === undefined) { | ||
|  |             this[DATALISTENERS] = 0; | ||
|  |             if (!this[DISCARDED] && !this[PIPES].length) { | ||
|  |                 this[FLOWING] = false; | ||
|  |             } | ||
|  |         } | ||
|  |         return ret; | ||
|  |     } | ||
|  |     /** | ||
|  |      * true if the 'end' event has been emitted | ||
|  |      */ | ||
|  |     get emittedEnd() { | ||
|  |         return this[EMITTED_END]; | ||
|  |     } | ||
|  |     [MAYBE_EMIT_END]() { | ||
|  |         if (!this[EMITTING_END] && | ||
|  |             !this[EMITTED_END] && | ||
|  |             !this[DESTROYED] && | ||
|  |             this[BUFFER].length === 0 && | ||
|  |             this[EOF]) { | ||
|  |             this[EMITTING_END] = true; | ||
|  |             this.emit('end'); | ||
|  |             this.emit('prefinish'); | ||
|  |             this.emit('finish'); | ||
|  |             if (this[CLOSED]) | ||
|  |                 this.emit('close'); | ||
|  |             this[EMITTING_END] = false; | ||
|  |         } | ||
|  |     } | ||
|  |     /** | ||
|  |      * Mostly identical to `EventEmitter.emit`, with the following | ||
|  |      * behavior differences to prevent data loss and unnecessary hangs: | ||
|  |      * | ||
|  |      * If the stream has been destroyed, and the event is something other | ||
|  |      * than 'close' or 'error', then `false` is returned and no handlers | ||
|  |      * are called. | ||
|  |      * | ||
|  |      * If the event is 'end', and has already been emitted, then the event | ||
|  |      * is ignored. If the stream is in a paused or non-flowing state, then | ||
|  |      * the event will be deferred until data flow resumes. If the stream is | ||
|  |      * async, then handlers will be called on the next tick rather than | ||
|  |      * immediately. | ||
|  |      * | ||
|  |      * If the event is 'close', and 'end' has not yet been emitted, then | ||
|  |      * the event will be deferred until after 'end' is emitted. | ||
|  |      * | ||
|  |      * If the event is 'error', and an AbortSignal was provided for the stream, | ||
|  |      * and there are no listeners, then the event is ignored, matching the | ||
|  |      * behavior of node core streams in the presense of an AbortSignal. | ||
|  |      * | ||
|  |      * If the event is 'finish' or 'prefinish', then all listeners will be | ||
|  |      * removed after emitting the event, to prevent double-firing. | ||
|  |      */ | ||
|  |     emit(ev, ...args) { | ||
|  |         const data = args[0]; | ||
|  |         // error and close are only events allowed after calling destroy()
 | ||
|  |         if (ev !== 'error' && | ||
|  |             ev !== 'close' && | ||
|  |             ev !== DESTROYED && | ||
|  |             this[DESTROYED]) { | ||
|  |             return false; | ||
|  |         } | ||
|  |         else if (ev === 'data') { | ||
|  |             return !this[OBJECTMODE] && !data | ||
|  |                 ? false | ||
|  |                 : this[ASYNC] | ||
|  |                     ? (defer(() => this[EMITDATA](data)), true) | ||
|  |                     : this[EMITDATA](data); | ||
|  |         } | ||
|  |         else if (ev === 'end') { | ||
|  |             return this[EMITEND](); | ||
|  |         } | ||
|  |         else if (ev === 'close') { | ||
|  |             this[CLOSED] = true; | ||
|  |             // don't emit close before 'end' and 'finish'
 | ||
|  |             if (!this[EMITTED_END] && !this[DESTROYED]) | ||
|  |                 return false; | ||
|  |             const ret = super.emit('close'); | ||
|  |             this.removeAllListeners('close'); | ||
|  |             return ret; | ||
|  |         } | ||
|  |         else if (ev === 'error') { | ||
|  |             this[EMITTED_ERROR] = data; | ||
|  |             super.emit(ERROR, data); | ||
|  |             const ret = !this[SIGNAL] || this.listeners('error').length | ||
|  |                 ? super.emit('error', data) | ||
|  |                 : false; | ||
|  |             this[MAYBE_EMIT_END](); | ||
|  |             return ret; | ||
|  |         } | ||
|  |         else if (ev === 'resume') { | ||
|  |             const ret = super.emit('resume'); | ||
|  |             this[MAYBE_EMIT_END](); | ||
|  |             return ret; | ||
|  |         } | ||
|  |         else if (ev === 'finish' || ev === 'prefinish') { | ||
|  |             const ret = super.emit(ev); | ||
|  |             this.removeAllListeners(ev); | ||
|  |             return ret; | ||
|  |         } | ||
|  |         // Some other unknown event
 | ||
|  |         const ret = super.emit(ev, ...args); | ||
|  |         this[MAYBE_EMIT_END](); | ||
|  |         return ret; | ||
|  |     } | ||
|  |     [EMITDATA](data) { | ||
|  |         for (const p of this[PIPES]) { | ||
|  |             if (p.dest.write(data) === false) | ||
|  |                 this.pause(); | ||
|  |         } | ||
|  |         const ret = this[DISCARDED] ? false : super.emit('data', data); | ||
|  |         this[MAYBE_EMIT_END](); | ||
|  |         return ret; | ||
|  |     } | ||
|  |     [EMITEND]() { | ||
|  |         if (this[EMITTED_END]) | ||
|  |             return false; | ||
|  |         this[EMITTED_END] = true; | ||
|  |         this.readable = false; | ||
|  |         return this[ASYNC] | ||
|  |             ? (defer(() => this[EMITEND2]()), true) | ||
|  |             : this[EMITEND2](); | ||
|  |     } | ||
|  |     [EMITEND2]() { | ||
|  |         if (this[DECODER]) { | ||
|  |             const data = this[DECODER].end(); | ||
|  |             if (data) { | ||
|  |                 for (const p of this[PIPES]) { | ||
|  |                     p.dest.write(data); | ||
|  |                 } | ||
|  |                 if (!this[DISCARDED]) | ||
|  |                     super.emit('data', data); | ||
|  |             } | ||
|  |         } | ||
|  |         for (const p of this[PIPES]) { | ||
|  |             p.end(); | ||
|  |         } | ||
|  |         const ret = super.emit('end'); | ||
|  |         this.removeAllListeners('end'); | ||
|  |         return ret; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Return a Promise that resolves to an array of all emitted data once | ||
|  |      * the stream ends. | ||
|  |      */ | ||
|  |     async collect() { | ||
|  |         const buf = Object.assign([], { | ||
|  |             dataLength: 0, | ||
|  |         }); | ||
|  |         if (!this[OBJECTMODE]) | ||
|  |             buf.dataLength = 0; | ||
|  |         // set the promise first, in case an error is raised
 | ||
|  |         // by triggering the flow here.
 | ||
|  |         const p = this.promise(); | ||
|  |         this.on('data', c => { | ||
|  |             buf.push(c); | ||
|  |             if (!this[OBJECTMODE]) | ||
|  |                 buf.dataLength += c.length; | ||
|  |         }); | ||
|  |         await p; | ||
|  |         return buf; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Return a Promise that resolves to the concatenation of all emitted data | ||
|  |      * once the stream ends. | ||
|  |      * | ||
|  |      * Not allowed on objectMode streams. | ||
|  |      */ | ||
|  |     async concat() { | ||
|  |         if (this[OBJECTMODE]) { | ||
|  |             throw new Error('cannot concat in objectMode'); | ||
|  |         } | ||
|  |         const buf = await this.collect(); | ||
|  |         return (this[ENCODING] | ||
|  |             ? buf.join('') | ||
|  |             : Buffer.concat(buf, buf.dataLength)); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Return a void Promise that resolves once the stream ends. | ||
|  |      */ | ||
|  |     async promise() { | ||
|  |         return new Promise((resolve, reject) => { | ||
|  |             this.on(DESTROYED, () => reject(new Error('stream destroyed'))); | ||
|  |             this.on('error', er => reject(er)); | ||
|  |             this.on('end', () => resolve()); | ||
|  |         }); | ||
|  |     } | ||
|  |     /** | ||
|  |      * Asynchronous `for await of` iteration. | ||
|  |      * | ||
|  |      * This will continue emitting all chunks until the stream terminates. | ||
|  |      */ | ||
|  |     [Symbol.asyncIterator]() { | ||
|  |         // set this up front, in case the consumer doesn't call next()
 | ||
|  |         // right away.
 | ||
|  |         this[DISCARDED] = false; | ||
|  |         let stopped = false; | ||
|  |         const stop = async () => { | ||
|  |             this.pause(); | ||
|  |             stopped = true; | ||
|  |             return { value: undefined, done: true }; | ||
|  |         }; | ||
|  |         const next = () => { | ||
|  |             if (stopped) | ||
|  |                 return stop(); | ||
|  |             const res = this.read(); | ||
|  |             if (res !== null) | ||
|  |                 return Promise.resolve({ done: false, value: res }); | ||
|  |             if (this[EOF]) | ||
|  |                 return stop(); | ||
|  |             let resolve; | ||
|  |             let reject; | ||
|  |             const onerr = (er) => { | ||
|  |                 this.off('data', ondata); | ||
|  |                 this.off('end', onend); | ||
|  |                 this.off(DESTROYED, ondestroy); | ||
|  |                 stop(); | ||
|  |                 reject(er); | ||
|  |             }; | ||
|  |             const ondata = (value) => { | ||
|  |                 this.off('error', onerr); | ||
|  |                 this.off('end', onend); | ||
|  |                 this.off(DESTROYED, ondestroy); | ||
|  |                 this.pause(); | ||
|  |                 resolve({ value, done: !!this[EOF] }); | ||
|  |             }; | ||
|  |             const onend = () => { | ||
|  |                 this.off('error', onerr); | ||
|  |                 this.off('data', ondata); | ||
|  |                 this.off(DESTROYED, ondestroy); | ||
|  |                 stop(); | ||
|  |                 resolve({ done: true, value: undefined }); | ||
|  |             }; | ||
|  |             const ondestroy = () => onerr(new Error('stream destroyed')); | ||
|  |             return new Promise((res, rej) => { | ||
|  |                 reject = rej; | ||
|  |                 resolve = res; | ||
|  |                 this.once(DESTROYED, ondestroy); | ||
|  |                 this.once('error', onerr); | ||
|  |                 this.once('end', onend); | ||
|  |                 this.once('data', ondata); | ||
|  |             }); | ||
|  |         }; | ||
|  |         return { | ||
|  |             next, | ||
|  |             throw: stop, | ||
|  |             return: stop, | ||
|  |             [Symbol.asyncIterator]() { | ||
|  |                 return this; | ||
|  |             }, | ||
|  |         }; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Synchronous `for of` iteration. | ||
|  |      * | ||
|  |      * The iteration will terminate when the internal buffer runs out, even | ||
|  |      * if the stream has not yet terminated. | ||
|  |      */ | ||
|  |     [Symbol.iterator]() { | ||
|  |         // set this up front, in case the consumer doesn't call next()
 | ||
|  |         // right away.
 | ||
|  |         this[DISCARDED] = false; | ||
|  |         let stopped = false; | ||
|  |         const stop = () => { | ||
|  |             this.pause(); | ||
|  |             this.off(ERROR, stop); | ||
|  |             this.off(DESTROYED, stop); | ||
|  |             this.off('end', stop); | ||
|  |             stopped = true; | ||
|  |             return { done: true, value: undefined }; | ||
|  |         }; | ||
|  |         const next = () => { | ||
|  |             if (stopped) | ||
|  |                 return stop(); | ||
|  |             const value = this.read(); | ||
|  |             return value === null ? stop() : { done: false, value }; | ||
|  |         }; | ||
|  |         this.once('end', stop); | ||
|  |         this.once(ERROR, stop); | ||
|  |         this.once(DESTROYED, stop); | ||
|  |         return { | ||
|  |             next, | ||
|  |             throw: stop, | ||
|  |             return: stop, | ||
|  |             [Symbol.iterator]() { | ||
|  |                 return this; | ||
|  |             }, | ||
|  |         }; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Destroy a stream, preventing it from being used for any further purpose. | ||
|  |      * | ||
|  |      * If the stream has a `close()` method, then it will be called on | ||
|  |      * destruction. | ||
|  |      * | ||
|  |      * After destruction, any attempt to write data, read data, or emit most | ||
|  |      * events will be ignored. | ||
|  |      * | ||
|  |      * If an error argument is provided, then it will be emitted in an | ||
|  |      * 'error' event. | ||
|  |      */ | ||
|  |     destroy(er) { | ||
|  |         if (this[DESTROYED]) { | ||
|  |             if (er) | ||
|  |                 this.emit('error', er); | ||
|  |             else | ||
|  |                 this.emit(DESTROYED); | ||
|  |             return this; | ||
|  |         } | ||
|  |         this[DESTROYED] = true; | ||
|  |         this[DISCARDED] = true; | ||
|  |         // throw away all buffered data, it's never coming out
 | ||
|  |         this[BUFFER].length = 0; | ||
|  |         this[BUFFERLENGTH] = 0; | ||
|  |         const wc = this; | ||
|  |         if (typeof wc.close === 'function' && !this[CLOSED]) | ||
|  |             wc.close(); | ||
|  |         if (er) | ||
|  |             this.emit('error', er); | ||
|  |         // if no error to emit, still reject pending promises
 | ||
|  |         else | ||
|  |             this.emit(DESTROYED); | ||
|  |         return this; | ||
|  |     } | ||
|  |     /** | ||
|  |      * Alias for {@link isStream} | ||
|  |      * | ||
|  |      * Former export location, maintained for backwards compatibility. | ||
|  |      * | ||
|  |      * @deprecated | ||
|  |      */ | ||
|  |     static get isStream() { | ||
|  |         return exports.isStream; | ||
|  |     } | ||
|  | } | ||
|  | exports.Minipass = Minipass; | ||
|  | //# sourceMappingURL=index.js.map
 |