如何使用ActionController :: Live以及Resque + Redis(用于聊天应用程序)

时间:2015-03-19 16:41:46

标签: ruby-on-rails ruby ruby-on-rails-3 redis resque

我正在尝试为我的rails应用程序构建聊天功能。我正在使用ActionController::LivePumaResqueRedis。所以基本上在这种情况下,redis subscribe方法使用resque在后​​台运行。 到目前为止,我所做的是每当用户在下面的表单字段(即聊天框)中输入文本时

    <%= form_tag chat_box_publish_path, method: :get do %>
        <%= text_field_tag :query, params[:query], class: "form-control", id: "chatSearchBox",
            placeholder: 'Search' %>
    <% end %>

..请求来自Publish中的ChatBoxController方法。

def publish
    $redis.publish("chat_message:1:1", "#{params[:query]}")
    respond_to do |format|
        format.js {render nothing: true}
    end
end

..现在我有一个以下后台Resque作业运行以下代码用于测试目的。因此,每当发布聊天消息时,它都会打印data,这很好。但是,如何将ActionController::Live功能添加到后台作业?或者我如何进行此实施?需要有关此设计的帮助。

class ChatBoxInitiator
    @queue = :chat_box_initiator

    private
    def self.perform
    $redis.subscribe('chat_message:1:1') do |on|
            on.message do |event, data|
                puts "====#{data}"
                return data
            end
        end
    end
end

我想在Server Sent Events(SSE)

中显示ActionController::LiveUsers/show的通知

1 个答案:

答案 0 :(得分:13)

预REQS:

  • Ruby 2.0.0 +
  • Rails 4.0.0 +
  • Redis的
  • 彪马

初始化器:

redis.rb目录中创建config/initializers初始化文件,全球化redis的实例。设置heartbeat线程也是一个好主意(根据您的要求,5秒至5分钟的任何时间都可以):

$redis = Redis.new

heartbeat_thread = Thread.new do
  while true
    $redis.publish("heartbeat","thump")
    sleep 15.seconds
  end
end

at_exit do
  heartbeat_thread.kill
  $redis.quit
end

控制器:

您需要向ChatControllerpubsub添加两种方法。 pub的作用是将聊天事件和消息发布到redis,并sub订阅这些事件。看起来应该是这样的:

class ChatController < ApplicationController
    include ActionController::Live

    skip_before_filter  :verify_authenticity_token

    def index
    end

    def pub
        $redis.publish 'chat_event', params[:chat_data].to_json
        render json: {}, status: 200
    end

    def sub
        response.headers["Content-Type"] = "text/event-stream"

        redis = Redis.new
        redis.subscribe(['chat_event', 'heartbeat']) do |on|
            on.message do |event, data|
                response.stream.write "event: #{event}\ndata: #{data}\n\n"
            end
        end
    rescue IOError
        logger.info "Stream Closed"
    ensure
        redis.quit
        response.stream.close
    end
end

routes中,将 pub 设为POST sub a GET,并将路径与/chat/publish/chat/subscribe匹配。


Coffeescript / Javascript:

假设您的聊天应用的实际网页位于/chat,您需要编写一些 Javascript 来实际发送和接收聊天消息。

为了便于理解,我们假设您的网页只有一个文本框和一个按钮。点击按钮应该将文本框的内容发布到聊天流,我们可以使用AJAX:

$('button#send').click (e) ->
    e.preventDefault()
    $.ajax '/chat/publish',
        type: 'POST'
        data:
            chat_data: {
                message: $("input#message").val()
                timestamp: $.now()
        error: (jqXHR, textStatus, errorThrown) ->
            console.log "Failed: " + textStatus 
        success: (data, textStatus, jqXHR) ->
            console.log "Success: " + textStatus

现在,您还需要能够订阅和接收聊天消息。您需要使用EventSource。使用 EventSource ,打开SSE的通道,以便您可以接收事件,并使用该数据更新视图。在这个例子中,我们只将它们记录到javascript控制台。

代码看起来像这样:

$(document).ready ->
    source = new EventSource('/chat/subscribe')
    source.addEventListener 'chat_event', (e) ->
        console.log(e.data)

注意: 将上面的两个代码块放在controllername.coffee文件中,对于此示例,它应为chat.js.coffee < / em>在您的app/assets/javascript目录中。您还需要确保它已在资产管道中加载。 requireapplication.js文件中(如果您还没有调用require tree .)。{/ p>


启用并行请求:

在您的开发环境中,您必须通过将这两行添加到config/environments/development.rb来启用并行请求:

config.preload_frameworks = true
config.allow_concurrency = true

现在启动浏览器,浏览到/chat并查看魔法。当您键入消息并单击该按钮时,该网页的所有实例都将收到该消息。


这就是您使用railsActionController::LiveRedis中制作基本聊天应用程序的方法。根据您的要求,最终的代码显然会有很大不同,但这应该可以帮助您入门。

您应该查看更多资源: