我正在尝试从Perl脚本检查SVN标记是否存在。所以我尝试调用svn info $url
,读取退出代码并禁止标准输出和标准错误流。但是,我很难优雅地做到这一点(可能有更好的方式向SVN询问标签,但这不是重点:)
my $output = `svn info $url/tags/$tag`;
这会在将输出放入$output
时抑制输出。退出代码丢失。
my $output = `svn info $url/tags/$tag 2>&1`;
这会抑制STDERR和STDOUT,并将它们都放入$output
。退出代码再次丢失。
my $exitcode = system("svn", "info", "$url/tags/$tag");
这会捕获退出代码,但实际输出和错误流对用户可见。
open( STDERR, q{>}, "/dev/null" );
open my $fh, q{>}, "/dev/null";
select($fh);
if (system("svn", "info", "$url/tags/$tag") != 0) {
select(STDOUT);
print ("Tag doesn't exist!");
do_something_with_exit();
}
select(STDOUT);
print "Exit code: $exitcode";
这会杀死STDOUT和STDERR并捕获退出代码,但它很难看,因为我必须记住将STDOUT切换回原来的状态。
那么,还有更优雅的解决方案吗?
答案 0 :(得分:8)
尝试使用$?
。
my $output = `svn info $url/tags/$tag`;
my $extcode = $?>>8;
答案 1 :(得分:3)
使用IPC::System::Simple尝试时会发生什么?该模块处理了这些问题的大部分细节:
use IPC::System::Simple qw(capturex $EXITVAL);
my $output = capturex( "some_command", @args );
my $exit = $EXITVAL;
答案 2 :(得分:1)
my $output = `svn info $url/tags/$tag 2>&1`;
这会抑制STDERR和STDOUT,并将它们都放入$ output。退出代码再次丢失
您确定退出代码丢失了吗?当我尝试这个时,我会在$?
中获得退出代码。
答案 3 :(得分:0)
模块IPC::Run3对输入和输出进行了非常细粒度的控制。
use IPC::Run3;
run3 \@cmd, \$in, \$out, \$err;
您可以将同一个变量传递给\$out
和\$err
,它会完成您所期望的,并将两个流组合在一起。输入是不必要的,因此您可以传递undef
(“从父进程继承”)或\undef
(“关闭文件句柄”)。
IPC::Run3::run3()
根据退出代码返回true或false,并根据'perlvar'将子进程的实际退出代码保留在$?
中。
在你的情况下你会运行
use IPC::Run3
my @cmd = ('svn', 'info', "$url/tags/$tag");
my $out;
my $rv = run3(\@cmd, \undef, \$out, \$out);
if ($rv) {
# process $out
}
else {
die "error: $@";
}