ruby popen3 - 如何反复写入stdin&读取标准输出而不重新打开过程?

时间:2014-04-04 01:01:39

标签: ruby popen3

我使用Open3popen3方法启动一个以类似控制台/ REPL方式运行的进程,以反复接受输入和返回输出。

我可以打开进程,发送输入,并接收输出就好了,代码如下:

Open3.popen3("console_REPL_process") do |stdin, stdout, stderr, wait_thr|
    stdin.puts "a string of input"
    stdin.close_write
    stdout.each_line { |line| puts line } #successfully prints all the output
end

我希望连续多次这样做,而无需重新打开过程,因为启动需要很长时间。

我知道我必须关闭标准输入以便stdout返回..但我不知道的是,如何重新打开'stdin以便我可以写更多输入?

理想情况下,我想做这样的事情:

Open3.popen3("console_REPL_process") do |stdin, stdout, stderr, wait_thr|
    stdin.puts "a string of input"
    stdin.close_write
    stdout.each_line { |line| puts line }

    stdin.reopen_somehow()

    stdin.puts "another string of input"
    stdin.close_write
    stdout.each_line { |line| puts line }
    # etc..
end

溶液

感谢pmoo的回答,我能够使用PTYexpect设计一个解决方案,期待进程在准备好输入时返回的提示字符串,如下所示:

PTY.spawn("console_REPL_process") do |output, input|
    output.expect("prompt >") do |result|
      input.puts "string of input"
    end
    output.expect("prompt >") do |result|
      puts result
      input.puts "another string of input"
    end
    output.expect("prompt >") do |result|
      puts result
      input.puts "a third string of input"
    end
    # and so forth
end

1 个答案:

答案 0 :(得分:3)

您可以使用expect库取得一些成功,并让子进程明确标记每个输出的结尾,例如:

require 'expect'
require 'open3'

Open3.popen3("/bin/bash") do
    | input, output, error, wait_thr |
    input.sync = true
    output.sync = true

    input.puts "ls /tmp"
    input.puts "echo '----'"
    puts output.expect("----", 5)

    input.puts "cal apr 2014"
    input.puts "echo '----'"
    puts output.expect("----", 5)
end

作为奖励,expect有一个timeout选项。