From 71d6be8964bb2a432256628e0c80dca0f235788b Mon Sep 17 00:00:00 2001 From: Dominik Mlynek Date: Mon, 25 Nov 2024 11:00:54 +0000 Subject: [PATCH 1/4] Change exceptions logging: Add logging of exceptions' causes --- lib/sidekiq/exception_utils.rb | 27 +++++++++++++++++++++++++++ lib/sidekiq/logging/shared.rb | 7 +++---- spec/sidekiq/logstash_spec.rb | 25 +++++++++++++++++++------ spec/workers/spec_worker.rb | 12 +++++++++++- 4 files changed, 60 insertions(+), 11 deletions(-) create mode 100644 lib/sidekiq/exception_utils.rb diff --git a/lib/sidekiq/exception_utils.rb b/lib/sidekiq/exception_utils.rb new file mode 100644 index 0000000..5259303 --- /dev/null +++ b/lib/sidekiq/exception_utils.rb @@ -0,0 +1,27 @@ +# frozen_string_literal: true + +# Utility that allows us to get a hash representation of an exception +module ExceptionUtils + def self.exception_to_hash(exc, parent_backtrace = nil) + backtrace = exc.backtrace || [] + if parent_backtrace + common_lines = backtrace.reverse.zip(parent_backtrace.reverse).take_while { |a, b| a == b } + + backtrace = backtrace[0...-common_lines.length] if common_lines.any? + end + + error_hash = { + 'class' => exc.class.to_s, + 'message' => exc.message, + 'backtrace' => backtrace + } + + cause = exc.cause + if cause + # Pass the current backtrace as the parent_backtrace to the cause to shorten cause's backtrace list + error_hash['cause'] = exception_to_hash(cause, exc.backtrace) + end + + error_hash + end +end diff --git a/lib/sidekiq/logging/shared.rb b/lib/sidekiq/logging/shared.rb index 67df5e5..2a767c5 100644 --- a/lib/sidekiq/logging/shared.rb +++ b/lib/sidekiq/logging/shared.rb @@ -1,5 +1,7 @@ # frozen_string_literal: true +require 'sidekiq/exception_utils' + module Sidekiq module Logging # Shared module with all the logics used by job loggers. @@ -50,10 +52,7 @@ def log_job_exception(job, started_at, exc) payload['message'] += ": fail: #{payload['duration']} sec" payload['job_status'] = 'fail' - exc = exc.cause || exc if exc.is_a? Sidekiq::JobRetry::Handled - payload['error_message'] = exc.message - payload['error'] = exc.class - payload['error_backtrace'] = %('#{exc.backtrace.join("\n")}') + payload['error'] = ExceptionUtils.exception_to_hash(exc) process_payload(payload) end diff --git a/spec/sidekiq/logstash_spec.rb b/spec/sidekiq/logstash_spec.rb index 6310298..848c29b 100644 --- a/spec/sidekiq/logstash_spec.rb +++ b/spec/sidekiq/logstash_spec.rb @@ -120,9 +120,17 @@ def process(worker, params = [], encrypt: false) it 'logs the exception with job retry' do expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) - expect(log_message['error_message']).to eq('You know nothing, Jon Snow.') - expect(log_message['error']).to eq('RuntimeError') - expect(log_message['error_backtrace'].split("\n").first).to include('workers/spec_worker.rb:7') + expect(log_message['error']['class']).to eq('Sidekiq::JobRetry::Handled') + expect(log_message['error']['message']).to eq('Sidekiq::JobRetry::Handled') + + expect(log_message['error']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['message']).to eq('You know nothing, Jon Snow.') + + expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['message']).to eq('Error rescuing error') + + expect(log_message['error']['cause']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['cause']['message']).to eq('Deepest error') end it 'logs the exception without job retry' do @@ -130,9 +138,14 @@ def process(worker, params = [], encrypt: false) expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) - expect(log_message['error_message']).to eq('You know nothing, Jon Snow.') - expect(log_message['error']).to eq('RuntimeError') - expect(log_message['error_backtrace'].split("\n").first).to include('workers/spec_worker.rb:7') + expect(log_message['error']['class']).to eq('RuntimeError') + expect(log_message['error']['message']).to eq('You know nothing, Jon Snow.') + + expect(log_message['error']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['message']).to eq('Error rescuing error') + + expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['message']).to eq('Deepest error') end end diff --git a/spec/workers/spec_worker.rb b/spec/workers/spec_worker.rb index 39e6547..d8dd3e7 100644 --- a/spec/workers/spec_worker.rb +++ b/spec/workers/spec_worker.rb @@ -4,6 +4,16 @@ class SpecWorker include Sidekiq::Worker def perform(fail = false, _params = {}) - raise 'You know nothing, Jon Snow.' if fail + raise_error if fail + rescue StandardError + raise 'You know nothing, Jon Snow.' + end + + private + + def raise_error + raise 'Deepest error' + rescue StandardError + raise 'Error rescuing error' end end From 235db634c3b466753dd09a9dca99f8c408cc5ae2 Mon Sep 17 00:00:00 2001 From: Dominik Mlynek Date: Thu, 28 Nov 2024 11:15:53 +0000 Subject: [PATCH 2/4] Change exceptions logging: Add `log_job_exception_with_causes` and `causes_logging_max_depth` configs so the new incompatible error logging format has to be turn on manually --- lib/sidekiq/exception_utils.rb | 7 +-- lib/sidekiq/logging/shared.rb | 10 +++- lib/sidekiq/logstash/configuration.rb | 9 ++- spec/sidekiq/logstash_spec.rb | 80 +++++++++++++++++++++------ 4 files changed, 83 insertions(+), 23 deletions(-) diff --git a/lib/sidekiq/exception_utils.rb b/lib/sidekiq/exception_utils.rb index 5259303..f7e1765 100644 --- a/lib/sidekiq/exception_utils.rb +++ b/lib/sidekiq/exception_utils.rb @@ -2,7 +2,7 @@ # Utility that allows us to get a hash representation of an exception module ExceptionUtils - def self.exception_to_hash(exc, parent_backtrace = nil) + def self.get_exception_with_cause_hash(exc, parent_backtrace = nil, max_depth_left) backtrace = exc.backtrace || [] if parent_backtrace common_lines = backtrace.reverse.zip(parent_backtrace.reverse).take_while { |a, b| a == b } @@ -16,10 +16,9 @@ def self.exception_to_hash(exc, parent_backtrace = nil) 'backtrace' => backtrace } - cause = exc.cause - if cause + if (cause = exc.cause) && max_depth_left.positive? # Pass the current backtrace as the parent_backtrace to the cause to shorten cause's backtrace list - error_hash['cause'] = exception_to_hash(cause, exc.backtrace) + error_hash['cause'] = get_exception_with_cause_hash(cause, exc.backtrace, max_depth_left - 1) end error_hash diff --git a/lib/sidekiq/logging/shared.rb b/lib/sidekiq/logging/shared.rb index 2a767c5..4001fd9 100644 --- a/lib/sidekiq/logging/shared.rb +++ b/lib/sidekiq/logging/shared.rb @@ -52,7 +52,15 @@ def log_job_exception(job, started_at, exc) payload['message'] += ": fail: #{payload['duration']} sec" payload['job_status'] = 'fail' - payload['error'] = ExceptionUtils.exception_to_hash(exc) + config = Sidekiq::Logstash.configuration + if config.log_job_exception_with_causes + payload['error'] = ExceptionUtils.get_exception_with_cause_hash(exc, config.causes_logging_max_depth) + else + exc = exc.cause || exc if exc.is_a? Sidekiq::JobRetry::Handled + payload['error_message'] = exc.message + payload['error'] = exc.class + payload['error_backtrace'] = %('#{exc.backtrace.join("\n")}') + end process_payload(payload) end diff --git a/lib/sidekiq/logstash/configuration.rb b/lib/sidekiq/logstash/configuration.rb index d03677d..c2f1d75 100644 --- a/lib/sidekiq/logstash/configuration.rb +++ b/lib/sidekiq/logstash/configuration.rb @@ -4,11 +4,18 @@ module Sidekiq module Logstash # Class that allows to configure the gem behaviour. class Configuration - attr_accessor :custom_options, :filter_args, :job_start_log, :keep_default_error_handler + attr_accessor :custom_options, + :filter_args, + :job_start_log, + :keep_default_error_handler, + :log_job_exception_with_causes, + :causes_logging_max_depth def initialize @filter_args = [] @job_start_log = false + @log_job_exception_with_causes = false + @causes_logging_max_depth = 2 end # Added to ensure custom_options is a Proc diff --git a/spec/sidekiq/logstash_spec.rb b/spec/sidekiq/logstash_spec.rb index 848c29b..c6bfd1b 100644 --- a/spec/sidekiq/logstash_spec.rb +++ b/spec/sidekiq/logstash_spec.rb @@ -120,17 +120,9 @@ def process(worker, params = [], encrypt: false) it 'logs the exception with job retry' do expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) - expect(log_message['error']['class']).to eq('Sidekiq::JobRetry::Handled') - expect(log_message['error']['message']).to eq('Sidekiq::JobRetry::Handled') - - expect(log_message['error']['cause']['class']).to eq('RuntimeError') - expect(log_message['error']['cause']['message']).to eq('You know nothing, Jon Snow.') - - expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') - expect(log_message['error']['cause']['cause']['message']).to eq('Error rescuing error') - - expect(log_message['error']['cause']['cause']['cause']['class']).to eq('RuntimeError') - expect(log_message['error']['cause']['cause']['cause']['message']).to eq('Deepest error') + expect(log_message['error_message']).to eq('You know nothing, Jon Snow.') + expect(log_message['error']).to eq('RuntimeError') + expect(log_message['error_backtrace'].split("\n").first).to include('workers/spec_worker.rb:') end it 'logs the exception without job retry' do @@ -138,14 +130,68 @@ def process(worker, params = [], encrypt: false) expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) - expect(log_message['error']['class']).to eq('RuntimeError') - expect(log_message['error']['message']).to eq('You know nothing, Jon Snow.') + expect(log_message['error_message']).to eq('You know nothing, Jon Snow.') + expect(log_message['error']).to eq('RuntimeError') + expect(log_message['error_backtrace'].split("\n").first).to include('workers/spec_worker.rb:') + end + + context 'log_job_exception_with_causes enabled' do + before do + Sidekiq::Logstash.configure do |config| + config.log_job_exception_with_causes = true + end + end + + after do + Sidekiq::Logstash.configure do |config| + config.log_job_exception_with_causes = false + end + end + + it 'logs the exception with its two causes' do + expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) + + expect(log_message['error']['class']).to eq('Sidekiq::JobRetry::Handled') + expect(log_message['error']['message']).to eq('Sidekiq::JobRetry::Handled') - expect(log_message['error']['cause']['class']).to eq('RuntimeError') - expect(log_message['error']['cause']['message']).to eq('Error rescuing error') + expect(log_message['error']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['message']).to eq('You know nothing, Jon Snow.') - expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') - expect(log_message['error']['cause']['cause']['message']).to eq('Deepest error') + expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['message']).to eq('Error rescuing error') + end + + context 'causes_logging_max_depth is set to 3' do + before do + Sidekiq::Logstash.configure do |config| + config.log_job_exception_with_causes = true + config.causes_logging_max_depth = 3 + end + end + + after do + Sidekiq::Logstash.configure do |config| + config.log_job_exception_with_causes = true + config.causes_logging_max_depth = 2 + end + end + + it 'logs the exception with its three causes' do + expect { process(SpecWorker, [true]) }.to raise_error(RuntimeError) + + expect(log_message['error']['class']).to eq('Sidekiq::JobRetry::Handled') + expect(log_message['error']['message']).to eq('Sidekiq::JobRetry::Handled') + + expect(log_message['error']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['message']).to eq('You know nothing, Jon Snow.') + + expect(log_message['error']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['message']).to eq('Error rescuing error') + + expect(log_message['error']['cause']['cause']['cause']['class']).to eq('RuntimeError') + expect(log_message['error']['cause']['cause']['cause']['message']).to eq('Deepest error') + end + end end end From 42fddfdaa6db927143e8209f17ddd27c2badad8a Mon Sep 17 00:00:00 2001 From: Dominik Mlynek Date: Thu, 28 Nov 2024 11:31:06 +0000 Subject: [PATCH 3/4] Change exceptions logging: README.md entry for `log_job_exception_with_causes` `causes_logging_max_depth` Settings --- README.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/README.md b/README.md index 94a4a74..d8b839f 100644 --- a/README.md +++ b/README.md @@ -83,6 +83,14 @@ Sidekiq::Logstash.configure do |config| # by default, sidekiq-logstash removes the default error handler # to keep it, simply set this to true config.keep_default_error_handler = true + + # To enable logging of Exception causes, + # which provides details about the exception chain, + # uncomment the following lines: + # + # config.log_job_exception_with_causes = true + # # To specify the maximum depth of causes to log: + # config.causes_logging_max_depth = 3 # Default is 2 end ``` From 5dcb3970d1366e0f0042e23df23a86a329678bee Mon Sep 17 00:00:00 2001 From: Dominik Mlynek Date: Thu, 28 Nov 2024 11:38:54 +0000 Subject: [PATCH 4/4] Change exceptions logging: refine README.md entry --- README.md | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/README.md b/README.md index d8b839f..e230495 100644 --- a/README.md +++ b/README.md @@ -84,8 +84,7 @@ Sidekiq::Logstash.configure do |config| # to keep it, simply set this to true config.keep_default_error_handler = true - # To enable logging of Exception causes, - # which provides details about the exception chain, + # To enable a log structure that includes details about exception causes, # uncomment the following lines: # # config.log_job_exception_with_causes = true