active_storage: fix blob update hooks

For some reason on Rails 6.1 the `after_update_commit` hook is properly
registered – but disappears from the record later, and in the end is
never run.

Fix it by using the general `after_commit` hook instead.
This commit is contained in:
Pierre de La Morinerie 2021-03-04 11:50:22 +00:00
parent 21980a37cc
commit 2f948f7e46
3 changed files with 71 additions and 14 deletions

View file

@ -1,8 +1,19 @@
# Request a watermark on blobs attached to a `Champs::TitreIdentiteChamp`
# after the virus scan has run.
#
# We're using a class extension here, but we could as well have a periodic
# job that watermarks relevant attachments.
#
# The `after_commit` hook is triggered, among other cases, when
# the analyzer or virus scan updates the blob metadata. When both the analyzer
# and the virus scan have run, it is now safe to start the watermarking,
# without risking to replace the picture while it is being scanned in a
# concurrent job.
module BlobTitreIdentiteWatermarkConcern module BlobTitreIdentiteWatermarkConcern
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
after_update_commit :enqueue_watermark_job after_commit :enqueue_watermark_job
end end
def watermark_pending? def watermark_pending?
@ -12,7 +23,7 @@ module BlobTitreIdentiteWatermarkConcern
private private
def watermark_required? def watermark_required?
attachments.find { |attachment| attachment.record.class.name == 'Champs::TitreIdentiteChamp' } attachments.any? { |attachment| attachment.record.class.name == 'Champs::TitreIdentiteChamp' }
end end
def watermark_done? def watermark_done?

View file

@ -1,13 +1,21 @@
# TODO: once we're using Rails 6, use the hooks on attachments creation # Run a virus scan on all blobs after they are analyzed.
# (rather than on blob creation). #
# This will help to avoid cloberring metadata accidentally (as metadata # We're using a class extension to ensure that all blobs get scanned,
# are more stable on attachment creation than on blob creation). # regardless on how they were created. This could be an ActiveStorage::Analyzer,
# but as of Rails 6.1 only the first matching analyzer is ever run on
# a blob (and we may want to analyze the dimension of a picture as well
# as scanning it).
#
# The `after_commit` hook is triggered, among other cases, when
# the analyzer updates the blob metadata. When the analyzer has run,
# it is now safe to start our own scanning, without risking to have
# two concurrent jobs overwriting the metadata of the blob.
module BlobVirusScannerConcern module BlobVirusScannerConcern
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
before_create :set_pending before_create :set_pending
after_update_commit :enqueue_virus_scan after_commit :enqueue_virus_scan
end end
def virus_scanner def virus_scanner

View file

@ -1,4 +1,6 @@
describe Champ do describe Champ do
include ActiveJob::TestHelper
require 'models/champ_shared_example.rb' require 'models/champ_shared_example.rb'
it_should_behave_like "champ_spec" it_should_behave_like "champ_spec"
@ -435,19 +437,56 @@ describe Champ do
end end
end end
describe '#enqueue_virus_check' do describe '#enqueue_virus_scan' do
let(:champ) { build(:champ_piece_justificative, type_de_champ: type_de_champ) }
context 'when type_champ is type_de_champ_piece_justificative' do context 'when type_champ is type_de_champ_piece_justificative' do
let(:type_de_champ) { create(:type_de_champ_piece_justificative) } let(:type_de_champ) { create(:type_de_champ_piece_justificative) }
let(:champ) { build(:champ_piece_justificative, type_de_champ: type_de_champ) }
context 'and there is a blob' do context 'and there is a blob' do
before do before do
champ.piece_justificative_file.attach(io: StringIO.new("toto"), filename: "toto.txt", content_type: "text/plain") allow(ClamavService).to receive(:safe_file?).and_return(true)
champ.save!
end end
it { expect(champ.piece_justificative_file.virus_scanner.started?).to be_truthy } subject do
champ.piece_justificative_file.attach(io: StringIO.new("toto"), filename: "toto.txt", content_type: "text/plain")
champ.save!
champ
end
it 'marks the file as pending virus scan' do
expect(subject.piece_justificative_file.virus_scanner.started?).to be_truthy
end
it 'marks the file as safe once the scan completes' do
perform_enqueued_jobs { subject }
expect(champ.reload.piece_justificative_file.virus_scanner.safe?).to be_truthy
end
end
end
end
describe '#enqueue_watermark_job' do
context 'when type_champ is type_de_champ_titre_identite' do
let(:type_de_champ) { create(:type_de_champ_titre_identite) }
let(:champ) { build(:champ_titre_identite, type_de_champ: type_de_champ) }
before do
allow(ClamavService).to receive(:safe_file?).and_return(true)
end
subject do
champ.piece_justificative_file.attach(fixture_file_upload('spec/fixtures/files/logo_test_procedure.png', 'image/png'))
champ.save!
champ
end
it 'enqueues a watermark job on file attachment' do
expect(subject.piece_justificative_file.watermark_pending?).to be_truthy
end
it 'watermarks the file' do
perform_enqueued_jobs { subject }
expect(champ.reload.piece_justificative_file.blob.metadata[:watermark]).to be_truthy
end end
end end
end end
@ -535,7 +574,6 @@ describe Champ do
end end
context "fetch_external_data_later" do context "fetch_external_data_later" do
include ActiveJob::TestHelper
let(:data) { 'some other data' } let(:data) { 'some other data' }
it "fill data from external source" do it "fill data from external source" do