module ActionCable
module Connection
class Base
include Identification
include InternalChannel
PING_INTERVAL = 3
attr_reader :server, :env
delegate :worker_pool, :pubsub, to: :server
attr_reader :subscriptions
attr_reader :logger
def initialize(server, env)
@started_at = Time.now
@server, @env = server, env
@accept_messages = false
@pending_messages = []
@logger = TaggedLoggerProxy.new(server.logger, tags: log_tags)
@subscriptions = ActionCable::Connection::Subscriptions.new(self)
end
def process
logger.info started_request_message
if websocket?
@websocket = Faye::WebSocket.new(@env)
@websocket.on(:open) do |event|
transmit_ping_timestamp
@ping_timer = EventMachine.add_periodic_timer(PING_INTERVAL) { transmit_ping_timestamp }
worker_pool.async.invoke(self, :on_open)
end
@websocket.on(:message) do |event|
message = event.data
if message.is_a?(String)
if @accept_messages
worker_pool.async.invoke(self, :receive, message)
else
@pending_messages << message
end
end
end
@websocket.on(:close) do |event|
logger.info finished_request_message
worker_pool.async.invoke(self, :on_close)
EventMachine.cancel_timer(@ping_timer) if @ping_timer
end
@websocket.rack_response
else
invalid_request
end
end
def receive(data_in_json)
if websocket_alive?
data = ActiveSupport::JSON.decode data_in_json
case data['command']
when 'subscribe' then subscriptions.add data
when 'unsubscribe' then subscriptions.remove data
when 'message' then process_message data
else
logger.error "Received unrecognized command in #{data.inspect}"
end
else
logger.error "Received data without a live websocket (#{data.inspect})"
end
end
def transmit(data)
@websocket.send data
end
def close
logger.error "Closing connection"
@websocket.close
end
def statistics
{
identifier: connection_identifier,
started_at: @started_at,
subscriptions: subscriptions.identifiers
}
end
protected
def request
@request ||= ActionDispatch::Request.new(Rails.application.env_config.merge(env))
end
def cookies
request.cookie_jar
end
private
def on_open
server.add_connection(self)
connect if respond_to?(:connect)
subscribe_to_internal_channel
@accept_messages = true
worker_pool.async.invoke(self, :receive, @pending_messages.shift) until @pending_messages.empty?
end
def on_close
server.remove_connection(self)
subscriptions.cleanup
unsubscribe_from_internal_channel
disconnect if respond_to?(:disconnect)
end
def transmit_ping_timestamp
transmit({ identifier: '_ping', message: Time.now.to_i }.to_json)
end
def process_message(message)
subscriptions.find(message['identifier']).perform_action(ActiveSupport::JSON.decode(message['data']))
rescue Exception => e
logger.error "Could not process message (#{message.inspect})"
log_exception(e)
end
def invalid_request
logger.info finished_request_message
[404, {'Content-Type' => 'text/plain'}, ['Page not found']]
end
def websocket_alive?
@websocket && @websocket.ready_state == Faye::WebSocket::API::OPEN
end
def websocket?
@is_websocket ||= Faye::WebSocket.websocket?(@env)
end
def started_request_message
'Started %s "%s"%s for %s at %s' % [
request.request_method,
request.filtered_path,
websocket? ? ' [Websocket]' : '',
request.ip,
Time.now.to_default_s ]
end
def finished_request_message
'Finished "%s"%s for %s at %s' % [
request.filtered_path,
websocket? ? ' [Websocket]' : '',
request.ip,
Time.now.to_default_s ]
end
def log_exception(e)
logger.error "There was an exception: #{e.class} - #{e.message}"
logger.error e.backtrace.join("\n")
end
def log_tags
server.log_tags.map { |tag| tag.respond_to?(:call) ? tag.call(request) : tag.to_s.camelize }
end
end
end
end