Search code examples
node.jsspawncapture-output

Node.js spawn child process and get terminal output live


I have a script that outputs 'hi', sleeps for a second, outputs 'hi', sleeps for 1 second, and so on and so forth. Now I thought I would be able to tackle this problem with this model.

var spawn = require('child_process').spawn,
temp    = spawn('PATH TO SCRIPT WITH THE ABOVE BEHAVIOUR');

temp.stdout.pipe(process.stdout);

Now the problem is that the task needs to be finished in order for the output to be displayed. As I am understanding it, this is due to the fact that the newly spawned process takes execution control. Obviously node.js does not support threads so any solutions? My idea was to possibly run two instances, first one for the specific purpose of creating the task and have it pipe the output to process of the second instance, considering this can be achieved.


Solution

  • I'm still getting my feet wet with Node.js, but I have a few ideas. first, I believe you need to use execFile instead of spawn; execFile is for when you have the path to a script, whereas spawn is for executing a well-known command that Node.js can resolve against your system path.

    1. Provide a callback to process the buffered output:

    var child = require('child_process').execFile('path/to/script', [ 
        'arg1', 'arg2', 'arg3', 
    ], function(err, stdout, stderr) { 
        // Node.js will invoke this callback when process terminates.
        console.log(stdout); 
    });  
    

    2. Add a listener to the child process' stdout stream (9thport.net)

    var child = require('child_process').execFile('path/to/script', [ 
        'arg1', 'arg2', 'arg3' ]); 
    // use event hooks to provide a callback to execute when data are available: 
    child.stdout.on('data', function(data) {
        console.log(data.toString()); 
    });
    

    Further, there appear to be options whereby you can detach the spawned process from Node's controlling terminal, which would allow it to run asynchronously. I haven't tested this yet, but there are examples in the API docs that go something like this:

    child = require('child_process').execFile('path/to/script', [ 
        'arg1', 'arg2', 'arg3', 
    ], { 
        // detachment and ignored stdin are the key here: 
        detached: true, 
        stdio: [ 'ignore', 1, 2 ]
    }); 
    // and unref() somehow disentangles the child's event loop from the parent's: 
    child.unref(); 
    child.stdout.on('data', function(data) {
        console.log(data.toString()); 
    });