# API references: # # * https://html.spec.whatwg.org/multipage/comms.html#network # * https://dom.spec.whatwg.org/#interface-eventtarget # * https://dom.spec.whatwg.org/#interface-event require 'forwardable' require 'stringio' require 'uri' require 'eventmachine' require 'websocket/driver' module Faye autoload :EventSource, File.expand_path('../eventsource', __FILE__) autoload :RackStream, File.expand_path('../rack_stream', __FILE__) class WebSocket root = File.expand_path('../websocket', __FILE__) autoload :Adapter, root + '/adapter' autoload :API, root + '/api' autoload :Client, root + '/client' autoload :SslVerifier, root + '/ssl_verifier' ADAPTERS = { 'goliath' => :Goliath, 'rainbows' => :Rainbows, 'thin' => :Thin } def self.determine_url(env, schemes = ['wss', 'ws']) scheme = schemes[secure_request?(env) ? 0 : 1] host = env['HTTP_HOST'] path = env['PATH_INFO'] query = env['QUERY_STRING'].to_s scheme + '://' + host + path + (query.empty? ? '' : '?' + query) end def self.ensure_reactor_running Thread.new { EventMachine.run } unless EventMachine.reactor_running? Thread.pass until EventMachine.reactor_running? end def self.load_adapter(backend) const = Kernel.const_get(ADAPTERS[backend]) rescue nil require(backend) unless const path = File.expand_path("../adapters/#{ backend }.rb", __FILE__) require(path) if File.file?(path) end 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' return false end def self.websocket?(env) ::WebSocket::Driver.websocket?(env) end attr_reader :env include API def initialize(env, protocols = nil, options = {}) WebSocket.ensure_reactor_running @env = env @url = WebSocket.determine_url(@env) super(options) { ::WebSocket::Driver.rack(self, :max_length => options[:max_length], :protocols => protocols) } @driver_started = false @stream = Stream.new(self) if callback = @env['async.callback'] callback.call([101, {}, @stream]) end end def start_driver return if @driver.nil? || @driver_started @driver_started = true EventMachine.schedule { @driver.start } end def rack_response start_driver [ -1, {}, [] ] end class Stream < RackStream def fail @socket_object.__send__(:finalize_close) end def receive(data) @socket_object.__send__(:parse, data) end end end end