mirror of https://github.com/rails/rails
Merge pull request #30622 from aidanharan/custom-discarded-job-handling
Allow for custom handling of exceptions that are discarded
This commit is contained in:
commit
4cfd40a580
|
@ -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*
|
||||
|
||||
## Rails 5.2.0.beta2 (November 28, 2017) ##
|
||||
|
||||
* No changes.
|
||||
|
|
|
@ -61,18 +61,28 @@ module ActiveJob
|
|||
# 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
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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
|
||||
|
@ -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. Message: #{exception.message}") }
|
||||
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
|
||||
|
|
Loading…
Reference in New Issue