如何使用Unix管理Node.js脚本管道(在命令行上)?

时间:2013-05-03 00:10:31

标签: javascript node.js unix pipe command-line-interface

我看到如何使用Node.js流将内容组合在一起,但是如何使用Unix |将多个脚本一起管道,假设其中一些脚本可以是异步的?

$ ./a.js | ./b.js

示例:

a.js (chmod 0755)

#!/usr/bin/env node

setTimeout(function(){
  console.log(JSON.stringify({ foo: 'bar' }));
}, 10);

b.js (chmod 0755)

#!/usr/bin/env node

console.log(process.argv);

这是输出:

$ ./a.js | ./b.js
[ 'node', '/Users/viatropos/tests/b.js' ]

events.js:72
        throw er; // Unhandled 'error' event
              ^
Error: write EPIPE
    at errnoException (net.js:883:11)
    at Object.afterWrite (net.js:700:19)
乍一看似乎有很多问题,所以不确定从哪里开始。有没有办法让这个工作?最终目标是能够从console.log获取./a.js输出并在./b.js中使用它。原因是,大多数情况下这些脚本将一次运行一个,但有时将它们组合在一起会很好,所以理想情况下系统应该能够处理这两种情况。

1 个答案:

答案 0 :(得分:51)

问题是你的b.js会立即结束并关闭其标准,导致a.js出错,因为它的标准输出被关闭而你没有处理这种可能性。您有两种选择:在a.js中处理stdout结束或接受b.js中的输入。

修复a.js

process.on("SIGPIPE", process.exit);

如果你添加那一行,那么当没有人再读取它的输出时它就会放弃。 SIGPIPE可能有更好的事情要做,具体取决于你的程序在做什么,但关键是要停止console.log

修复b.js

#!/usr/bin/env node

var stdin = process.openStdin();

var data = "";

stdin.on('data', function(chunk) {
  data += chunk;
});

stdin.on('end', function() {
  console.log("DATA:\n" + data + "\nEND DATA");
});

当然,您 无法对该数据执行任何操作。关键是要有一些东西来保持流程的运行;如果你正在使用它,stdin.on('data', fx)似乎是一件很有用的事情。

请记住,其中任何一个都可以防止该错误。如果你计划在程序之间进行管道,我希望第二个最有用。