class ActionCable::Connection::ClientSocket
:nodoc:
Copyright © 2010-2015 James Coglan
This class is heavily based on faye-websocket-ruby
–
def self.determine_url(env)
Copyright (c) 2010-2015 James Coglan
This class is heavily based on faye-websocket-ruby
--
def self.determine_url(env) scheme = secure_request?(env) ? "wss:" : "ws:" "#{ scheme }//#{ env['HTTP_HOST'] }#{ env['REQUEST_URI'] }" end
def self.secure_request?(env)
def self.secure_request?(env) return true if env["HTTPS"] == "on" return true if env["HTTP_X_FORWARDED_SSL"] == "on" return true if env["HTTP_X_FORWARDED_SCHEME"] == "https" return true if env["HTTP_X_FORWARDED_PROTO"] == "https" return true if env["rack.url_scheme"] == "https" false end
def alive?
def alive? @ready_state == OPEN end
def begin_close(reason, code)
def begin_close(reason, code) return if @ready_state == CLOSED @ready_state = CLOSING @close_params = [reason, code] @stream.shutdown if @stream finalize_close end
def client_gone
def client_gone finalize_close end
def close(code = nil, reason = nil)
def close(code = nil, reason = nil) code ||= 1000 reason ||= "" unless code == 1000 || (code >= 3000 && code <= 4999) raise ArgumentError, "Failed to execute 'close' on WebSocket: " \ "The code must be either 1000, or between 3000 and 4999. " \ "#{code} is neither." end @ready_state = CLOSING unless @ready_state == CLOSED @driver.close(reason, code) end
def emit_error(message)
def emit_error(message) return if @ready_state >= CLOSING @event_target.on_error(message) end
def finalize_close
def finalize_close return if @ready_state == CLOSED @ready_state = CLOSED @event_target.on_close(*@close_params) end
def initialize(env, event_target, event_loop, protocols)
def initialize(env, event_target, event_loop, protocols) @env = env @event_target = event_target @event_loop = event_loop @url = ClientSocket.determine_url(@env) @driver = @driver_started = nil @close_params = ["", 1006] @ready_state = CONNECTING # The driver calls +env+, +url+, and +write+ @driver = ::WebSocket::Driver.rack(self, protocols: protocols) @driver.on(:open) { |e| open } @driver.on(:message) { |e| receive_message(e.data) } @driver.on(:close) { |e| begin_close(e.reason, e.code) } @driver.on(:error) { |e| emit_error(e.message) } @stream = ActionCable::Connection::Stream.new(@event_loop, self) end
def open
def open return unless @ready_state == CONNECTING @ready_state = OPEN @event_target.on_open end
def parse(data)
def parse(data) @driver.parse(data) end
def protocol
def protocol @driver.protocol end
def rack_response
def rack_response start_driver [ -1, {}, [] ] end
def receive_message(data)
def receive_message(data) return unless @ready_state == OPEN @event_target.on_message(data) end
def start_driver
def start_driver return if @driver.nil? || @driver_started @stream.hijack_rack_socket if callback = @env["async.callback"] callback.call([101, {}, @stream]) end @driver_started = true @driver.start end
def transmit(message)
def transmit(message) return false if @ready_state > OPEN case message when Numeric then @driver.text(message.to_s) when String then @driver.text(message) when Array then @driver.binary(message) else false end end
def write(data)
def write(data) @stream.write(data) rescue => e emit_error e.message end