Skip to content

Commit

Permalink
Allow for custom handling of exceptions that are discarded
Browse files Browse the repository at this point in the history
  • Loading branch information
Aidan Haran authored and Aidan Haran committed Sep 16, 2017
1 parent 34956f7 commit 3291fa3
Show file tree
Hide file tree
Showing 4 changed files with 36 additions and 1 deletion.
16 changes: 16 additions & 0 deletions activejob/CHANGELOG.md
@@ -1,3 +1,19 @@
* Allow block to be passed to ActiveJob::Base.discard_on to allow custom handling of discard jobs.

Example:

class RemoteServiceJob < ActiveJob::Base
discard_on(CustomAppException) do |job, exception|
ExceptionNotifier.caught(exception)
end

def perform(*args)
# Might raise CustomAppException for something domain specific
end
end

*Aidan Haran*

* Change logging instrumentation to log errors when a job raises an exception.

Fixes #26848.
Expand Down
12 changes: 11 additions & 1 deletion activejob/lib/active_job/exceptions.rb
Expand Up @@ -61,18 +61,28 @@ def retry_on(exception, wait: 3.seconds, attempts: 5, queue: nil, priority: nil)
# Discard the job with no attempts to retry, if the exception is raised. This is useful when the subject of the job,
# like an Active Record, is no longer available, and the job is thus no longer relevant.
#
# You can also pass a block that'll be invoked. This block is yielded with the job instance as the first and the error instance as the second parameter.
#
# ==== Example
#
# class SearchIndexingJob < ActiveJob::Base
# discard_on ActiveJob::DeserializationError
# discard_on(CustomAppException) do |job, exception|
# ExceptionNotifier.caught(exception)
# end
#
# def perform(record)
# # Will raise ActiveJob::DeserializationError if the record can't be deserialized
# # Might raise CustomAppException for something domain specific
# end
# end
def discard_on(exception)
rescue_from exception do |error|
logger.error "Discarded #{self.class} due to a #{exception}. The original exception was #{error.cause.inspect}."
if block_given?
yield self, exception
else
logger.error "Discarded #{self.class} due to a #{exception}. The original exception was #{error.cause.inspect}."
end
end
end
end
Expand Down
7 changes: 7 additions & 0 deletions activejob/test/cases/exceptions_test.rb
Expand Up @@ -58,6 +58,13 @@ class ExceptionsTest < ActiveJob::TestCase
end
end

test "custom handling of discarded job" do
perform_enqueued_jobs do
RetryJob.perform_later "CustomDiscardableError", 2
assert_equal "Dealt with a job that was discarded in a custom way", JobBuffer.last_value
end
end

test "custom handling of job that exceeds retry attempts" do
perform_enqueued_jobs do
RetryJob.perform_later "CustomCatchError", 6
Expand Down
2 changes: 2 additions & 0 deletions activejob/test/jobs/retry_job.rb
Expand Up @@ -10,6 +10,7 @@ class ExponentialWaitTenAttemptsError < StandardError; end
class CustomWaitTenAttemptsError < StandardError; end
class CustomCatchError < StandardError; end
class DiscardableError < StandardError; end
class CustomDiscardableError < StandardError; end

class RetryJob < ActiveJob::Base
retry_on DefaultsError
Expand All @@ -19,6 +20,7 @@ class RetryJob < ActiveJob::Base
retry_on CustomWaitTenAttemptsError, wait: ->(executions) { executions * 2 }, attempts: 10
retry_on(CustomCatchError) { |job, exception| JobBuffer.add("Dealt with a job that failed to retry in a custom way after #{job.arguments.second} attempts") }
discard_on DiscardableError
discard_on(CustomDiscardableError) { |job, exception| JobBuffer.add("Dealt with a job that was discarded in a custom way") }

def perform(raising, attempts)
if executions < attempts
Expand Down

0 comments on commit 3291fa3

Please sign in to comment.