Open3 throws exception when preceded by fork

I am attempting to create a daemon that (when required) spawns off
child processes to perform "jobs". It is important that this daemon
does not block so it can start concurrent jobs, and that it knows when
jobs have completed. Some of the jobs are written in C++ and send
their output over standard error (stderr).

Here's an outline of my daemon program.

#!/usr/bin/ruby
require 'open3'
stop = false
Signal.trap('CLD') do
  child = Process.wait
  puts "child #{child} closed."
  stop = true
end
child = Kernel.fork()
if(child == nil)
  stdin, stdout, stderr =
Open3.popen3('program_that_sends_output_to_stderr')
  stderr.each do |line|
    puts line
  end
  Kernel.exit!
end
while(! stop)
  sleep 5
end
puts "finished"

When tested separately the fork and popen3 calls seem to be just what
I need to meet my objectives, but when they are run sequentially the
popen3 call throws an exception:
Error: "/usr/lib/ruby/1.8/open3.rb:75:in `waitpid': No child processes
(Errno::ECHILD)"

Is this a limitation of Open3? Any alternative approaches would be
gratefully received.
Thank you.

require 'rubygems'
require 'slave' # gem install slave

class Server
   def run job
     system job
   end
end

slave = Slave.new{ Sever.new }
server = slave.object

server.run 'echo "this is in another process you do not have to ever worry about"'

a @ http://codeforpeople.com/

···

On Jul 14, 2008, at 1:05 PM, trickyvail wrote:

s this a limitation of Open3? Any alternative approaches would be
gratefully received.

--
we can deny everything, except that we have the possibility of being better. simply reflect on that.
h.h. the 14th dalai lama

Thank you for you reply. Unfortunately I am unable to utilize the
slave ruby gem (and open4 gem also) due to some rubygems package
corruption? on my Ubuntu machine. I've given up fighting with it and
come up with a different solution.

daemon:
#!/usr/bin/ruby
stop = false
Signal.trap('CLD') do
  child = Process.wait
  puts "child #{child} closed."
  stop = true
end
exec('./job.rb') if fork.nil?
while(! stop)
  puts 'not finished, do some other stuff'
  sleep 5
end
puts "finished"

job:
#!/usr/bin/ruby
require 'open3'
stdin, stdout, stderr = Open3.popen3('real_job_binary')
stderr.each do |line|
  puts line
end

This way the calls to fork and open3 are isolated in different ruby
interpreter instances where they can't interfere with each other.

···

On Jul 14, 2:52 pm, "ara.t.howard" <ara.t.how...@gmail.com> wrote:

require 'rubygems'
require 'slave' # gem install slave

class Server
   def run job
     system job
   end
end

slave = Slave.new{ Sever.new }
server = slave.object

server.run 'echo "this is in another process you do not have to ever
worry about"'

the tgz files are on rubyforge. just unpack and run

   'sudo ruby install.rb'

cheers.

a @ http://codeforpeople.com/

···

On Jul 14, 2008, at 5:50 PM, trickyvail wrote:

Thank you for you reply. Unfortunately I am unable to utilize the
slave ruby gem (and open4 gem also) due to some rubygems package
corruption? on my Ubuntu machine. I've given up fighting with it and
come up with a different solution.

--
we can deny everything, except that we have the possibility of being better. simply reflect on that.
h.h. the 14th dalai lama