我看到如何使用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
中使用它。原因是,大多数情况下这些脚本将一次运行一个,但有时将它们组合在一起会很好,所以理想情况下系统应该能够处理这两种情况。
答案 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)
似乎是一件很有用的事情。
请记住,其中任何一个都可以防止该错误。如果你计划在程序之间进行管道,我希望第二个最有用。