this command will generate a whole bunch of output to stdout
pipe.puts “my_script mymodel”
sleep(5) # <— this is what I have to do in order to get the
output
pipe.close_write
output = pipe.read
end
What is annoying is that unless I sleep for 2-5 seconds I don’t get
output or (even worse) my output is cut off. There has to be a way
besides arbitrarily sleeping to ensure that the command is finished.
I’ve tried things like piping in “quit”, but that doesn’t seem to
work. Anybody know how to solve this? I’m happy to use a different
approach if it means I can be sure I get all of my output out. –
Thanks!
I don’t know about pymol, but some apps do not respond well to having
its stdin closed on it with close_write. pipe.read should just block
until you get output, otherwise put an IO.select([pipe) in front of it.
Here is what I’ve worked out for the time being. It opens a thread
for reading the output and asks if there is any output every 1/2
second. Once we go 1/2 second without output we kill the thread.
Here’s a version without threads, should work the same:
my_string = “”
Open3.popen3(“pymol -cq -p”) do |si, so, se|
si.puts “load file.pdb, mymodel\n”
si.puts “run my_script.py\n”
si.puts “myscript mymodel\n”
# await input for 0.5 seconds, will return nil and
# break the loop if there is nothing to read from so after 0.5s
while ready = IO.select([so], nil, nil, 0.5)
# ready.first == so # in this case
# read until the current pipe buffer is empty
begin
my_string << so.read_nonblock(4096)
rescue Errno::EAGAIN
break
end while true
end
sleep(5) # <— this is what I have to do in order to get the
approach if it means I can be sure I get all of my output out. –
IO.popen(“pymol -cq -p”, ‘w+’) do |pipe|
Eric W.
both of those hang indefinitely. It doesn’t hang if I add the line:
pipe.puts “quit”
but that doesn’t give me all the output either. So maybe pymol
behaves differently than many commandline programs?
Here is what I’ve worked out for the time being. It opens a thread
for reading the output and asks if there is any output every 1/2
second. Once we go 1/2 second without output we kill the thread.
my_string = “”
Open3.popen3(“pymol -cq -p”) do |si, so, se|
si.puts “load file.pdb, mymodel\n”
si.puts “run my_script.py\n”
si.puts “myscript mymodel\n”
forstdout = Thread.new do
Thread.current[‘lines’] = []
while line = so.gets
Thread.current[‘lines’] << line
end
end
past_size = -1
loop do
sleep(0.5)
current_size = forstdout[‘lines’].size
break if current_size == past_size
past_size = current_size
end
my_string = forstdout[‘lines’]
forstdout.kill
end
This is not the fastest method, but it does seem to work OK. I would
still love to hear other ideas.
@Robert: I’ve looked through a lot of pymol docs and haven’t found
anything on it yet, will post if I do.
The solution you post looks very intriguing. It runs fine, but closes
before I get back the output I need.
Frankly, I don’t see how this can be: the reader thread should read everthing that comes out of the pipe and only stops if there is EOF.
The only explanation I have so far is that maybe some CTRL-D or other
EOF character is sent through the pipe making the reader think it has
reached EOF.
It seems like there must be some
kind of a pause to let pymol start spitting out output. I may play
around with it some to see if I can get it working.
Even if there is a pause the reader must wait. There must be something
else happening. Maybe try to do it on the shell and use ‘od’ to see
what is sent across the pipe.
@Eric: this solution works nicely and returns all the expected
output. I will be using it unless something better comes along.
Good that you have a working solution. However, what you state worries
me a bit. I believe we might not have identified the root cause of the
issue.
Thanks to both of you for your excellent input on this. --John
@Robert: I’ve looked through a lot of pymol docs and haven’t found
anything on it yet, will post if I do.
The solution you post looks very intriguing. It runs fine, but closes
before I get back the output I need. It seems like there must be some
kind of a pause to let pymol start spitting out output. I may play
around with it some to see if I can get it working.
@Eric: this solution works nicely and returns all the expected
output. I will be using it unless something better comes along.
Thanks to both of you for your excellent input on this. --John
This forum is not affiliated to the Ruby language, Ruby on Rails framework, nor any Ruby applications discussed here.