Ruby - Open3没有完成子进程

时间:2014-10-27 01:55:37

标签: ruby child-process popen3

我正在使用:

- Ruby 1.9.3-p448
- Windows Server 2008

我有一个包含程序使用的命令的文件,我以这种方式使用它

C:\> PATH_TO_FOLDER/program.exe file.txt

File.txt有一些命令,所以" program.exe"将执行以下操作:

- Execute commands 
- Reads from a DB using an ODBC method used by program
- Outputs result in a txt file

使用powershell这个命令可以正常工作。

现在我把它放在一个文件(app.rb)

require 'sinatra'
require 'open3'

get '/process' do
  program_path = "path to program.exe"
  file_name = "file.txt"
  Open3.popen3(program_path, file_name) do |i, o, e, w|
    # I have some commands here to execute but just as an example I'm using o.read
    puts o.read
  end
end

现在通过访问http://localhost/process来使用此功能时,Open3可以通过这样做(我不是100%肯定,但经过多次尝试后我认为是唯一的选择)

  • 读取命令并执行它们(这没关系)

  • 尝试使用ODBC方法从DB读取(这是我的问题。我 需要从Open3接收一些输出,所以我可以在浏览器中显示它,但我想当它试图读取它时会启动Open3不知道的另一个进程,所以Open3继续完成而不等待它)

  • 退出

退出

我发现了以下内容:

  • 使用Thread.join(在本例中为w.join)以等待进程完成,但它不起作用
  • Open4似乎处理子进程但不适用于Windows
  • Process.wait(pid),在这种情况下为pid = w.pid,但也不起作用
  • Timeout.timeout(n),这里的问题是我不确定多久 会不会。

有没有办法解决这个问题? (等待Open3子进程,以便我得到正确的输出)

提前致谢

2 个答案:

答案 0 :(得分:0)

我们在退出状态方面遇到了类似的问题,这就是我们做的事情

Open3.popen3(*cmd) do |stdin, stdout, stderr, wait_thr|

  # print stdout and stderr as it comes in
  threads = [stdout, stderr].collect do |output|
    Thread.new do
      while ((line = output.gets rescue '') != nil) do
        unless line.blank?
          puts line
        end
      end
    end
  end

  # get exit code as a Process::Status object
  process_status = wait_thr.value #.exitstatus

  # wait for logging threads to finish before continuing
  # so we don't lose any logging output
  threads.each(&:join)

  # wait up to 5 minutes to make sure the process has really exited
  Timeout::timeout(300) do
    while !process_status.exited?
      sleep(1)
    end
  end rescue nil

  process_status.exitstatus.to_i
end

答案 1 :(得分:0)

使用Open3.popen3对于琐碎的案例很容易。我不知道处理子进程的输入,输出和错误通道的真实代码。我也不知道你的子进程的确切行为:它是否写在stdout上?它写在stderr上吗?它试图从stdin读取吗?

这就是为什么我假设您用puts o.read替换的代码中存在问题。 关于您可能遇到的问题的一个很好的总结是http://coldattic.info/shvedsky/pro/blogs/a-foo-walks-into-a-bar/posts/63

虽然我不同意文章的作者Pavel Shved在寻找解决方案时的观点。他推荐自己的解决方案。我只是在我的项目中使用popen3的一个包装函数:Open3.capture*。他们做所有困难的事情,比如同时等待stdout和stderr