[6a3a178] | 1 | var once = require('once')
|
---|
| 2 | var eos = require('end-of-stream')
|
---|
| 3 | var fs = require('fs') // we only need fs to get the ReadStream and WriteStream prototypes
|
---|
| 4 |
|
---|
| 5 | var noop = function () {}
|
---|
| 6 | var ancient = /^v?\.0/.test(process.version)
|
---|
| 7 |
|
---|
| 8 | var isFn = function (fn) {
|
---|
| 9 | return typeof fn === 'function'
|
---|
| 10 | }
|
---|
| 11 |
|
---|
| 12 | var isFS = function (stream) {
|
---|
| 13 | if (!ancient) return false // newer node version do not need to care about fs is a special way
|
---|
| 14 | if (!fs) return false // browser
|
---|
| 15 | return (stream instanceof (fs.ReadStream || noop) || stream instanceof (fs.WriteStream || noop)) && isFn(stream.close)
|
---|
| 16 | }
|
---|
| 17 |
|
---|
| 18 | var isRequest = function (stream) {
|
---|
| 19 | return stream.setHeader && isFn(stream.abort)
|
---|
| 20 | }
|
---|
| 21 |
|
---|
| 22 | var destroyer = function (stream, reading, writing, callback) {
|
---|
| 23 | callback = once(callback)
|
---|
| 24 |
|
---|
| 25 | var closed = false
|
---|
| 26 | stream.on('close', function () {
|
---|
| 27 | closed = true
|
---|
| 28 | })
|
---|
| 29 |
|
---|
| 30 | eos(stream, {readable: reading, writable: writing}, function (err) {
|
---|
| 31 | if (err) return callback(err)
|
---|
| 32 | closed = true
|
---|
| 33 | callback()
|
---|
| 34 | })
|
---|
| 35 |
|
---|
| 36 | var destroyed = false
|
---|
| 37 | return function (err) {
|
---|
| 38 | if (closed) return
|
---|
| 39 | if (destroyed) return
|
---|
| 40 | destroyed = true
|
---|
| 41 |
|
---|
| 42 | if (isFS(stream)) return stream.close(noop) // use close for fs streams to avoid fd leaks
|
---|
| 43 | if (isRequest(stream)) return stream.abort() // request.destroy just do .end - .abort is what we want
|
---|
| 44 |
|
---|
| 45 | if (isFn(stream.destroy)) return stream.destroy()
|
---|
| 46 |
|
---|
| 47 | callback(err || new Error('stream was destroyed'))
|
---|
| 48 | }
|
---|
| 49 | }
|
---|
| 50 |
|
---|
| 51 | var call = function (fn) {
|
---|
| 52 | fn()
|
---|
| 53 | }
|
---|
| 54 |
|
---|
| 55 | var pipe = function (from, to) {
|
---|
| 56 | return from.pipe(to)
|
---|
| 57 | }
|
---|
| 58 |
|
---|
| 59 | var pump = function () {
|
---|
| 60 | var streams = Array.prototype.slice.call(arguments)
|
---|
| 61 | var callback = isFn(streams[streams.length - 1] || noop) && streams.pop() || noop
|
---|
| 62 |
|
---|
| 63 | if (Array.isArray(streams[0])) streams = streams[0]
|
---|
| 64 | if (streams.length < 2) throw new Error('pump requires two streams per minimum')
|
---|
| 65 |
|
---|
| 66 | var error
|
---|
| 67 | var destroys = streams.map(function (stream, i) {
|
---|
| 68 | var reading = i < streams.length - 1
|
---|
| 69 | var writing = i > 0
|
---|
| 70 | return destroyer(stream, reading, writing, function (err) {
|
---|
| 71 | if (!error) error = err
|
---|
| 72 | if (err) destroys.forEach(call)
|
---|
| 73 | if (reading) return
|
---|
| 74 | destroys.forEach(call)
|
---|
| 75 | callback(error)
|
---|
| 76 | })
|
---|
| 77 | })
|
---|
| 78 |
|
---|
| 79 | return streams.reduce(pipe)
|
---|
| 80 | }
|
---|
| 81 |
|
---|
| 82 | module.exports = pump
|
---|