require 'socket' require 'websocket/driver' require 'aasm' require 'openssl' require 'tamashii/common' require 'tamashii/agent/config' require 'tamashii/agent/component' require 'tamashii/agent/request_pool' require 'tamashii/agent/handler' module Tamashii module Agent class Connection < Component include AASM aasm do state :init, initial: true state :connecting state :auth_pending state :ready event :connect do transitions from: :init, to: :connecting, after: Proc.new { logger.info "Start connecting" } end event :auth_request do transitions from: :connecting, to: :auth_pending, after: Proc.new { logger.info "Sending authentication request" } end event :auth_success do transitions from: :auth_pending, to: :ready, after: Proc.new { logger.info "Authentication finished. Tag = #{@tag}" } end event :reset do transitions to: :init, after: Proc.new { logger.info "Connection state reset" } end end attr_reader :url attr_reader :master attr_reader :request_pool def initialize(master, host, port) super() @master = master @url = "#{Config.use_ssl ? "wss" : "ws"}://#{host}:#{port}/#{Config.entry_point}" self.reset @host = host @port = port @tag = 0 @request_pool = RequestPool.new @request_pool.set_handler(:request_timedout, method(:handle_request_timedout)) @request_pool.set_handler(:request_meet, method(:handle_request_meet)) @request_pool.set_handler(:send_request, method(:handle_send_request)) env_data = {connection: self} Resolver.config do [Type::REBOOT, Type::POWEROFF, Type::RESTART, Type::UPDATE].each do |type| handle type, Handler::System, env_data end handle Type::BUZZER_SOUND, Handler::Buzzer, env_data handle Type::RFID_RESPONSE_JSON, Handler::RequestPoolResponse, env_data end end def handle_request_timedout(req) @master.send_event(EVENT_CONNECTION_NOT_READY, "Connection not ready for #{req.ev_type}:#{req.ev_body}") end def handle_request_meet(req, res) logger.debug "Got packet: #{res.ev_type}: #{res.ev_body}" case res.ev_type when Type::RFID_RESPONSE_JSON json = JSON.parse(res.ev_body) handle_card_result(json) else logger.warn "Unhandled packet result: #{res.ev_type}: #{res.ev_body}" end end def handle_card_result(result) if result["auth"] @master.send_event(EVENT_BEEP, "ok") else @master.send_event(EVENT_BEEP, "no") end end def handle_send_request(req) if self.ready? @driver.binary(Packet.new(req.ev_type, @tag, req.wrap_body).dump) true else false end end # override def worker_loop loop do @request_pool.update ready = @selector.select(1) ready.each { |m| m.value.call } if ready if @io.nil? @io = try_create_socket if @io # socket io opened register_socket_io # start ws start_web_driver end end end end def send_auth_request # TODO: other types of auth @driver.binary(Packet.new(Type::AUTH_TOKEN, 0, [Type::CLIENT[:agent], @master.serial_number,Config.token].join(",")).dump) end def start_web_driver # TODO: Improve below code socket = Config.use_ssl ? OpenSSL::SSL::SSLSocket.new(self) : self @driver = WebSocket::Driver.client(socket) @driver.on :open, proc { |e| logger.info "Server opened" self.auth_request send_auth_request } @driver.on :close, proc { |e| logger.info "Server closed" close_socket_io self.reset } @driver.on :message, proc { |e| pkt = Packet.load(e.data) process_packet(pkt) if pkt } @driver.start self.connect end def register_socket_io _monitor = @selector.register(@io, :r) _monitor.value = proc do begin msg = @io.recv_nonblock(65535) if msg.empty? # socket closed logger.info "No message received from server. Connection reset" close_socket_io self.reset sleep 1 else @driver.parse(msg) end rescue => e logger.error "#{e.message}" logger.debug "Backtrace:" e.backtrace.each {|msg| logger.debug msg} end end end def try_create_socket logger.info "try to open socket..." TCPSocket.new(@host, @port) rescue nil end def close_socket_io logger.info "Socket IO Closed and Deregistered" @selector.deregister(@io) @io.close @io = nil end def write(string) @io.write(string) rescue logger.error "Write Error" close_socket_io self.reset end def process_packet(pkt) if self.auth_pending? if pkt.type == Type::AUTH_RESPONSE if pkt.body == Packet::STRING_TRUE @tag = pkt.tag self.auth_success else logger.error "Authentication failed. Delay for 3 seconds" sleep 3 end else logger.error "Authentication error: Not an authentication result packet" end else if pkt.tag == @tag || pkt.tag == 0 Resolver.resolve(pkt) else logger.debug "Tag mismatch packet: tag: #{pkt.tag}, type: #{pkt.type}" end end end def process_event(ev_type, ev_body) case ev_type when EVENT_CARD_DATA req = RequestPool::Request.new(Type::RFID_NUMBER , ev_body, ev_body) @request_pool.add_request(req) end end def clean_up super if @io @driver.close close_socket_io end rescue => e logger.warn "Error occured when clean up: #{e.to_s}" end end end end