require 'date' module Angelo class Responder include Celluloid::Logger class << self attr_writer :default_headers def content_type type dhs = self.default_headers case type when :json self.default_headers = dhs.merge CONTENT_TYPE_HEADER_KEY => JSON_TYPE when :html self.default_headers = dhs.merge CONTENT_TYPE_HEADER_KEY => HTML_TYPE else raise ArgumentError.new "invalid content_type: #{type}" end end def default_headers @default_headers ||= DEFAULT_RESPONSE_HEADERS @default_headers end def symhash Hash.new {|hash,key| hash[key.to_s] if Symbol === key } end end attr_writer :connection attr_reader :request def initialize &block @response_handler = Base.compile! :request_handler, &block end def base= base @base = base @base.responder = self end def request= request @params = nil @redirect = nil @request = request handle_request respond end def handle_request if @response_handler @base.before if @base.respond_to? :before @body = @response_handler.bind(@base).call || '' @base.after if @base.respond_to? :after else raise NotImplementedError end rescue JSON::ParserError => jpe handle_error jpe, :bad_request, false rescue FormEncodingError => fee handle_error fee, :bad_request, false rescue => e handle_error e end def handle_error _error, type = :internal_server_error, report = true err_msg = error_message _error Angelo.log @connection, @request, nil, type, err_msg.size @connection.respond type, headers, err_msg @connection.close if report error "#{_error.class} - #{_error.message}" ::STDERR.puts _error.backtrace end end def error_message _error case when respond_with?(:json) { error: _error.message }.to_json else _error.message end end def headers hs = nil @headers ||= self.class.default_headers.dup @headers.merge! hs if hs @headers end def content_type type case type when :json headers CONTENT_TYPE_HEADER_KEY => JSON_TYPE when :html headers CONTENT_TYPE_HEADER_KEY => HTML_TYPE else raise ArgumentError.new "invalid content_type: #{type}" end end def respond_with? type case headers[CONTENT_TYPE_HEADER_KEY] when JSON_TYPE type == :json else type == :html end end def respond @body = case @body when String JSON.parse @body if respond_with? :json # for the raises @body when Hash raise 'html response requires String' if respond_with? :html @body.to_json if respond_with? :json when NilClass EMPTY_STRING end status = @redirect.nil? ? :ok : :moved_permanently headers LOCATION_HEADER_KEY => @redirect if @redirect Angelo.log @connection, @request, nil, status, @body.size @connection.respond status, headers, @body rescue => e handle_error e, :internal_server_error, false end def redirect url @redirect = url end end end