This is part of the Semicolon&Sons Code Diary - consisting of lessons learned on the job. You're in the architecture category.
Last Updated: 2025-01-18
I have a background job in my code that emails every zip file contained in an order to customers. Roughly, the code is as follows:
class EmailFilesJob
def perform(order)
order.zip_files_available_for_download.each do |zip_file|
file = Paperclip.io_adapters.for(zip_file.zip)
GenericEmail.send_file(file, order.user.email).deliver_now
end
end
end
One customer an obscenely large order with ~60 zip files, one of which was too big to email. When this email files job executed, it got through the first 15 files, then on the 16th it hit the huge file and the job aborted.
Nothing too bad so far. Except that my job-runner retried this job two more times, leading to the first 15 files being delivered three times apiece to the user's inbox, leaving an unprofessional mess.
The lesson here is that background jobs that take many smaller actions (e.g. send 60 emails) should somehow maintain progress (e.g. with state or by queuing up sub-background jobs) to avoid breaking idempotency.
This could have been achieved with a simple change:
class EmailFilesJob
def perform(order)
order.zip_files_available_for_download.each do |zip_file|
file = Paperclip.io_adapters.for(zip_file.zip)
# deliver_later queues up another job
GenericEmail.send_file(file, order.user.email).deliver_later
end
end
end
deliver_later
would queue up the creation of another job to send that one zip
file, meaning that the overall EmailFilesJob would execute perfectly every time,
since all it does is queue up other jobs. The individual email that was too
large would exclusively fail, and this can be safely retried.