gpt4 book ai didi

ruby-on-rails - 如何捕获 popen4 找不到的命令

转载 作者:数据小太阳 更新时间:2023-10-29 08:32:54 36 4
gpt4 key购买 nike

我正在使用 popen4 来捕获标准输出、标准错误和命令行的退出状态。只要我能捕捉到上面的那三件事,我就不受 popen4 的束缚。目前我还没有找到捕获命令未发现错误的好方法。我想我可以在预任务中执行 which cmd,但希望有内置的东西。

您可以在下面运行好任务、坏任务和假任务以查看差异。我正在使用 popen4 gem

在全新的 rails new app 中执行此操作
#!/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

最佳答案

首先有几点。

  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 上收到错误,没有异常(exception)。您还可以通过在要执行的命令中指定来强制使用 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

更新

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

> status = Open4.popen4("sh -c 'wibble -abc -def'") {|pid, stdin, stdout, stderr| stderr.read }
=> #<Process::Status: pid 1704 exit 127>

关于ruby-on-rails - 如何捕获 popen4 找不到的命令,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14916478/

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com