Phusion Passenger + Workling + RabbitMQ

时间:2009-07-06 21:57:12

标签: ruby-on-rails asynchronous rabbitmq workling

我正在尝试部署一个执行某项异步任务的RoR应用程序。我使用workling,消息队列是RabbitMQ。这种组合与Starling完美配合,但我们决定更换MQ for Rabbit。 我在某处读到了我应该在我的environment.rb中包含以下代码

require 'mq' 
if defined?(PhusionPassenger) 
  PhusionPassenger.on_event(:starting_worker_process) do |forked| 
    if forked 
      if EM.reactor_running? 
        EM.stop_event_loop 
        EM.release_machine 
        EM.instance_variable_set( '@reactor_running', false ) 
      end 
      Thread.current[:mq] = nil 
      AMQP.instance_variable_set('@conn', nil) 
    end 
    th = Thread.current 
    Thread.new{ 
      AMQP.connect(:host => 'localhost'){ 
        th.wakeup 
      } 
    } 
    Thread.stop 
  end 
end 

但现在Apache完全失败并显示消息:服务器遇到内部错误或配置错误,无法完成您的请求

2 个答案:

答案 0 :(得分:1)

不是真正的答案,但除非你致力于AMQP,否则我建议使用https://github.com/defunkt/resque - 它可以非常好地完成异步作业+ fork演出。

答案 1 :(得分:1)

编辑:自从发布此代码后,我已经改进了以下代码。可在此处获取:http://www.hiringthing.com/2011/11/04/eventmachine-with-rails.html

我只花了一年时间试图让它发挥作用,最后做到了。这是我的代码:

require 'amqp'
module HiringThingEM
  def self.start
    if defined?(PhusionPassenger)
      PhusionPassenger.on_event(:starting_worker_process) do |forked|
      if forked && EM.reactor_running?
          EM.stop
      end
      Thread.new {
      EM.run do
         AMQP.channel ||= AMQP::Channel.new(AMQP.connect(:host=> Q_SERVER, :user=> Q_USER, :pass => Q_PASS, :vhost => Q_VHOST ))
      end
      }
      die_gracefully_on_signal
      end
    end
  end

  def self.die_gracefully_on_signal
    Signal.trap("INT")  { EM.stop }
    Signal.trap("TERM") { EM.stop }
  end
end

HiringThingEM.start

现在我可以使用:

EM.next_tick { AMQP.channel.queue(Q_Q).publish("hi mom") }

在我的Rails应用程序的控制器内部。

希望这有助于某人。