45 lines
1.2 KiB
Ruby
45 lines
1.2 KiB
Ruby
|
module WaitableWorker
|
||
|
extend ActiveSupport::Concern
|
||
|
|
||
|
module ClassMethods
|
||
|
# Schedules multiple jobs and waits for them to be completed.
|
||
|
def bulk_perform_and_wait(args_list, timeout: 10)
|
||
|
# Short-circuit: it's more efficient to do small numbers of jobs inline
|
||
|
return bulk_perform_inline(args_list) if args_list.size <= 3
|
||
|
|
||
|
waiter = Gitlab::JobWaiter.new(args_list.size)
|
||
|
|
||
|
# Point all the bulk jobs at the same JobWaiter. Converts, [[1], [2], [3]]
|
||
|
# into [[1, "key"], [2, "key"], [3, "key"]]
|
||
|
waiting_args_list = args_list.map { |args| [*args, waiter.key] }
|
||
|
bulk_perform_async(waiting_args_list)
|
||
|
|
||
|
waiter.wait(timeout)
|
||
|
end
|
||
|
|
||
|
# Performs multiple jobs directly. Failed jobs will be put into sidekiq so
|
||
|
# they can benefit from retries
|
||
|
def bulk_perform_inline(args_list)
|
||
|
failed = []
|
||
|
|
||
|
args_list.each do |args|
|
||
|
begin
|
||
|
new.perform(*args)
|
||
|
rescue
|
||
|
failed << args
|
||
|
end
|
||
|
end
|
||
|
|
||
|
bulk_perform_async(failed) if failed.present?
|
||
|
end
|
||
|
end
|
||
|
|
||
|
def perform(*args)
|
||
|
notify_key = args.pop if Gitlab::JobWaiter.key?(args.last)
|
||
|
|
||
|
super(*args)
|
||
|
ensure
|
||
|
Gitlab::JobWaiter.notify(notify_key, jid) if notify_key
|
||
|
end
|
||
|
end
|