From f9d0be72a8048a5e8ae54200c84a5dff2fe513fb Mon Sep 17 00:00:00 2001 From: Walt Jones Date: Fri, 30 Aug 2024 07:24:10 -0400 Subject: [PATCH] Enable Rails 7 error reporter via Rollbar config (#1161) * allow both hash syntaxes * enable Rails 7 error reporter via Rollbar config * initialize logger base class * use Logger version * apply capture_uncaught config to rails error subscriber --- .rubocop.yml | 4 +- lib/rollbar/configuration.rb | 16 ++++ lib/rollbar/exception_reporter.rb | 3 +- lib/rollbar/logger.rb | 24 +---- lib/rollbar/plugins/rails.rb | 8 ++ lib/rollbar/plugins/rails/error_subscriber.rb | 12 +++ spec/controllers/home_controller_spec.rb | 96 +++++++++++++++++++ .../app/controllers/home_controller.rb | 16 +++- spec/dummyapp/config/routes.rb | 2 + spec/rollbar/logger_spec.rb | 34 +------ 10 files changed, 157 insertions(+), 58 deletions(-) create mode 100644 lib/rollbar/plugins/rails/error_subscriber.rb diff --git a/.rubocop.yml b/.rubocop.yml index cd6ddbad..6dcfe055 100644 --- a/.rubocop.yml +++ b/.rubocop.yml @@ -65,9 +65,9 @@ Style/FrozenStringLiteralComment: Enabled: false Style/HashSyntax: - EnforcedStyle: hash_rockets + EnforcedStyle: no_mixed_keys SupportedStyles: - - hash_rockets + - no_mixed_keys Style/Lambda: Enabled: false diff --git a/lib/rollbar/configuration.rb b/lib/rollbar/configuration.rb index 202e8741..de2a881a 100644 --- a/lib/rollbar/configuration.rb +++ b/lib/rollbar/configuration.rb @@ -76,6 +76,7 @@ class Configuration :web_base, :write_to_file attr_reader :before_process, + :enable_rails_error_subscriber, :logger_level, :project_gem_paths, :send_extra_frame_data, @@ -104,6 +105,7 @@ def initialize @disable_core_monkey_patch = false @disable_rack_monkey_patch = false @enable_error_context = true + @enable_rails_error_subscriber = false @dj_threshold = 0 @dj_use_scoped_block = false @async_skip_report_handler = nil @@ -337,6 +339,20 @@ def send_extra_frame_data=(value) @send_extra_frame_data = value end + def enable_rails_error_subscriber=(enable) + return if !defined?(::Rails) || ::Rails.gem_version < ::Gem::Version.new('7.1.0') + + if @enable_rails_error_subscriber && !enable + ::Rails.error.unsubscribe(Rollbar::ErrorSubscriber) + end + + if !@enable_rails_error_subscriber && enable + ::Rails.error.subscribe(Rollbar::ErrorSubscriber.new) + end + + @enable_rails_error_subscriber = enable + end + # allow params to be read like a hash def [](option) send(option) diff --git a/lib/rollbar/exception_reporter.rb b/lib/rollbar/exception_reporter.rb index 1dc73317..18b334f3 100644 --- a/lib/rollbar/exception_reporter.rb +++ b/lib/rollbar/exception_reporter.rb @@ -27,7 +27,8 @@ def report_exception_to_rollbar(env, exception) end def capture_uncaught? - Rollbar.configuration.capture_uncaught != false + Rollbar.configuration.capture_uncaught != false && + !Rollbar.configuration.enable_rails_error_subscriber end def log_exception_message(exception) diff --git a/lib/rollbar/logger.rb b/lib/rollbar/logger.rb index 3a98a9a2..f3f9b0b1 100644 --- a/lib/rollbar/logger.rb +++ b/lib/rollbar/logger.rb @@ -17,12 +17,10 @@ module Rollbar class Logger < ::Logger class Error < RuntimeError; end - class DatetimeFormatNotSupported < Error; end - - class FormatterNotSupported < Error; end - def initialize - @level = ERROR + super(nil) + + self.level = ERROR end def add(severity, message = nil, progname = nil) @@ -39,22 +37,6 @@ def <<(message) error(message) end - def formatter=(_) - raise(FormatterNotSupported) - end - - def formatter - raise(FormatterNotSupported) - end - - def datetime_format=(_) - raise(DatetimeFormatNotSupported) - end - - def datetime_format - raise(DatetimeFormatNotSupported) - end - # Returns a Rollbar::Notifier instance with the current global scope and # with a logger writing to /dev/null so we don't have a infinite loop # when Rollbar.configuration.logger is Rails.logger. diff --git a/lib/rollbar/plugins/rails.rb b/lib/rollbar/plugins/rails.rb index 24e787b3..a6cc9e80 100644 --- a/lib/rollbar/plugins/rails.rb +++ b/lib/rollbar/plugins/rails.rb @@ -79,3 +79,11 @@ def secure_headers_middleware? Rollbar::Js::Frameworks::Rails.new.load(self) end end + +Rollbar.plugins.define('rails-error-subscriber') do + dependency { defined?(Rails::VERSION) && Rails::VERSION::MAJOR >= 7 } + + execute! do + require 'rollbar/plugins/rails/error_subscriber' + end +end diff --git a/lib/rollbar/plugins/rails/error_subscriber.rb b/lib/rollbar/plugins/rails/error_subscriber.rb new file mode 100644 index 00000000..dd5ba0ae --- /dev/null +++ b/lib/rollbar/plugins/rails/error_subscriber.rb @@ -0,0 +1,12 @@ +module Rollbar + class ErrorSubscriber + def report(error, handled:, severity:, context:, source: nil) + # The default `nil` for capture_uncaught means `true`. so check for false. + return unless handled || Rollbar.configuration.capture_uncaught != false + + extra = context.is_a?(Hash) ? context.deep_dup : {} + extra[:custom_data_method_context] = source + Rollbar.log(severity, error, extra) + end + end +end diff --git a/spec/controllers/home_controller_spec.rb b/spec/controllers/home_controller_spec.rb index e9640752..0e038a0b 100644 --- a/spec/controllers/home_controller_spec.rb +++ b/spec/controllers/home_controller_spec.rb @@ -297,6 +297,102 @@ def send_req(meth, path, args) end end + describe 'rails error subscriber', :type => 'request' do + let(:notifier) { Rollbar.notifier } + + before do + Rollbar.configure do |config| + config.enable_rails_error_subscriber = true + config.capture_uncaught = nil + end + end + + after do + Rollbar.configure do |config| + config.enable_rails_error_subscriber = false + config.capture_uncaught = nil + end + end + + context 'when Rails Error Subscriber is enabled', if: ::Rails.gem_version >= ::Gem::Version.new('7.1.0') do + it '`handle` should not raise an error and report a warning via rails error subscriber' do + logger_mock.should_receive(:info).with('[Rollbar] Success').never + + expect(Rollbar).to receive(:log) do |level, _, extra| + expect(extra[:custom_data_method_context]).to be_eql('application') + expect(level.to_s).to be_eql('warning') + end + + get '/handle_rails_error' + end + + it '`handle` should report a warning via rails error subscriber when capture_uncaught is false' do + logger_mock.should_receive(:info).with('[Rollbar] Success').never + + Rollbar.configure do |config| + config.capture_uncaught = false + end + + expect(Rollbar).to receive(:log) do |level, _, extra| + expect(extra[:custom_data_method_context]).to be_eql('application') + expect(level.to_s).to be_eql('warning') + end + + get '/handle_rails_error' + end + + it '`report` should raise an error and report an error via rails error subscriber' do + logger_mock.should_receive(:info).with('[Rollbar] Success').never + + expect(Rollbar).to receive(:log) do |level, _, extra| + expect(extra[:custom_data_method_context]).to be_eql('application') + expect(level.to_s).to be_eql('error') + end + + expect do + get '/record_rails_error' + end.to raise_exception(RuntimeError, 'Record Rails error') + end + + it 'uncaught exception should raise an error and report an error via rails error subscriber' do + logger_mock.should_receive(:info).with('[Rollbar] Success').never + + expect(Rollbar).to receive(:log) do |level, _, extra| + expect(extra[:custom_data_method_context]).to be_eql('application.action_dispatch') + expect(level.to_s).to be_eql('error') + end + + expect do + get '/cause_exception' + end.to raise_exception(NameError, 'Uncaught Rails error') + end + + it 'uncaught exception should not report an error when capture_uncaught is not set' do + logger_mock.should_receive(:info).with('[Rollbar] Success').never + + Rollbar.configure do |config| + config.capture_uncaught = false + end + + expect(Rollbar).to receive(:log).never + + expect do + get '/cause_exception' + end.to raise_exception(NameError, 'Uncaught Rails error') + end + end + + context 'when Rails Error Subscriber is enabled in unsupported Rails', if: ::Rails.gem_version < ::Gem::Version.new('7.1.0') do + it 'uncaught exception should raise an error and report via middleware' do + logger_mock.should_receive(:info).with('[Rollbar] Success').once + + expect do + get '/cause_exception' + end.to raise_exception(NameError, 'Uncaught Rails error') + end + end + end + describe 'configuration.locals', :type => 'request', :if => RUBY_VERSION >= '2.3.0' && !(defined?(RUBY_ENGINE) && diff --git a/spec/dummyapp/app/controllers/home_controller.rb b/spec/dummyapp/app/controllers/home_controller.rb index be38840e..5ced1b75 100644 --- a/spec/dummyapp/app/controllers/home_controller.rb +++ b/spec/dummyapp/app/controllers/home_controller.rb @@ -22,8 +22,22 @@ def deprecated_report_exception render :json => {} end + def handle_rails_error + Rails.error.handle do + raise 'Handle Rails error' + end + + render :json => {} + end + + def record_rails_error + Rails.error.record do + raise 'Record Rails error' + end + end + def cause_exception - _foo = bar + raise NameError, 'Uncaught Rails error' end def cause_exception_with_locals diff --git a/spec/dummyapp/config/routes.rb b/spec/dummyapp/config/routes.rb index 9cdb49c6..520942b9 100644 --- a/spec/dummyapp/config/routes.rb +++ b/spec/dummyapp/config/routes.rb @@ -4,6 +4,8 @@ member { post :start_session } end + match '/handle_rails_error' => 'home#handle_rails_error', :via => [:get, :post] + match '/record_rails_error' => 'home#record_rails_error', :via => [:get, :post] match '/cause_exception' => 'home#cause_exception', :via => [:get, :post] match '/cause_exception_with_locals' => 'home#cause_exception_with_locals', :via => [:get, :post] diff --git a/spec/rollbar/logger_spec.rb b/spec/rollbar/logger_spec.rb index 15b2319c..0bccf484 100644 --- a/spec/rollbar/logger_spec.rb +++ b/spec/rollbar/logger_spec.rb @@ -85,45 +85,13 @@ end end - describe '#formatter=' do - it 'fails with FormatterNotSupported' do - expect do - subject.formatter = double - end.to raise_error(Rollbar::Logger::FormatterNotSupported) - end - end - - describe '#formatter' do - it 'fails with FormatterNotSupported' do - expect do - subject.formatter - end.to raise_error(Rollbar::Logger::FormatterNotSupported) - end - end - - describe '#datetime_format=' do - it 'fails with DatetimeFormatNotSupported' do - expect do - subject.datetime_format = double - end.to raise_error(Rollbar::Logger::DatetimeFormatNotSupported) - end - end - - describe '#datetime_format' do - it 'fails with DatetimeFormatNotSupported' do - expect do - subject.datetime_format - end.to raise_error(Rollbar::Logger::DatetimeFormatNotSupported) - end - end - describe '#rollbar' do it 'returns a Rollbar notifier with a logger pointing to /dev/null' do notifier = subject.rollbar logger = notifier.configuration.logger logdev = logger.instance_eval { @logdev } - if Gem::Version.new(RUBY_VERSION) >= Gem::Version.new('3.0') + if Gem::Version.new(Logger::VERSION) >= Gem::Version.new('1.4.3') # The Logger class no longer creates a LogDevice when the device is `File::NULL` # See: ruby/ruby@f3e12caa088cc893a54bc2810ff511e4c89b322b expect(logdev).to be_nil