Class: ActionCable::Connection::Base
Relationships & Source Files | |
Super Chains via Extension / Inclusion / Inheritance | |
Class Chain:
|
|
Instance Chain:
|
|
Inherits: | Object |
Defined in: | actioncable/lib/action_cable/connection/base.rb |
Overview
For every WebSocket
connection the Action Cable server accepts, a ::ActionCable::Connection
object will be instantiated. This instance becomes the parent of all of the channel subscriptions that are created from there on. Incoming messages are then routed to these channel subscriptions based on an identifier sent by the Action Cable consumer. The Connection itself does not deal with any specific application logic beyond authentication and authorization.
Here’s a basic example:
module ApplicationCable
class Connection < ActionCable::Connection::Base
identified_by :current_user
def connect
self.current_user = find_verified_user
logger. current_user.name
end
def disconnect
# Any cleanup work needed when the cable connection is cut.
end
private
def find_verified_user
User.find_by_identity( .encrypted[:identity_id]) ||
end
end
end
First, we declare that this connection can be identified by its current_user. This allows us to later be able to find all connections established for that current_user (and potentially disconnect them). You can declare as many identification indexes as you like. Declaring an identification means that an attr_accessor is automatically set for that key.
Second, we rely on the fact that the WebSocket
connection is established with the cookies from the domain being sent along. This makes it easy to use signed cookies that were set when logging in via a web interface to authorize the WebSocket
connection.
Finally, we add a tag to the connection-specific logger with the name of the current user to easily distinguish their messages in the log.
Pretty simple, eh?
Constant Summary
::ActiveSupport::Callbacks
- Included
Identification
- Attributes & Methods
::ActiveSupport::Rescuable
- Attributes & Methods
Class Method Summary
Instance Attribute Summary
- #config readonly
- #env readonly
- #event_loop readonly
- #logger readonly
- #protocol readonly
- #pubsub readonly
- #server readonly
- #subscriptions readonly
- #worker_pool readonly
- #allow_request_origin? ⇒ Boolean readonly private
- #message_buffer readonly private
- #websocket readonly private
Callbacks
- Included
Instance Method Summary
- #beat
-
#close(reason: nil, reconnect: true)
Close the
WebSocket
connection. - #handle_channel_command(payload)
-
#send_async(method, *arguments)
Invoke a method on the connection asynchronously through the pool of thread workers.
-
#statistics
Return a basic hash of statistics for the connection keyed with
identifier
,started_at
, #subscriptions, andrequest_id
. -
#cookies
private
The cookies of the request that initiated the
WebSocket
connection. - #decode(websocket_message) private
- #encode(cable_message) private
- #finished_request_message private
- #handle_close private
- #handle_open private
- #invalid_request_message private
-
#new_tagged_logger
private
Tags are declared in the server but computed in the connection.
-
#request
private
The request that initiated the
WebSocket
connection is available here. - #respond_to_invalid_request private
- #respond_to_successful_request private
- #send_welcome_message private
- #started_request_message private
- #successful_request_message private
- #dispatch_websocket_message(websocket_message) Internal use only
- #inspect Internal use only
- #on_close(reason, code) Internal use only
- #on_error(message) Internal use only
- #on_message(message) Internal use only
- #on_open Internal use only
-
#process
Internal use only
Called by the server when a new
WebSocket
connection is established. -
#receive(websocket_message)
Internal use only
Decodes WebSocket messages and dispatches them to subscribed channels.
- #transmit(cable_message) Internal use only
::ActiveSupport::Rescuable
- Included
#rescue_with_handler | Delegates to the class method, but uses the instance as the subject for rescue_from handlers (method calls, |
#handler_for_rescue | Internal handler lookup. |
::ActiveSupport::Callbacks
- Included
#run_callbacks | Runs the callbacks for the given event. |
#halted_callback_hook | A hook invoked every time a before callback is halted. |
Authorization
- Included
#reject_unauthorized_connection | Closes the |
InternalChannel
- Included
#internal_channel, #process_internal_message, #subscribe_to_internal_channel, #unsubscribe_from_internal_channel |
Identification
- Included
#connection_identifier | Return a single connection identifier that combines the value of all the registered identifiers into a single gid. |
#connection_gid |
Constructor Details
.new(server, env, coder: ActiveSupport::JSON) ⇒ Base
# File 'actioncable/lib/action_cable/connection/base.rb', line 67
def initialize(server, env, coder: ActiveSupport::JSON) @server, @env, @coder = server, env, coder @worker_pool = server.worker_pool @logger = new_tagged_logger @websocket = ActionCable::Connection::WebSocket.new(env, self, event_loop) @subscriptions = ActionCable::Connection::Subscriptions.new(self) @message_buffer = ActionCable::Connection::MessageBuffer.new(self) @_internal_subscriptions = nil @started_at = Time.now end
Class Attribute Details
.identifiers (rw)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/identification.rb', line 13
class_attribute :identifiers, default: Set.new
.identifiers? ⇒ Boolean
(rw)
[ GitHub ]
# File 'actioncable/lib/action_cable/connection/identification.rb', line 13
class_attribute :identifiers, default: Set.new
.rescue_handlers (rw)
[ GitHub ]# File 'activesupport/lib/active_support/rescuable.rb', line 15
class_attribute :rescue_handlers, default: []
.rescue_handlers? ⇒ Boolean
(rw)
[ GitHub ]
# File 'activesupport/lib/active_support/rescuable.rb', line 15
class_attribute :rescue_handlers, default: []
Instance Attribute Details
#allow_request_origin? ⇒ Boolean
(readonly, private)
[ GitHub ]
# File 'actioncable/lib/action_cable/connection/base.rb', line 228
def allow_request_origin? return true if server.config.disable_request_forgery_protection proto = Rack::Request.new(env).ssl? ? "https" : "http" if server.config.allow_same_origin_as_host && env["HTTP_ORIGIN"] == "#{proto}://#{env['HTTP_HOST']}" true elsif Array(server.config.allowed_request_origins).any? { |allowed_origin| allowed_origin === env["HTTP_ORIGIN"] } true else logger.error("Request origin not allowed: #{env['HTTP_ORIGIN']}") false end end
#config (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 65
delegate :event_loop, :pubsub, :config, to: :server
#env (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
#event_loop (readonly)
[ GitHub ]#identifiers (rw)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/identification.rb', line 13
class_attribute :identifiers, default: Set.new
#identifiers? ⇒ Boolean
(rw)
[ GitHub ]
# File 'actioncable/lib/action_cable/connection/identification.rb', line 13
class_attribute :identifiers, default: Set.new
#logger (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
#message_buffer (readonly, private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 174
attr_reader :
#protocol (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
#pubsub (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 65
delegate :event_loop, :pubsub, :config, to: :server
#rescue_handlers (rw)
[ GitHub ]# File 'activesupport/lib/active_support/rescuable.rb', line 15
class_attribute :rescue_handlers, default: []
#rescue_handlers? ⇒ Boolean
(rw)
[ GitHub ]
# File 'activesupport/lib/active_support/rescuable.rb', line 15
class_attribute :rescue_handlers, default: []
#server (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
#subscriptions (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
#websocket (readonly, private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 173
attr_reader :websocket
#worker_pool (readonly)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 64
attr_reader :server, :env, :subscriptions, :logger, :worker_pool, :protocol
Instance Method Details
#beat
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 147
def beat transmit type: ActionCable::INTERNAL[: ][:ping], message: Time.now.to_i end
#close(reason: nil, reconnect: true)
Close the WebSocket
connection.
# File 'actioncable/lib/action_cable/connection/base.rb', line 120
def close(reason: nil, reconnect: true) transmit( type: ActionCable::INTERNAL[: ][:disconnect], reason: reason, reconnect: reconnect ) websocket.close end
#cookies (private)
The cookies of the request that initiated the WebSocket
connection. Useful for performing authorization checks.
# File 'actioncable/lib/action_cable/connection/base.rb', line 187
def # :doc: request. end
#decode(websocket_message) (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 195
def decode( ) @coder.decode end
#dispatch_websocket_message(websocket_message)
# File 'actioncable/lib/action_cable/connection/base.rb', line 101
def ( ) # :nodoc: if websocket.alive? handle_channel_command decode( ) else logger.error "Ignoring message processed after the WebSocket was closed: #{ .inspect})" end end
#encode(cable_message) (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 191
def encode( ) @coder.encode end
#finished_request_message (private)
[ GitHub ]#handle_channel_command(payload)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 109
def handle_channel_command(payload) run_callbacks :command do subscriptions.execute_command payload end end
#handle_close (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 211
def handle_close logger.info server.remove_connection(self) subscriptions.unsubscribe_from_all unsubscribe_from_internal_channel disconnect if respond_to?(:disconnect) end
#handle_open (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 199
def handle_open @protocol = websocket.protocol connect if respond_to?(:connect) subscribe_to_internal_channel .process! server.add_connection(self) rescue ActionCable::Connection::Authorization::UnauthorizedError close(reason: ActionCable::INTERNAL[:disconnect_reasons][: ], reconnect: false) if websocket.alive? end
#inspect
# File 'actioncable/lib/action_cable/connection/base.rb', line 168
def inspect # :nodoc: "#<#{self.class.name}:#{'%#016x' % (object_id << 1)}>" end
#invalid_request_message (private)
[ GitHub ]#new_tagged_logger (private)
Tags are declared in the server but computed in the connection. This allows us per-connection tailored tags.
#on_close(reason, code)
# File 'actioncable/lib/action_cable/connection/base.rb', line 164
def on_close(reason, code) # :nodoc: send_async :handle_close end
#on_error(message)
# File 'actioncable/lib/action_cable/connection/base.rb', line 159
def on_error( ) # :nodoc: # log errors to make diagnosing socket errors easier logger.error "WebSocket error occurred: #{}" end
#on_message(message)
# File 'actioncable/lib/action_cable/connection/base.rb', line 155
def ( ) # :nodoc: .append end
#on_open
# File 'actioncable/lib/action_cable/connection/base.rb', line 151
def on_open # :nodoc: send_async :handle_open end
#process
Called by the server when a new WebSocket
connection is established. This configures the callbacks intended for overwriting by the user. This method should not be called directly – instead rely upon on the #connect
(and #disconnect
) callbacks.
# File 'actioncable/lib/action_cable/connection/base.rb', line 85
def process # :nodoc: logger.info if websocket.possible? && allow_request_origin? respond_to_successful_request else respond_to_invalid_request end end
#receive(websocket_message)
Decodes WebSocket messages and dispatches them to subscribed channels. WebSocket
message transfer encoding is always JSON.
# File 'actioncable/lib/action_cable/connection/base.rb', line 97
def receive( ) # :nodoc: send_async :, end
#request (private)
The request that initiated the WebSocket
connection is available here. This gives access to the environment, cookies, etc.
# File 'actioncable/lib/action_cable/connection/base.rb', line 178
def request # :doc: @request ||= begin environment = Rails.application.env_config.merge(env) if defined?(Rails.application) && Rails.application ActionDispatch::Request.new(environment || env) end end
#respond_to_invalid_request (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 247
def respond_to_invalid_request close(reason: ActionCable::INTERNAL[:disconnect_reasons][:invalid_request]) if websocket.alive? logger.error logger.info [ 404, { Rack::CONTENT_TYPE => "text/plain; charset=utf-8" }, [ "Page not found" ] ] end
#respond_to_successful_request (private)
[ GitHub ]#send_async(method, *arguments)
Invoke a method on the connection asynchronously through the pool of thread workers.
# File 'actioncable/lib/action_cable/connection/base.rb', line 131
def send_async(method, *arguments) worker_pool.async_invoke(self, method, *arguments) end
#send_welcome_message (private)
[ GitHub ]# File 'actioncable/lib/action_cable/connection/base.rb', line 222
def # Send welcome message to the internal connection monitor channel. This ensures # the connection monitor state is reset after a successful websocket connection. transmit type: ActionCable::INTERNAL[: ][:welcome] end
#started_request_message (private)
[ GitHub ]#statistics
Return a basic hash of statistics for the connection keyed with identifier
, started_at
, #subscriptions, and request_id
. This can be returned by a health check against the connection.
# File 'actioncable/lib/action_cable/connection/base.rb', line 138
def statistics { identifier: connection_identifier, started_at: @started_at, subscriptions: subscriptions.identifiers, request_id: @env["action_dispatch.request_id"] } end