155

在 node.js 中,我想找到一种获取 Unix 终端命令输出的方法。有没有办法做到这一点?

function getCommandOutput(commandString){
    // now how can I implement this function?
    // getCommandOutput("ls") should print the terminal output of the shell command "ls"
}
4

7 回答 7

179

这是我在我目前正在从事的项目中使用的方法。

var exec = require('child_process').exec;
function execute(command, callback){
    exec(command, function(error, stdout, stderr){ callback(stdout); });
};

检索 git 用户的示例:

module.exports.getGitUser = function(callback){
    execute("git config --global user.name", function(name){
        execute("git config --global user.email", function(email){
            callback({ name: name.replace("\n", ""), email: email.replace("\n", "") });
        });
    });
};
于 2012-10-17T18:47:32.707 回答
68

如果您使用的 node 是 7.6 之后的版本并且您不喜欢回调样式,您还可以使用 node-util 的promisify函数 withasync / await来获取清晰读取的 shell 命令。这是使用此技术的已接受答案的示例:

const { promisify } = require('util');
const exec = promisify(require('child_process').exec)

module.exports.getGitUser = async function getGitUser () {
  // Exec output contains both stderr and stdout outputs
  const nameOutput = await exec('git config --global user.name')
  const emailOutput = await exec('git config --global user.email')

  return { 
    name: nameOutput.stdout.trim(), 
    email: emailOutput.stdout.trim()
  }
};

这还有一个额外的好处,即在失败的命令上返回一个被拒绝的承诺,这可以try / catch在异步代码中进行处理。

于 2018-05-14T16:30:59.583 回答
37

您正在寻找child_process

var exec = require('child_process').exec;
var child;

child = exec(command,
   function (error, stdout, stderr) {
      console.log('stdout: ' + stdout);
      console.log('stderr: ' + stderr);
      if (error !== null) {
          console.log('exec error: ' + error);
      }
   });

正如 Renato 所指出的,现在也有一些同步 exec 包,请参阅sync-exec,这可能是您正在寻找的更多内容。但请记住,node.js 被设计为单线程高性能网络服务器,因此如果您希望使用它,请远离 sync-exec 之类的东西,除非您只在启动期间使用它或者其他的东西。

于 2012-10-17T18:44:51.757 回答
22

要求

这需要支持 Promises 和 Async/Await 的 Node.js 7 或更高版本。

解决方案

创建一个包装函数,利用 promise 来控制child_process.exec命令的行为。

解释

使用 Promise 和异步函数,您可以模仿 shell 返回输出的行为,而不会陷入回调地狱,并且 API 非常简洁。使用await关键字,您可以创建一个易于阅读的脚本,同时仍然能够child_process.exec完成工作。

代码示例

const childProcess = require("child_process");

/**
 * @param {string} command A shell command to execute
 * @return {Promise<string>} A promise that resolve to the output of the shell command, or an error
 * @example const output = await execute("ls -alh");
 */
function execute(command) {
  /**
   * @param {Function} resolve A function that resolves the promise
   * @param {Function} reject A function that fails the promise
   * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise
   */
  return new Promise(function(resolve, reject) {
    /**
     * @param {Error} error An error triggered during the execution of the childProcess.exec command
     * @param {string|Buffer} standardOutput The result of the shell command execution
     * @param {string|Buffer} standardError The error resulting of the shell command execution
     * @see https://nodejs.org/api/child_process.html#child_process_child_process_exec_command_options_callback
     */
    childProcess.exec(command, function(error, standardOutput, standardError) {
      if (error) {
        reject();

        return;
      }

      if (standardError) {
        reject(standardError);

        return;
      }

      resolve(standardOutput);
    });
  });
}

用法

async function main() {
  try {
    const passwdContent = await execute("cat /etc/passwd");

    console.log(passwdContent);
  } catch (error) {
    console.error(error.toString());
  }

  try {
    const shadowContent = await execute("cat /etc/shadow");

    console.log(shadowContent);
  } catch (error) {
    console.error(error.toString());
  }
}

main();

样本输出

root:x:0:0::/root:/bin/bash
[output trimmed, bottom line it succeeded]

Error: Command failed: cat /etc/shadow
cat: /etc/shadow: Permission denied

在线尝试。

复制.it

外部资源

承诺

child_process.exec.

Node.js 支持表

于 2019-08-08T20:35:56.207 回答
21

感谢 Renato 的回答,我创建了一个非常基本的示例:

const exec = require('child_process').exec

exec('git config --global user.name', (err, stdout, stderr) => console.log(stdout))

它只会打印您的全局 git 用户名:)

于 2018-07-17T13:15:42.063 回答
2

您可以使用 nodejs 附带的 util 库从 exec 命令中获取承诺,并可以根据需要使用该输出。使用重组将 stdout 和 stderr 存储在变量中。

const util = require('util');
const exec = util.promisify(require('child_process').exec);

async function lsExample() {
  const {
    stdout,
    stderr
  } = await exec('ls');
  console.log('stdout:', stdout);
  console.error('stderr:', stderr);
}
lsExample();

于 2021-02-25T08:04:19.457 回答
0

你可以使用ShellJS包。
ShellJS 是基于 Node.js API 的 Unix shell 命令的可移植(Windows/Linux/OS X)实现。
见:https ://www.npmjs.com/package/shelljs#execcommand--options--callback

import * as shell from "shelljs";

//usage:
//exec(command [, options] [, callback])

//example:
const version = shell.exec("node --version", {async: false}).stdout;
console.log("nodejs version", version);
于 2021-07-13T05:15:57.040 回答