Faye ruby客户端只发布一次



我有一个运行在localhost上的faye服务器(nodejs),我正在尝试设置一个需要定期在服务器上发布的服务器端ruby客户端。这是我尝试使用的代码。(请忽略要开始的注释代码)
我创建了一个类变量@@client,并在类加载后立即对其进行初始化。我定义了一个类方法pub,其任务是在faye服务器上发布一些内容
最后,我只调用了两次pub方法。成功接收到第一个发布回调,但第二个发布不包含callbackerrback。由于控制权没有还给应用程序,应用程序就挂在那里
如果我将gobal变量设为$client(当前已注释),则行为是相同的。但是,如果每次调用pub时我都创建客户端,那么发布就会顺利进行。我在EM.run循环或外部启动它,行为相同。(如预期)

我不想每次我想发布一些东西时都建立新的连接,因为这违背了目的。此外,如果每次调用该方法时都在EM.run中创建一个新客户端,则客户端连接不会自行关闭。我可以在lsof命令中看到它们作为打开文件打开,我想很快就会出现too many open files错误。

我真的没有正确理解事件机器,也许我错过了一些东西。

require 'faye'
require 'eventmachine'
# $client = Faye::Client.new('http://localhost:5050/faye')
class Fayeclient
  puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
  # if !defined? @@client or @@client.nil?
    @@client = Faye::Client.new('http://localhost:5050/faye')
    puts "Created client: " + @@client.inspect
  # end
  def self.pub
    puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
    # client = Faye::Client.new('http://localhost:5050/faye') #$client
    # client = @@client
    EM.run {
      #client = Faye::Client.new('http://localhost:5050/faye') #$client
      puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
      puts @@client.inspect
      publication = @@client.publish('/foo', 'text' =>'Hello world')
      puts "Publishing: #{publication.inspect}"
      # puts "Publication methods: #{publication.methods}"
      publication.callback do
        puts "Did it #{publication.inspect}"
        EM.stop_event_loop
        puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
        # puts "#{client.methods}"
#        puts client.inspect
#        client.remove_all_listeners
#        puts client.inspect
      end
      publication.errback do |error |
          puts error.inspect
          EM.stop_event_loop
      end
    }
    puts "Outside event loop"
    puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
  end
end
Fayeclient.pub
Fayeclient.pub

EM.run调用是阻塞的,您必须在一个单独的线程上运行它,并最终在一切结束后加入它。在这个例子中,我使用Singleton,但这取决于你。

这正确地完成了2个faye调用。

#!/usr/bin/env ruby
#
require 'faye'
require 'singleton'
require 'eventmachine'

class Fayeclient
  include Singleton
  attr_accessor :em_thread, :client
  def initialize
    self.em_thread = Thread.new do
      EM.run
    end
    self.client = Faye::Client.new('http://localhost:8890/faye')
  end
  def pub
    puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
    puts client.inspect
    publication = client.publish('/foo', 'text' =>'Hello world')
    puts "Publishing: #{publication.inspect}"
    publication.callback do
      puts "Did it #{publication.inspect}"
      EM.stop_event_loop
      puts "#{__LINE__}: Reactor running: " + EM.reactor_running?.to_s
    end
    publication.errback do |error |
        puts error.inspect
        EM.stop_event_loop
    end
  end
end
Fayeclient.instance.pub
Fayeclient.instance.pub
Fayeclient.instance.em_thread.join

无论如何,根据我的个人经验,必须在Rails应用程序中处理EventMachine可能会一团糟,有些Web服务器使用EM,另一些则不使用EM,当您想从控制台进行测试时,它可能无法按预期工作。

我的解决方案是回退到http调用:

RestClient.post "http://localhost:#{Rails.configuration.faye_port}/faye", message: {foo: 'bar'}.to_json

如果您不需要从这段代码中接收消息,我发现这个解决方案更简单、更易于定制。

相关内容

  • 没有找到相关文章

最新更新