2

我正在使用 popen4 来捕获 stdout、stderr 和命令行的退出状态。只要我能捕捉到上面的这三件事,我就不会被 popen4 束缚。目前我还没有找到捕获命令未找到错误的好方法。我想我可以做一个which cmd预先任务,但希望有一些内置的东西。

您可以在下面运行一个好任务、坏任务和一个假任务来查看差异。我正在rails new apppopen4宝石做这件事

#!/usr/bin/env rake
# Add your own tasks in files placed in lib/tasks ending in .rake,
# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.

require File.expand_path('../config/application', __FILE__)

require 'open4'

# returns exit status 0, all is good
task :convert_good do
  puts "convert good"
  `wget https://www.google.com/images/srpr/logo3w.png`
  status = Open4.popen4("convert logo3w.png output.jpg") do |pid, stdin,stdout,stderr|
    stdin.close
    puts "stdout:"
    stdout.each_line { |line| puts line }
    puts "stderr: #{stderr.inspect}"
    stderr.each_line { |line| puts line }
  end
  puts "status: #{status.inspect}"
  puts "exit:   #{status.exitstatus}"
end

# returns exit status 1, we messed up our command
task :convert_bad do
  puts "convert bad"
  status = Open4.popen4("convert logo3w-asdfasdf.png output.jpg") do |pid, stdin,stdout,stderr|
    stdin.close
    puts "stdout:"
    stdout.each_line { |line| puts line }
    puts "stderr: #{stderr.inspect}"
    stderr.each_line { |line| puts line }
  end
  puts "status: #{status.inspect}"
  puts "exit:   #{status.exitstatus}"
end

# I want this to return exit code 127 for command not found
task :convert_none do
  puts "convert bad"
  status = Open4.popen4("convert_not_installed") do |pid, stdin,stdout,stderr|
    stdin.close
    puts "stdout:"
    stdout.each_line { |line| puts line }
    puts "stderr: #{stderr.inspect}"
    #it doesnt like stderr in this case
    #stderr.each_line { |line| puts line }
  end
  puts "status: #{status.inspect}"
  puts "exit:   #{status.exitstatus}"
end

这是 3 个本地输出

# good
stdout:
stderr: #<IO:fd 11>
status: #<Process::Status: pid 17520 exit 0>
exit:   0

# bad arguments
convert bad
stdout:
stderr: #<IO:fd 11>
convert: unable to open image `logo3w-asdfasdf.png': No such file or directory @ blob.c/OpenBlob/2480.
convert: unable to open file `logo3w-asdfasdf.png' @ png.c/ReadPNGImage/2889.
convert: missing an image filename `output.jpg' @ convert.c/ConvertImageCommand/2800.
status: #<Process::Status: pid 17568 exit 1>
exit:   1

# fake command not found, but returns exit 1 and stderr has no lines
convert bad
stdout:
stderr: #<IO:fd 11>
status: #<Process::Status: pid 17612 exit 1>
exit:   1
4

1 回答 1

5

先说几点。

  1. 您实际上并没有使用popen4 gem - 它是 open4 gem 的包装器至少如果您在 Unix 系统上运行) - 您直接使用open4 gem。如果你想使用popen4,你可以这样称呼它:

    status = POpen4.popen4('cmd') do |stdout, stderr, stdin, pid|
      # ...
    end
    
  2. popen4方法最终通过Kernel#exec方法执行指定的命令,其行为取决于它是否确定给定命令应该在 shell 中运行。(您可以查看http://www.ruby-doc.org/core-1.9.3/Kernel.html#method-i-exec,但这并不是很有帮助。源代码是更好的选择。)

例如:

>  fork { exec "wibble" }
 => 1570 
> (irb):56:in `exec': No such file or directory - wibble (Errno::ENOENT)
    from (irb):56:in `irb_binding'
    from (irb):56:in `fork'
    from (irb):56:in `irb_binding'
    from /Users/evilrich/.rvm/rubies/ree-1.8.7-2011.03/lib/ruby/1.8/irb/workspace.rb:52:in `irb_binding'
    from :0

在这里,exec试图直接执行不存在的命令“wibble”——因此出现了异常。

> fork { exec "wibble &2>1" }
 => 1572 
> sh: wibble: command not found

在这里,exec看到我正在使用重定向,因此在 shell 中执行了我的命令。区别?我在 STDERR 上收到错误,没有例外。您还可以通过在要执行的命令中指定来强制使用 shell:

> fork { exec "sh -c 'wibble -abc -def'" }

无论如何,了解Kernel#exec的行为可能有助于让popen4方法按照您希望的方式运行。

要回答您的问题,如果我使用popen4 gem 并以这样的方式构造命令(根据 exec 的规则)它将在 shell 中运行,或者如果我自己在命令中使用“sh -c ...”,然后我得到了我认为你正在寻找的那种行为:

> status = POpen4.popen4("sh -c 'wibble -abc -def'") {|stdout, stderr, stdin, pid| puts "Pid: #{pid}"}
Pid: 1663
 => #<Process::Status: pid=1663,exited(127)> 
> puts status.exitstatus
127

更新

有趣的。如果您从 stderr 读取,Open4.popen也将返回 127 退出状态。所以,没有必要使用popen gem。

> status = Open4.popen4("sh -c 'wibble -abc -def'") {|pid, stdin, stdout, stderr| stderr.read }
 => #<Process::Status: pid 1704 exit 127> 
于 2013-02-17T01:59:04.363 回答