Search code examples
javascriptnode.jsstreampipenode.js-stream

Node.js Piping the same readable stream into multiple (writable) targets


I need to run two commands in series that need to read data from the same stream. After piping a stream into another the buffer is emptied so i can't read data from that stream again so this doesn't work:

var spawn = require('child_process').spawn;
var fs = require('fs');
var request = require('request');

var inputStream = request('http://placehold.it/640x360');
var identify = spawn('identify',['-']);

inputStream.pipe(identify.stdin);

var chunks = [];
identify.stdout.on('data',function(chunk) {
  chunks.push(chunk);
});

identify.stdout.on('end',function() {
  var size = getSize(Buffer.concat(chunks)); //width
  var convert = spawn('convert',['-','-scale',size * 0.5,'png:-']);
  inputStream.pipe(convert.stdin);
  convert.stdout.pipe(fs.createWriteStream('half.png'));
});

function getSize(buffer){
  return parseInt(buffer.toString().split(' ')[2].split('x')[0]);
}

Request complains about this

Error: You cannot pipe after data has been emitted from the response.

and changing the inputStream to fs.createWriteStream yields the same issue of course. I don't want to write into a file but reuse in some way the stream that request produces (or any other for that matter).

Is there a way to reuse a readable stream once it finishes piping? What would be the best way to accomplish something like the above example?


Solution

  • You have to create duplicate of the stream by piping it to two streams. You can create a simple stream with a PassThrough stream, it simply passes the input to the output.

    const spawn = require('child_process').spawn;
    const PassThrough = require('stream').PassThrough;
    
    const a = spawn('echo', ['hi user']);
    const b = new PassThrough();
    const c = new PassThrough();
    
    a.stdout.pipe(b);
    a.stdout.pipe(c);
    
    let count = 0;
    b.on('data', function (chunk) {
      count += chunk.length;
    });
    b.on('end', function () {
      console.log(count);
      c.pipe(process.stdout);
    });
    

    Output:

    8
    hi user