我正在使用以下脚本从作为参数传递的命令中捕获STDIN
,STDOUT
和STDERR
。
#!/usr/bin/perl
use strict;
use warnings;
use IPC::Open3;
local(*CMD_IN, *CMD_OUT, *CMD_ERR);
my $pid = open3(*CMD_IN, *CMD_OUT, *CMD_ERR, $ARGV[0]);
close(CMD_IN);
my @stdout_output = <CMD_OUT>;
my @stderr_output = <CMD_ERR>;
close(CMD_OUT);
close(CMD_ERR);
waitpid ($pid, 0); # reap the exit code
print "OUT:\n", @stdout_output;
print "ERR:\n", @stderr_output;
这一切都很好,除了我不确定如何监视传递的命令是否挂起。你能建议一个方法吗?
我最初从'Programming Perl'中借用了这个片段。
答案 0 :(得分:5)
您可以使用select
或IO::Select
并提供超时。如果你想从stdout和stderr中读取它们,你应该这样做(参见IPC::Open3
的文档。)
以下是使用IO::Select
的示例程序:
#!/usr/bin/perl
use strict;
use warnings;
use IO::Select;
use IPC::Open3;
use Symbol 'gensym';
my ($cmd_in, $cmd_out, $cmd_err);
$cmd_err = gensym;
my $pid = open3($cmd_in, $cmd_out, $cmd_err, $ARGV[0]);
close($cmd_in);
my $select = IO::Select->new($cmd_out, $cmd_err);
my $stdout_output = '';
my $stderr_output = '';
while (my @ready = $select->can_read(5)) {
foreach my $handle (@ready) {
if (sysread($handle, my $buf, 4096)) {
if ($handle == $cmd_out) {
$stdout_output .= $buf;
}
else {
$stderr_output .= $buf;
}
}
else {
# EOF or error
$select->remove($handle);
}
}
}
if ($select->count) {
print "Timed out\n";
kill('TERM', $pid);
}
close($cmd_out);
close($cmd_err);
waitpid($pid, 0); # reap the exit code
print "OUT:\n", $stdout_output;
print "ERR:\n", $stderr_output;
注意:
gensym
作为stderr句柄。can_read
的参数是以秒为单位的超时。sysread
用于非缓冲IO。答案 1 :(得分:1)
我基于this回答大量提出了以下解决方案。
然而,使用select
并避免使用nwellnhof示例中的信号看起来更清晰,这就是我接受它的原因。如果有人感兴趣,我会在这里张贴:
my $pid = open3(*CMD_IN, *CMD_OUT, *CMD_ERR, $cmd);
if ($pid > 0){
eval{
local $SIG{ALRM} = sub {kill 9, $pid;};
alarm 6;
waitpid($pid, 0);
alarm 0;
};
}