Execute a command line binary with Node.js

后端 未结 12 2235
星月不相逢
星月不相逢 2020-11-22 01:39

I am in the process of porting a CLI library from Ruby over to Node.js. In my code I execute several third party binaries when necessary. I am not sure how best to accomplis

相关标签:
12条回答
  • 2020-11-22 02:09

    For even newer version of Node.js (v8.1.4), the events and calls are similar or identical to older versions, but it's encouraged to use the standard newer language features. Examples:

    For buffered, non-stream formatted output (you get it all at once), use child_process.exec:

    const { exec } = require('child_process');
    exec('cat *.js bad_file | wc -l', (err, stdout, stderr) => {
      if (err) {
        // node couldn't execute the command
        return;
      }
    
      // the *entire* stdout and stderr (buffered)
      console.log(`stdout: ${stdout}`);
      console.log(`stderr: ${stderr}`);
    });
    

    You can also use it with Promises:

    const util = require('util');
    const exec = util.promisify(require('child_process').exec);
    
    async function ls() {
      const { stdout, stderr } = await exec('ls');
      console.log('stdout:', stdout);
      console.log('stderr:', stderr);
    }
    ls();
    

    If you wish to receive the data gradually in chunks (output as a stream), use child_process.spawn:

    const { spawn } = require('child_process');
    const child = spawn('ls', ['-lh', '/usr']);
    
    // use child.stdout.setEncoding('utf8'); if you want text chunks
    child.stdout.on('data', (chunk) => {
      // data from standard output is here as buffers
    });
    
    // since these are streams, you can pipe them elsewhere
    child.stderr.pipe(dest);
    
    child.on('close', (code) => {
      console.log(`child process exited with code ${code}`);
    });
    

    Both of these functions have a synchronous counterpart. An example for child_process.execSync:

    const { execSync } = require('child_process');
    // stderr is sent to stderr of parent process
    // you can set options.stdio if you want it to go elsewhere
    let stdout = execSync('ls');
    

    As well as child_process.spawnSync:

    const { spawnSync} = require('child_process');
    const child = spawnSync('ls', ['-lh', '/usr']);
    
    console.log('error', child.error);
    console.log('stdout ', child.stdout);
    console.log('stderr ', child.stderr);
    

    Note: The following code is still functional, but is primarily targeted at users of ES5 and before.

    The module for spawning child processes with Node.js is well documented in the documentation (v5.0.0). To execute a command and fetch its complete output as a buffer, use child_process.exec:

    var exec = require('child_process').exec;
    var cmd = 'prince -v builds/pdf/book.html -o builds/pdf/book.pdf';
    
    exec(cmd, function(error, stdout, stderr) {
      // command output is in stdout
    });
    

    If you need to use handle process I/O with streams, such as when you are expecting large amounts of output, use child_process.spawn:

    var spawn = require('child_process').spawn;
    var child = spawn('prince', [
      '-v', 'builds/pdf/book.html',
      '-o', 'builds/pdf/book.pdf'
    ]);
    
    child.stdout.on('data', function(chunk) {
      // output will be here in chunks
    });
    
    // or if you want to send output elsewhere
    child.stdout.pipe(dest);
    

    If you are executing a file rather than a command, you might want to use child_process.execFile, which parameters which are almost identical to spawn, but has a fourth callback parameter like exec for retrieving output buffers. That might look a bit like this:

    var execFile = require('child_process').execFile;
    execFile(file, args, options, function(error, stdout, stderr) {
      // command output is in stdout
    });
    

    As of v0.11.12, Node now supports synchronous spawn and exec. All of the methods described above are asynchronous, and have a synchronous counterpart. Documentation for them can be found here. While they are useful for scripting, do note that unlike the methods used to spawn child processes asynchronously, the synchronous methods do not return an instance of ChildProcess.

    0 讨论(0)
  • 2020-11-22 02:09

    Use this lightweight npm package: system-commands

    Look at it here.

    Import it like this:

    const system = require('system-commands')
    

    Run commands like this:

    system('ls').then(output => {
        console.log(output)
    }).catch(error => {
        console.error(error)
    })
    
    0 讨论(0)
  • 2020-11-22 02:10

    @hexacyanide's answer is almost a complete one. On Windows command prince could be prince.exe, prince.cmd, prince.bat or just prince (I'm no aware of how gems are bundled, but npm bins come with a sh script and a batch script - npm and npm.cmd). If you want to write a portable script that would run on Unix and Windows, you have to spawn the right executable.

    Here is a simple yet portable spawn function:

    function spawn(cmd, args, opt) {
        var isWindows = /win/.test(process.platform);
    
        if ( isWindows ) {
            if ( !args ) args = [];
            args.unshift(cmd);
            args.unshift('/c');
            cmd = process.env.comspec;
        }
    
        return child_process.spawn(cmd, args, opt);
    }
    
    var cmd = spawn("prince", ["-v", "builds/pdf/book.html", "-o", "builds/pdf/book.pdf"])
    
    // Use these props to get execution results:
    // cmd.stdin;
    // cmd.stdout;
    // cmd.stderr;
    
    0 讨论(0)
  • 2020-11-22 02:15

    If you don't mind a dependency and want to use promises, child-process-promise works:

    installation

    npm install child-process-promise --save
    

    exec Usage

    var exec = require('child-process-promise').exec;
    
    exec('echo hello')
        .then(function (result) {
            var stdout = result.stdout;
            var stderr = result.stderr;
            console.log('stdout: ', stdout);
            console.log('stderr: ', stderr);
        })
        .catch(function (err) {
            console.error('ERROR: ', err);
        });
    

    spawn usage

    var spawn = require('child-process-promise').spawn;
    
    var promise = spawn('echo', ['hello']);
    
    var childProcess = promise.childProcess;
    
    console.log('[spawn] childProcess.pid: ', childProcess.pid);
    childProcess.stdout.on('data', function (data) {
        console.log('[spawn] stdout: ', data.toString());
    });
    childProcess.stderr.on('data', function (data) {
        console.log('[spawn] stderr: ', data.toString());
    });
    
    promise.then(function () {
            console.log('[spawn] done!');
        })
        .catch(function (err) {
            console.error('[spawn] ERROR: ', err);
        });
    
    0 讨论(0)
  • 2020-11-22 02:16

    You are looking for child_process.exec

    Here is the example:

    const exec = require('child_process').exec;
    const child = exec('cat *.js bad_file | wc -l',
        (error, stdout, stderr) => {
            console.log(`stdout: ${stdout}`);
            console.log(`stderr: ${stderr}`);
            if (error !== null) {
                console.log(`exec error: ${error}`);
            }
    });
    
    0 讨论(0)
  • 2020-11-22 02:19

    If you want something that closely resembles the top answer but is also synchronous then this will work.

    var execSync = require('child_process').execSync;
    var cmd = "echo 'hello world'";
    
    var options = {
      encoding: 'utf8'
    };
    
    console.log(execSync(cmd, options));
    
    0 讨论(0)
提交回复
热议问题