module Roqua module Support module Errors def self.extra_parameters @extra_parameters || {} end def self.extra_parameters=(hash) @extra_parameters = hash end def self.report(exception, context = {}) return if const_defined?(:Rails) and Rails.env.test? parameters, controller, skip_backtrace = merge_parameters(context) notification_urls = [] # Notify Airbrake notification_urls << notify_airbrake(exception, controller, parameters) # Notify AppSignal notification_urls << notify_appsignal(exception, parameters) # Notify Roqua logging log_exception(exception, parameters, notification_urls.compact, skip_backtrace) end private def self.merge_parameters(context) begin if context.is_a?(Hash) controller = context.delete :controller skip_backtrace = context.delete :skip_backtrace if extra_parameters.is_a?(Hash) parameters = extra_parameters.merge(context) else parameters = context end elsif extra_parameters.is_a?(Hash) parameters = extra_parameters end [parameters, controller, skip_backtrace] rescue Exception end end def self.log_exception(exception, parameters = {}, notification_urls = [], skip_backtrace = false) begin if Roqua.respond_to?(:logger) exception_info = {class_name: exception.class.to_s, message: exception.message, parameters: parameters} exception_info[:notification_urls] = notification_urls if notification_urls.present? exception_info[:backtrace] = exception.backtrace unless skip_backtrace puts exception_info.inspect Roqua.logger.error('roqua.exception', exception_info) end rescue Exception end end def self.notify_airbrake(exception, controller, parameters = {}) begin if const_defined?(:Airbrake) if controller && controller.respond_to?(:airbrake_request_data) request_data = controller.airbrake_request_data if request_data.is_a?(Hash) request_data[:parameters] ||= {} if request_data[:parameters].is_a?(Hash) request_data[:parameters] = parameters.merge request_data[:parameters] end else request_data = nil end end request_data ||= {parameters: parameters} uuid = Airbrake.notify_or_ignore(exception, request_data) "https://airbrake.io/locate/#{uuid}" if uuid end rescue Exception end end def self.notify_appsignal(exception, parameters = {}) begin if const_defined?(:Appsignal) and not Appsignal.is_ignored_exception?(exception) # TODO: If and when https://github.com/appsignal/appsignal/pull/9 is merged, # this functionality should be supported directly by Appsignal.send_exception. # Appsignal.send_exception(exception, parameters: parameters) if Appsignal.active? # Hackety hack around stateful mess of Appsignal gem if Appsignal::Transaction.current Appsignal::Transaction.current.set_tags(parameters) Appsignal::Transaction.current.add_exception(exception) else transaction = Appsignal::Transaction.create(SecureRandom.uuid, ENV.to_hash) transaction.set_tags(parameters) transaction.add_exception(exception) transaction.complete! Appsignal.agent.send_queue end end end rescue Exception end end end end end