lib/faye/websocket/client.rb



require 'forwardable'

module Faye
  class WebSocket

    class Client
      extend Forwardable
      include API

      DEFAULT_PORTS    = { 'http' => 80, 'https' => 443, 'ws' => 80, 'wss' => 443 }
      SECURE_PROTOCOLS = ['https', 'wss']

      def_delegators :@driver, :headers, :status

      def initialize(url, protocols = nil, options = {})
        @url = url
        super(options) { ::WebSocket::Driver.client(self, :max_length => options[:max_length], :protocols => protocols) }

        proxy       = options.fetch(:proxy, {})
        @endpoint   = URI.parse(proxy[:origin] || @url)
        port        = @endpoint.port || DEFAULT_PORTS[@endpoint.scheme]
        @origin_tls = options.fetch(:tls, {})
        @socket_tls = proxy[:origin] ? proxy.fetch(:tls, {}) : @origin_tls

        configure_proxy(proxy)

        EventMachine.connect(@endpoint.host, port, Connection) do |conn|
          conn.parent = self
        end
      rescue => error
        on_network_error(error)
      end

    private

      def configure_proxy(proxy)
        return unless proxy[:origin]

        @proxy = @driver.proxy(proxy[:origin])
        @proxy.on(:error) { |error| @driver.emit(:error, error) }

        if headers = proxy[:headers]
          headers.each { |name, value| @proxy.set_header(name, value) }
        end

        @proxy.on(:connect) do
          @proxy = nil
          start_tls(URI.parse(@url), @origin_tls)
          @driver.start
        end
      end

      def start_tls(uri, options)
        return unless SECURE_PROTOCOLS.include?(uri.scheme)

        tls_options = { :sni_hostname => uri.host, :verify_peer => true }.merge(options)
        @ssl_verifier = SslVerifier.new(uri.host, tls_options)
        @stream.start_tls(tls_options)
      end

      def on_connect(stream)
        @stream = stream
        start_tls(@endpoint, @socket_tls)

        worker = @proxy || @driver
        worker.start
      end

      def on_network_error(error)
        emit_error("Network error: #{ @url }: #{ error.message }")
        finalize_close
      end

      def ssl_verify_peer(cert)
        @ssl_verifier.ssl_verify_peer(cert)
      rescue => error
        on_network_error(error)
      end

      def ssl_handshake_completed
        @ssl_verifier.ssl_handshake_completed
      rescue => error
        on_network_error(error)
      end

      module Connection
        attr_accessor :parent

        def connection_completed
          parent.__send__(:on_connect, self)
        end

        def ssl_verify_peer(cert)
          parent.__send__(:ssl_verify_peer, cert)
        end

        def ssl_handshake_completed
          parent.__send__(:ssl_handshake_completed)
        end

        def receive_data(data)
          parent.__send__(:parse, data)
        end

        def unbind(error = nil)
          parent.__send__(:emit_error, error) if error
          parent.__send__(:finalize_close)
        end

        def write(data)
          send_data(data) rescue nil
        end
      end
    end

  end
end