smartinez87/exception_notification

View on GitHub
lib/exception_notification/rack.rb

Summary

Maintainability
B
4 hrs
Test Coverage
# frozen_string_literal: true

module ExceptionNotification
  class Rack
    class CascadePassException < RuntimeError; end

    def initialize(app, options = {})
      @app = app

      ExceptionNotifier.tap do |en|
        en.ignored_exceptions = options.delete(:ignore_exceptions) if options.key?(:ignore_exceptions)
        en.error_grouping = options.delete(:error_grouping) if options.key?(:error_grouping)
        en.error_grouping_period = options.delete(:error_grouping_period) if options.key?(:error_grouping_period)
        en.notification_trigger = options.delete(:notification_trigger) if options.key?(:notification_trigger)

        if options.key?(:error_grouping_cache)
          en.error_grouping_cache = options.delete(:error_grouping_cache)
        elsif defined?(Rails) && Rails.respond_to?(:cache)
          en.error_grouping_cache = Rails.cache
        end
      end

      if options.key?(:ignore_if)
        rack_ignore = options.delete(:ignore_if)
        ExceptionNotifier.ignore_if do |exception, opts|
          opts.key?(:env) && rack_ignore.call(opts[:env], exception)
        end
      end

      if options.key?(:ignore_notifier_if)
        rack_ignore_by_notifier = options.delete(:ignore_notifier_if)
        rack_ignore_by_notifier.each do |notifier, proc|
          ExceptionNotifier.ignore_notifier_if(notifier) do |exception, opts|
            opts.key?(:env) && proc.call(opts[:env], exception)
          end
        end
      end

      ExceptionNotifier.ignore_crawlers(options.delete(:ignore_crawlers)) if options.key?(:ignore_crawlers)

      @ignore_cascade_pass = options.delete(:ignore_cascade_pass) { true }

      options.each do |notifier_name, opts|
        ExceptionNotifier.register_exception_notifier(notifier_name, opts)
      end
    end

    def call(env)
      _, headers, = response = @app.call(env)

      if !@ignore_cascade_pass && headers['X-Cascade'] == 'pass'
        msg = "This exception means that the preceding Rack middleware set the 'X-Cascade' header to 'pass' -- in " \
              'Rails, this often means that the route was not found (404 error).'
        raise CascadePassException, msg
      end

      response
    rescue Exception => e
      env['exception_notifier.delivered'] = true if ExceptionNotifier.notify_exception(e, env: env)

      raise e unless e.is_a?(CascadePassException)

      response
    end
  end
end