require 'websocket-eventmachine-base' require 'uri' module WebSocket module EventMachine # WebSocket Client (using EventMachine) # @example # ws = WebSocket::EventMachine::Client.connect(:host => "0.0.0.0", :port => 8080) # ws.onmessage { |msg| ws.send "Pong: #{msg}" } # ws.send "data" class Client < Base ########### ### API ### ########### # Connect to websocket server # @param args [Hash] The request arguments # @option args [String] :host The host IP/DNS name # @option args [Integer] :port The port to connect too(default = 80) # @option args [Integer] :version Version of protocol to use(default = 13) def self.connect(args = {}) host = nil port = nil if args[:uri] uri = URI.parse(args[:uri]) host = uri.host port = uri.port end host = args[:host] if args[:host] port = args[:port] if args[:port] port ||= 80 ::EventMachine.connect host, port, self, args end # Initialize connection # @param args [Hash] Arguments for connection # @option args [String] :host The host IP/DNS name # @option args [Integer] :port The port to connect too(default = 80) # @option args [Integer] :version Version of protocol to use(default = 13) def initialize(args) @args = args end ############################ ### EventMachine methods ### ############################ # Called after initialize of connection, but before connecting to server # Eventmachine internal # @private def post_init @state = :connecting @handshake = ::WebSocket::Handshake::Client.new(@args) end # Called by EventMachine after connecting. # Sends handshake to server # Eventmachine internal # @private def connection_completed send(@handshake.to_s, :type => :plain) end private def incoming_frame ::WebSocket::Frame::Incoming::Client end def outgoing_frame ::WebSocket::Frame::Outgoing::Client end public ######################### ### Inherited methods ### ######################### # Called when connection is opened. # No parameters are passed to block def onopen(&blk); super; end # Called when connection is closed. # No parameters are passed to block def onclose(&blk); super; end # Called when error occurs. # One parameter passed to block: # error - string with error message def onerror(&blk); super; end # Called when message is received. # Two parameters passed to block: # message - string with received message # type - type of message. Valid values are :text and :binary def onmessage(&blk); super; end # Called when ping message is received # One parameter passed to block: # message - string with ping message def onping(&blk); super; end # Called when pond message is received # One parameter passed to block: # message - string with pong message def onpong(&blk); super; end # Send data # @param data [String] Data to send # @param args [Hash] Arguments for send # @option args [String] :type Type of frame to send - available types are "text", "binary", "ping", "pong" and "close" # @option args [Integer] :code Code for close frame # @return [Boolean] true if data was send, otherwise call on_error if needed def send(data, args = {}); super; end # Close connection # @return [Boolean] true if connection is closed immediately, false if waiting for other side to close connection def close(code = 1000, data = nil); super; end # Send ping message # @return [Boolean] false if protocol version is not supporting ping requests def ping(data = ''); super; end # Send pong message # @return [Boolean] false if protocol version is not supporting pong requests def pong(data = ''); super; end end end end