1 | 'use strict'
|
---|
2 | const proc = typeof process === 'object' && process ? process : {
|
---|
3 | stdout: null,
|
---|
4 | stderr: null,
|
---|
5 | }
|
---|
6 | const EE = require('events')
|
---|
7 | const Stream = require('stream')
|
---|
8 | const Yallist = require('yallist')
|
---|
9 | const SD = require('string_decoder').StringDecoder
|
---|
10 |
|
---|
11 | const EOF = Symbol('EOF')
|
---|
12 | const MAYBE_EMIT_END = Symbol('maybeEmitEnd')
|
---|
13 | const EMITTED_END = Symbol('emittedEnd')
|
---|
14 | const EMITTING_END = Symbol('emittingEnd')
|
---|
15 | const EMITTED_ERROR = Symbol('emittedError')
|
---|
16 | const CLOSED = Symbol('closed')
|
---|
17 | const READ = Symbol('read')
|
---|
18 | const FLUSH = Symbol('flush')
|
---|
19 | const FLUSHCHUNK = Symbol('flushChunk')
|
---|
20 | const ENCODING = Symbol('encoding')
|
---|
21 | const DECODER = Symbol('decoder')
|
---|
22 | const FLOWING = Symbol('flowing')
|
---|
23 | const PAUSED = Symbol('paused')
|
---|
24 | const RESUME = Symbol('resume')
|
---|
25 | const BUFFERLENGTH = Symbol('bufferLength')
|
---|
26 | const BUFFERPUSH = Symbol('bufferPush')
|
---|
27 | const BUFFERSHIFT = Symbol('bufferShift')
|
---|
28 | const OBJECTMODE = Symbol('objectMode')
|
---|
29 | const DESTROYED = Symbol('destroyed')
|
---|
30 |
|
---|
31 | // TODO remove when Node v8 support drops
|
---|
32 | const doIter = global._MP_NO_ITERATOR_SYMBOLS_ !== '1'
|
---|
33 | const ASYNCITERATOR = doIter && Symbol.asyncIterator
|
---|
34 | || Symbol('asyncIterator not implemented')
|
---|
35 | const ITERATOR = doIter && Symbol.iterator
|
---|
36 | || Symbol('iterator not implemented')
|
---|
37 |
|
---|
38 | // events that mean 'the stream is over'
|
---|
39 | // these are treated specially, and re-emitted
|
---|
40 | // if they are listened for after emitting.
|
---|
41 | const isEndish = ev =>
|
---|
42 | ev === 'end' ||
|
---|
43 | ev === 'finish' ||
|
---|
44 | ev === 'prefinish'
|
---|
45 |
|
---|
46 | const isArrayBuffer = b => b instanceof ArrayBuffer ||
|
---|
47 | typeof b === 'object' &&
|
---|
48 | b.constructor &&
|
---|
49 | b.constructor.name === 'ArrayBuffer' &&
|
---|
50 | b.byteLength >= 0
|
---|
51 |
|
---|
52 | const isArrayBufferView = b => !Buffer.isBuffer(b) && ArrayBuffer.isView(b)
|
---|
53 |
|
---|
54 | module.exports = class Minipass extends Stream {
|
---|
55 | constructor (options) {
|
---|
56 | super()
|
---|
57 | this[FLOWING] = false
|
---|
58 | // whether we're explicitly paused
|
---|
59 | this[PAUSED] = false
|
---|
60 | this.pipes = new Yallist()
|
---|
61 | this.buffer = new Yallist()
|
---|
62 | this[OBJECTMODE] = options && options.objectMode || false
|
---|
63 | if (this[OBJECTMODE])
|
---|
64 | this[ENCODING] = null
|
---|
65 | else
|
---|
66 | this[ENCODING] = options && options.encoding || null
|
---|
67 | if (this[ENCODING] === 'buffer')
|
---|
68 | this[ENCODING] = null
|
---|
69 | this[DECODER] = this[ENCODING] ? new SD(this[ENCODING]) : null
|
---|
70 | this[EOF] = false
|
---|
71 | this[EMITTED_END] = false
|
---|
72 | this[EMITTING_END] = false
|
---|
73 | this[CLOSED] = false
|
---|
74 | this[EMITTED_ERROR] = null
|
---|
75 | this.writable = true
|
---|
76 | this.readable = true
|
---|
77 | this[BUFFERLENGTH] = 0
|
---|
78 | this[DESTROYED] = false
|
---|
79 | }
|
---|
80 |
|
---|
81 | get bufferLength () { return this[BUFFERLENGTH] }
|
---|
82 |
|
---|
83 | get encoding () { return this[ENCODING] }
|
---|
84 | set encoding (enc) {
|
---|
85 | if (this[OBJECTMODE])
|
---|
86 | throw new Error('cannot set encoding in objectMode')
|
---|
87 |
|
---|
88 | if (this[ENCODING] && enc !== this[ENCODING] &&
|
---|
89 | (this[DECODER] && this[DECODER].lastNeed || this[BUFFERLENGTH]))
|
---|
90 | throw new Error('cannot change encoding')
|
---|
91 |
|
---|
92 | if (this[ENCODING] !== enc) {
|
---|
93 | this[DECODER] = enc ? new SD(enc) : null
|
---|
94 | if (this.buffer.length)
|
---|
95 | this.buffer = this.buffer.map(chunk => this[DECODER].write(chunk))
|
---|
96 | }
|
---|
97 |
|
---|
98 | this[ENCODING] = enc
|
---|
99 | }
|
---|
100 |
|
---|
101 | setEncoding (enc) {
|
---|
102 | this.encoding = enc
|
---|
103 | }
|
---|
104 |
|
---|
105 | get objectMode () { return this[OBJECTMODE] }
|
---|
106 | set objectMode (om) { this[OBJECTMODE] = this[OBJECTMODE] || !!om }
|
---|
107 |
|
---|
108 | write (chunk, encoding, cb) {
|
---|
109 | if (this[EOF])
|
---|
110 | throw new Error('write after end')
|
---|
111 |
|
---|
112 | if (this[DESTROYED]) {
|
---|
113 | this.emit('error', Object.assign(
|
---|
114 | new Error('Cannot call write after a stream was destroyed'),
|
---|
115 | { code: 'ERR_STREAM_DESTROYED' }
|
---|
116 | ))
|
---|
117 | return true
|
---|
118 | }
|
---|
119 |
|
---|
120 | if (typeof encoding === 'function')
|
---|
121 | cb = encoding, encoding = 'utf8'
|
---|
122 |
|
---|
123 | if (!encoding)
|
---|
124 | encoding = 'utf8'
|
---|
125 |
|
---|
126 | // convert array buffers and typed array views into buffers
|
---|
127 | // at some point in the future, we may want to do the opposite!
|
---|
128 | // leave strings and buffers as-is
|
---|
129 | // anything else switches us into object mode
|
---|
130 | if (!this[OBJECTMODE] && !Buffer.isBuffer(chunk)) {
|
---|
131 | if (isArrayBufferView(chunk))
|
---|
132 | chunk = Buffer.from(chunk.buffer, chunk.byteOffset, chunk.byteLength)
|
---|
133 | else if (isArrayBuffer(chunk))
|
---|
134 | chunk = Buffer.from(chunk)
|
---|
135 | else if (typeof chunk !== 'string')
|
---|
136 | // use the setter so we throw if we have encoding set
|
---|
137 | this.objectMode = true
|
---|
138 | }
|
---|
139 |
|
---|
140 | // this ensures at this point that the chunk is a buffer or string
|
---|
141 | // don't buffer it up or send it to the decoder
|
---|
142 | if (!this.objectMode && !chunk.length) {
|
---|
143 | if (this[BUFFERLENGTH] !== 0)
|
---|
144 | this.emit('readable')
|
---|
145 | if (cb)
|
---|
146 | cb()
|
---|
147 | return this.flowing
|
---|
148 | }
|
---|
149 |
|
---|
150 | // fast-path writing strings of same encoding to a stream with
|
---|
151 | // an empty buffer, skipping the buffer/decoder dance
|
---|
152 | if (typeof chunk === 'string' && !this[OBJECTMODE] &&
|
---|
153 | // unless it is a string already ready for us to use
|
---|
154 | !(encoding === this[ENCODING] && !this[DECODER].lastNeed)) {
|
---|
155 | chunk = Buffer.from(chunk, encoding)
|
---|
156 | }
|
---|
157 |
|
---|
158 | if (Buffer.isBuffer(chunk) && this[ENCODING])
|
---|
159 | chunk = this[DECODER].write(chunk)
|
---|
160 |
|
---|
161 | if (this.flowing) {
|
---|
162 | // if we somehow have something in the buffer, but we think we're
|
---|
163 | // flowing, then we need to flush all that out first, or we get
|
---|
164 | // chunks coming in out of order. Can't emit 'drain' here though,
|
---|
165 | // because we're mid-write, so that'd be bad.
|
---|
166 | if (this[BUFFERLENGTH] !== 0)
|
---|
167 | this[FLUSH](true)
|
---|
168 | this.emit('data', chunk)
|
---|
169 | } else
|
---|
170 | this[BUFFERPUSH](chunk)
|
---|
171 |
|
---|
172 | if (this[BUFFERLENGTH] !== 0)
|
---|
173 | this.emit('readable')
|
---|
174 |
|
---|
175 | if (cb)
|
---|
176 | cb()
|
---|
177 |
|
---|
178 | return this.flowing
|
---|
179 | }
|
---|
180 |
|
---|
181 | read (n) {
|
---|
182 | if (this[DESTROYED])
|
---|
183 | return null
|
---|
184 |
|
---|
185 | try {
|
---|
186 | if (this[BUFFERLENGTH] === 0 || n === 0 || n > this[BUFFERLENGTH])
|
---|
187 | return null
|
---|
188 |
|
---|
189 | if (this[OBJECTMODE])
|
---|
190 | n = null
|
---|
191 |
|
---|
192 | if (this.buffer.length > 1 && !this[OBJECTMODE]) {
|
---|
193 | if (this.encoding)
|
---|
194 | this.buffer = new Yallist([
|
---|
195 | Array.from(this.buffer).join('')
|
---|
196 | ])
|
---|
197 | else
|
---|
198 | this.buffer = new Yallist([
|
---|
199 | Buffer.concat(Array.from(this.buffer), this[BUFFERLENGTH])
|
---|
200 | ])
|
---|
201 | }
|
---|
202 |
|
---|
203 | return this[READ](n || null, this.buffer.head.value)
|
---|
204 | } finally {
|
---|
205 | this[MAYBE_EMIT_END]()
|
---|
206 | }
|
---|
207 | }
|
---|
208 |
|
---|
209 | [READ] (n, chunk) {
|
---|
210 | if (n === chunk.length || n === null)
|
---|
211 | this[BUFFERSHIFT]()
|
---|
212 | else {
|
---|
213 | this.buffer.head.value = chunk.slice(n)
|
---|
214 | chunk = chunk.slice(0, n)
|
---|
215 | this[BUFFERLENGTH] -= n
|
---|
216 | }
|
---|
217 |
|
---|
218 | this.emit('data', chunk)
|
---|
219 |
|
---|
220 | if (!this.buffer.length && !this[EOF])
|
---|
221 | this.emit('drain')
|
---|
222 |
|
---|
223 | return chunk
|
---|
224 | }
|
---|
225 |
|
---|
226 | end (chunk, encoding, cb) {
|
---|
227 | if (typeof chunk === 'function')
|
---|
228 | cb = chunk, chunk = null
|
---|
229 | if (typeof encoding === 'function')
|
---|
230 | cb = encoding, encoding = 'utf8'
|
---|
231 | if (chunk)
|
---|
232 | this.write(chunk, encoding)
|
---|
233 | if (cb)
|
---|
234 | this.once('end', cb)
|
---|
235 | this[EOF] = true
|
---|
236 | this.writable = false
|
---|
237 |
|
---|
238 | // if we haven't written anything, then go ahead and emit,
|
---|
239 | // even if we're not reading.
|
---|
240 | // we'll re-emit if a new 'end' listener is added anyway.
|
---|
241 | // This makes MP more suitable to write-only use cases.
|
---|
242 | if (this.flowing || !this[PAUSED])
|
---|
243 | this[MAYBE_EMIT_END]()
|
---|
244 | return this
|
---|
245 | }
|
---|
246 |
|
---|
247 | // don't let the internal resume be overwritten
|
---|
248 | [RESUME] () {
|
---|
249 | if (this[DESTROYED])
|
---|
250 | return
|
---|
251 |
|
---|
252 | this[PAUSED] = false
|
---|
253 | this[FLOWING] = true
|
---|
254 | this.emit('resume')
|
---|
255 | if (this.buffer.length)
|
---|
256 | this[FLUSH]()
|
---|
257 | else if (this[EOF])
|
---|
258 | this[MAYBE_EMIT_END]()
|
---|
259 | else
|
---|
260 | this.emit('drain')
|
---|
261 | }
|
---|
262 |
|
---|
263 | resume () {
|
---|
264 | return this[RESUME]()
|
---|
265 | }
|
---|
266 |
|
---|
267 | pause () {
|
---|
268 | this[FLOWING] = false
|
---|
269 | this[PAUSED] = true
|
---|
270 | }
|
---|
271 |
|
---|
272 | get destroyed () {
|
---|
273 | return this[DESTROYED]
|
---|
274 | }
|
---|
275 |
|
---|
276 | get flowing () {
|
---|
277 | return this[FLOWING]
|
---|
278 | }
|
---|
279 |
|
---|
280 | get paused () {
|
---|
281 | return this[PAUSED]
|
---|
282 | }
|
---|
283 |
|
---|
284 | [BUFFERPUSH] (chunk) {
|
---|
285 | if (this[OBJECTMODE])
|
---|
286 | this[BUFFERLENGTH] += 1
|
---|
287 | else
|
---|
288 | this[BUFFERLENGTH] += chunk.length
|
---|
289 | return this.buffer.push(chunk)
|
---|
290 | }
|
---|
291 |
|
---|
292 | [BUFFERSHIFT] () {
|
---|
293 | if (this.buffer.length) {
|
---|
294 | if (this[OBJECTMODE])
|
---|
295 | this[BUFFERLENGTH] -= 1
|
---|
296 | else
|
---|
297 | this[BUFFERLENGTH] -= this.buffer.head.value.length
|
---|
298 | }
|
---|
299 | return this.buffer.shift()
|
---|
300 | }
|
---|
301 |
|
---|
302 | [FLUSH] (noDrain) {
|
---|
303 | do {} while (this[FLUSHCHUNK](this[BUFFERSHIFT]()))
|
---|
304 |
|
---|
305 | if (!noDrain && !this.buffer.length && !this[EOF])
|
---|
306 | this.emit('drain')
|
---|
307 | }
|
---|
308 |
|
---|
309 | [FLUSHCHUNK] (chunk) {
|
---|
310 | return chunk ? (this.emit('data', chunk), this.flowing) : false
|
---|
311 | }
|
---|
312 |
|
---|
313 | pipe (dest, opts) {
|
---|
314 | if (this[DESTROYED])
|
---|
315 | return
|
---|
316 |
|
---|
317 | const ended = this[EMITTED_END]
|
---|
318 | opts = opts || {}
|
---|
319 | if (dest === proc.stdout || dest === proc.stderr)
|
---|
320 | opts.end = false
|
---|
321 | else
|
---|
322 | opts.end = opts.end !== false
|
---|
323 |
|
---|
324 | const p = { dest: dest, opts: opts, ondrain: _ => this[RESUME]() }
|
---|
325 | this.pipes.push(p)
|
---|
326 |
|
---|
327 | dest.on('drain', p.ondrain)
|
---|
328 | this[RESUME]()
|
---|
329 | // piping an ended stream ends immediately
|
---|
330 | if (ended && p.opts.end)
|
---|
331 | p.dest.end()
|
---|
332 | return dest
|
---|
333 | }
|
---|
334 |
|
---|
335 | addListener (ev, fn) {
|
---|
336 | return this.on(ev, fn)
|
---|
337 | }
|
---|
338 |
|
---|
339 | on (ev, fn) {
|
---|
340 | try {
|
---|
341 | return super.on(ev, fn)
|
---|
342 | } finally {
|
---|
343 | if (ev === 'data' && !this.pipes.length && !this.flowing)
|
---|
344 | this[RESUME]()
|
---|
345 | else if (isEndish(ev) && this[EMITTED_END]) {
|
---|
346 | super.emit(ev)
|
---|
347 | this.removeAllListeners(ev)
|
---|
348 | } else if (ev === 'error' && this[EMITTED_ERROR]) {
|
---|
349 | fn.call(this, this[EMITTED_ERROR])
|
---|
350 | }
|
---|
351 | }
|
---|
352 | }
|
---|
353 |
|
---|
354 | get emittedEnd () {
|
---|
355 | return this[EMITTED_END]
|
---|
356 | }
|
---|
357 |
|
---|
358 | [MAYBE_EMIT_END] () {
|
---|
359 | if (!this[EMITTING_END] &&
|
---|
360 | !this[EMITTED_END] &&
|
---|
361 | !this[DESTROYED] &&
|
---|
362 | this.buffer.length === 0 &&
|
---|
363 | this[EOF]) {
|
---|
364 | this[EMITTING_END] = true
|
---|
365 | this.emit('end')
|
---|
366 | this.emit('prefinish')
|
---|
367 | this.emit('finish')
|
---|
368 | if (this[CLOSED])
|
---|
369 | this.emit('close')
|
---|
370 | this[EMITTING_END] = false
|
---|
371 | }
|
---|
372 | }
|
---|
373 |
|
---|
374 | emit (ev, data) {
|
---|
375 | // error and close are only events allowed after calling destroy()
|
---|
376 | if (ev !== 'error' && ev !== 'close' && ev !== DESTROYED && this[DESTROYED])
|
---|
377 | return
|
---|
378 | else if (ev === 'data') {
|
---|
379 | if (!data)
|
---|
380 | return
|
---|
381 |
|
---|
382 | if (this.pipes.length)
|
---|
383 | this.pipes.forEach(p =>
|
---|
384 | p.dest.write(data) === false && this.pause())
|
---|
385 | } else if (ev === 'end') {
|
---|
386 | // only actual end gets this treatment
|
---|
387 | if (this[EMITTED_END] === true)
|
---|
388 | return
|
---|
389 |
|
---|
390 | this[EMITTED_END] = true
|
---|
391 | this.readable = false
|
---|
392 |
|
---|
393 | if (this[DECODER]) {
|
---|
394 | data = this[DECODER].end()
|
---|
395 | if (data) {
|
---|
396 | this.pipes.forEach(p => p.dest.write(data))
|
---|
397 | super.emit('data', data)
|
---|
398 | }
|
---|
399 | }
|
---|
400 |
|
---|
401 | this.pipes.forEach(p => {
|
---|
402 | p.dest.removeListener('drain', p.ondrain)
|
---|
403 | if (p.opts.end)
|
---|
404 | p.dest.end()
|
---|
405 | })
|
---|
406 | } else if (ev === 'close') {
|
---|
407 | this[CLOSED] = true
|
---|
408 | // don't emit close before 'end' and 'finish'
|
---|
409 | if (!this[EMITTED_END] && !this[DESTROYED])
|
---|
410 | return
|
---|
411 | } else if (ev === 'error') {
|
---|
412 | this[EMITTED_ERROR] = data
|
---|
413 | }
|
---|
414 |
|
---|
415 | // TODO: replace with a spread operator when Node v4 support drops
|
---|
416 | const args = new Array(arguments.length)
|
---|
417 | args[0] = ev
|
---|
418 | args[1] = data
|
---|
419 | if (arguments.length > 2) {
|
---|
420 | for (let i = 2; i < arguments.length; i++) {
|
---|
421 | args[i] = arguments[i]
|
---|
422 | }
|
---|
423 | }
|
---|
424 |
|
---|
425 | try {
|
---|
426 | return super.emit.apply(this, args)
|
---|
427 | } finally {
|
---|
428 | if (!isEndish(ev))
|
---|
429 | this[MAYBE_EMIT_END]()
|
---|
430 | else
|
---|
431 | this.removeAllListeners(ev)
|
---|
432 | }
|
---|
433 | }
|
---|
434 |
|
---|
435 | // const all = await stream.collect()
|
---|
436 | collect () {
|
---|
437 | const buf = []
|
---|
438 | if (!this[OBJECTMODE])
|
---|
439 | buf.dataLength = 0
|
---|
440 | // set the promise first, in case an error is raised
|
---|
441 | // by triggering the flow here.
|
---|
442 | const p = this.promise()
|
---|
443 | this.on('data', c => {
|
---|
444 | buf.push(c)
|
---|
445 | if (!this[OBJECTMODE])
|
---|
446 | buf.dataLength += c.length
|
---|
447 | })
|
---|
448 | return p.then(() => buf)
|
---|
449 | }
|
---|
450 |
|
---|
451 | // const data = await stream.concat()
|
---|
452 | concat () {
|
---|
453 | return this[OBJECTMODE]
|
---|
454 | ? Promise.reject(new Error('cannot concat in objectMode'))
|
---|
455 | : this.collect().then(buf =>
|
---|
456 | this[OBJECTMODE]
|
---|
457 | ? Promise.reject(new Error('cannot concat in objectMode'))
|
---|
458 | : this[ENCODING] ? buf.join('') : Buffer.concat(buf, buf.dataLength))
|
---|
459 | }
|
---|
460 |
|
---|
461 | // stream.promise().then(() => done, er => emitted error)
|
---|
462 | promise () {
|
---|
463 | return new Promise((resolve, reject) => {
|
---|
464 | this.on(DESTROYED, () => reject(new Error('stream destroyed')))
|
---|
465 | this.on('error', er => reject(er))
|
---|
466 | this.on('end', () => resolve())
|
---|
467 | })
|
---|
468 | }
|
---|
469 |
|
---|
470 | // for await (let chunk of stream)
|
---|
471 | [ASYNCITERATOR] () {
|
---|
472 | const next = () => {
|
---|
473 | const res = this.read()
|
---|
474 | if (res !== null)
|
---|
475 | return Promise.resolve({ done: false, value: res })
|
---|
476 |
|
---|
477 | if (this[EOF])
|
---|
478 | return Promise.resolve({ done: true })
|
---|
479 |
|
---|
480 | let resolve = null
|
---|
481 | let reject = null
|
---|
482 | const onerr = er => {
|
---|
483 | this.removeListener('data', ondata)
|
---|
484 | this.removeListener('end', onend)
|
---|
485 | reject(er)
|
---|
486 | }
|
---|
487 | const ondata = value => {
|
---|
488 | this.removeListener('error', onerr)
|
---|
489 | this.removeListener('end', onend)
|
---|
490 | this.pause()
|
---|
491 | resolve({ value: value, done: !!this[EOF] })
|
---|
492 | }
|
---|
493 | const onend = () => {
|
---|
494 | this.removeListener('error', onerr)
|
---|
495 | this.removeListener('data', ondata)
|
---|
496 | resolve({ done: true })
|
---|
497 | }
|
---|
498 | const ondestroy = () => onerr(new Error('stream destroyed'))
|
---|
499 | return new Promise((res, rej) => {
|
---|
500 | reject = rej
|
---|
501 | resolve = res
|
---|
502 | this.once(DESTROYED, ondestroy)
|
---|
503 | this.once('error', onerr)
|
---|
504 | this.once('end', onend)
|
---|
505 | this.once('data', ondata)
|
---|
506 | })
|
---|
507 | }
|
---|
508 |
|
---|
509 | return { next }
|
---|
510 | }
|
---|
511 |
|
---|
512 | // for (let chunk of stream)
|
---|
513 | [ITERATOR] () {
|
---|
514 | const next = () => {
|
---|
515 | const value = this.read()
|
---|
516 | const done = value === null
|
---|
517 | return { value, done }
|
---|
518 | }
|
---|
519 | return { next }
|
---|
520 | }
|
---|
521 |
|
---|
522 | destroy (er) {
|
---|
523 | if (this[DESTROYED]) {
|
---|
524 | if (er)
|
---|
525 | this.emit('error', er)
|
---|
526 | else
|
---|
527 | this.emit(DESTROYED)
|
---|
528 | return this
|
---|
529 | }
|
---|
530 |
|
---|
531 | this[DESTROYED] = true
|
---|
532 |
|
---|
533 | // throw away all buffered data, it's never coming out
|
---|
534 | this.buffer = new Yallist()
|
---|
535 | this[BUFFERLENGTH] = 0
|
---|
536 |
|
---|
537 | if (typeof this.close === 'function' && !this[CLOSED])
|
---|
538 | this.close()
|
---|
539 |
|
---|
540 | if (er)
|
---|
541 | this.emit('error', er)
|
---|
542 | else // if no error to emit, still reject pending promises
|
---|
543 | this.emit(DESTROYED)
|
---|
544 |
|
---|
545 | return this
|
---|
546 | }
|
---|
547 |
|
---|
548 | static isStream (s) {
|
---|
549 | return !!s && (s instanceof Minipass || s instanceof Stream ||
|
---|
550 | s instanceof EE && (
|
---|
551 | typeof s.pipe === 'function' || // readable
|
---|
552 | (typeof s.write === 'function' && typeof s.end === 'function') // writable
|
---|
553 | ))
|
---|
554 | }
|
---|
555 | }
|
---|