| // Copyright Joyent, Inc. and other Node contributors. | 
| // | 
| // Permission is hereby granted, free of charge, to any person obtaining a | 
| // copy of this software and associated documentation files (the | 
| // "Software"), to deal in the Software without restriction, including | 
| // without limitation the rights to use, copy, modify, merge, publish, | 
| // distribute, sublicense, and/or sell copies of the Software, and to permit | 
| // persons to whom the Software is furnished to do so, subject to the | 
| // following conditions: | 
| // | 
| // The above copyright notice and this permission notice shall be included | 
| // in all copies or substantial portions of the Software. | 
| // | 
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | 
| // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | 
| // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN | 
| // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, | 
| // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR | 
| // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE | 
| // USE OR OTHER DEALINGS IN THE SOFTWARE. | 
|   | 
| module.exports = Stream; | 
|   | 
| var EE = require('events').EventEmitter; | 
| var inherits = require('inherits'); | 
|   | 
| inherits(Stream, EE); | 
| Stream.Readable = require('readable-stream/readable.js'); | 
| Stream.Writable = require('readable-stream/writable.js'); | 
| Stream.Duplex = require('readable-stream/duplex.js'); | 
| Stream.Transform = require('readable-stream/transform.js'); | 
| Stream.PassThrough = require('readable-stream/passthrough.js'); | 
|   | 
| // Backwards-compat with node 0.4.x | 
| Stream.Stream = Stream; | 
|   | 
|   | 
|   | 
| // old-style streams.  Note that the pipe method (the only relevant | 
| // part of this class) is overridden in the Readable class. | 
|   | 
| function Stream() { | 
|   EE.call(this); | 
| } | 
|   | 
| Stream.prototype.pipe = function(dest, options) { | 
|   var source = this; | 
|   | 
|   function ondata(chunk) { | 
|     if (dest.writable) { | 
|       if (false === dest.write(chunk) && source.pause) { | 
|         source.pause(); | 
|       } | 
|     } | 
|   } | 
|   | 
|   source.on('data', ondata); | 
|   | 
|   function ondrain() { | 
|     if (source.readable && source.resume) { | 
|       source.resume(); | 
|     } | 
|   } | 
|   | 
|   dest.on('drain', ondrain); | 
|   | 
|   // If the 'end' option is not supplied, dest.end() will be called when | 
|   // source gets the 'end' or 'close' events.  Only dest.end() once. | 
|   if (!dest._isStdio && (!options || options.end !== false)) { | 
|     source.on('end', onend); | 
|     source.on('close', onclose); | 
|   } | 
|   | 
|   var didOnEnd = false; | 
|   function onend() { | 
|     if (didOnEnd) return; | 
|     didOnEnd = true; | 
|   | 
|     dest.end(); | 
|   } | 
|   | 
|   | 
|   function onclose() { | 
|     if (didOnEnd) return; | 
|     didOnEnd = true; | 
|   | 
|     if (typeof dest.destroy === 'function') dest.destroy(); | 
|   } | 
|   | 
|   // don't leave dangling pipes when there are errors. | 
|   function onerror(er) { | 
|     cleanup(); | 
|     if (EE.listenerCount(this, 'error') === 0) { | 
|       throw er; // Unhandled stream error in pipe. | 
|     } | 
|   } | 
|   | 
|   source.on('error', onerror); | 
|   dest.on('error', onerror); | 
|   | 
|   // remove all the event listeners that were added. | 
|   function cleanup() { | 
|     source.removeListener('data', ondata); | 
|     dest.removeListener('drain', ondrain); | 
|   | 
|     source.removeListener('end', onend); | 
|     source.removeListener('close', onclose); | 
|   | 
|     source.removeListener('error', onerror); | 
|     dest.removeListener('error', onerror); | 
|   | 
|     source.removeListener('end', cleanup); | 
|     source.removeListener('close', cleanup); | 
|   | 
|     dest.removeListener('close', cleanup); | 
|   } | 
|   | 
|   source.on('end', cleanup); | 
|   source.on('close', cleanup); | 
|   | 
|   dest.on('close', cleanup); | 
|   | 
|   dest.emit('pipe', source); | 
|   | 
|   // Allow for unix-like usage: A.pipe(B).pipe(C) | 
|   return dest; | 
| }; |