module HydraulicBrake # Sends out the notice to Airbrake class Sender NOTICES_URI = '/notifier_api/v2/notices/'.freeze HTTP_ERRORS = [Timeout::Error, Errno::EINVAL, Errno::ECONNRESET, EOFError, Net::HTTPBadResponse, Net::HTTPHeaderSyntaxError, Net::ProtocolError, Errno::ECONNREFUSED].freeze def initialize(options = {}) [ :proxy_host, :proxy_port, :proxy_user, :proxy_pass, :protocol, :host, :port, :secure, :use_system_ssl_cert_chain, :http_open_timeout, :http_read_timeout ].each do |option| instance_variable_set("@#{option}", options[option]) end end # Sends the notice data off to Airbrake for processing. # # @param [Notice] notice The notice to be sent off def send_to_airbrake(notice) data = notice.to_xml http = setup_http_connection response = begin http.post(url.path, data, HEADERS) rescue *HTTP_ERRORS => e log :level => :error, :message => "Unable to contact the Airbrake server. HTTP Error=#{e}" nil end case response when Net::HTTPSuccess then log :level => :info, :message => success_message_from_response(response) error_id_from_response(response) else log :level => :error, :message => "Failure: #{response.class}", :response => response, :notice => notice end rescue => e log :level => :error, :message => "[HydraulicBrake::Sender#send_to_airbrake] Cannot send notification. Error: #{e.class}" + " - #{e.message}\nBacktrace:\n#{e.backtrace.join("\n\t")}" nil end attr_reader :proxy_host, :proxy_port, :proxy_user, :proxy_pass, :protocol, :host, :port, :secure, :use_system_ssl_cert_chain, :http_open_timeout, :http_read_timeout alias_method :secure?, :secure alias_method :use_system_ssl_cert_chain?, :use_system_ssl_cert_chain private def success_message_from_response(response) error_url = error_url_from_response(response) if error_url "Success: sent error to Airbrake: #{error_url}" else "Success: sent error to Airbrake" end end def error_id_from_response(response) if response && response.respond_to?(:body) error_id = response.body.match(%r{]*>(.*?)}) return error_id[1] if error_id end nil rescue nil end def error_url_from_response(response) if response && response.respond_to?(:body) error_url = response.body.match(%r{]*>(.*?)}) return error_url[1] if error_url end nil rescue nil end def url URI.parse("#{protocol}://#{host}:#{port}").merge(NOTICES_URI) end def log(opts = {}) logger.send opts[:level], LOG_PREFIX + opts[:message] HydraulicBrake.report_environment_info HydraulicBrake.report_response_body(opts[:response].body) if opts[:response] && opts[:response].respond_to?(:body) HydraulicBrake.report_notice(opts[:notice]) if opts[:notice] end def logger HydraulicBrake.logger end def setup_http_connection http = Net::HTTP::Proxy(proxy_host, proxy_port, proxy_user, proxy_pass). new(url.host, url.port) http.read_timeout = http_read_timeout http.open_timeout = http_open_timeout if secure? http.use_ssl = true http.ca_file = HydraulicBrake.configuration.ca_bundle_path http.verify_mode = OpenSSL::SSL::VERIFY_PEER else http.use_ssl = false end http rescue => e log :level => :error, :message => "[HydraulicBrake::Sender#setup_http_connection] Failure initializing the HTTP connection.\n" + "Error: #{e.class} - #{e.message}\nBacktrace:\n#{e.backtrace.join("\n\t")}" raise e end end end