我正在使用:
- 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继续完成而不等待它)
退出
我发现了以下内容:
有没有办法解决这个问题? (等待Open3子进程,以便我得到正确的输出)
提前致谢
答案 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 。