Commit 7915af8a authored by Vasilii Iakliushin's avatar Vasilii Iakliushin

Don't fail DeleteStoredFilesWorker if one of the files is missing

Contributes to https://gitlab.com/gitlab-org/gitlab/-/issues/342691

**Problem**

`DeleteStoredFilesWorker` sidekiq job fails when trying to delete
already missing files from the GoogleCloud bucket. Other
provides (AWS, ...) does not raise an error when the file is not
found.

**Solution**

Handle ClientError exception and proceed deleting even if one of the
files is already removed.

Changelog: fixed
parent 071ef85a
...@@ -15,13 +15,21 @@ module Uploads ...@@ -15,13 +15,21 @@ module Uploads
end end
def delete_keys(keys) def delete_keys(keys)
keys.each do |key| keys.each { |key| delete_object(key) }
connection.delete_object(bucket_name, key)
end
end end
private private
def delete_object(key)
connection.delete_object(bucket_name, key)
# So far, only GoogleCloudStorage raises an exception when the file is not found.
# Other providers support idempotent requests and does not raise an error
# when the file is missing.
rescue ::Google::Apis::ClientError => e
Gitlab::ErrorTracking.log_exception(e)
end
def object_store def object_store
Gitlab.config.uploads.object_store Gitlab.config.uploads.object_store
end end
......
...@@ -40,7 +40,9 @@ RSpec.describe Uploads::Fog do ...@@ -40,7 +40,9 @@ RSpec.describe Uploads::Fog do
end end
describe '#delete_keys' do describe '#delete_keys' do
let(:connection) { ::Fog::Storage.new(FileUploader.object_store_credentials) }
let(:keys) { data_store.keys(relation) } let(:keys) { data_store.keys(relation) }
let(:paths) { relation.pluck(:path) }
let!(:uploads) { create_list(:upload, 2, :with_file, :issuable_upload, model: project) } let!(:uploads) { create_list(:upload, 2, :with_file, :issuable_upload, model: project) }
subject { data_store.delete_keys(keys) } subject { data_store.delete_keys(keys) }
...@@ -50,17 +52,32 @@ RSpec.describe Uploads::Fog do ...@@ -50,17 +52,32 @@ RSpec.describe Uploads::Fog do
end end
it 'deletes multiple data' do it 'deletes multiple data' do
paths = relation.pluck(:path) paths.each do |path|
expect(connection.get_object('uploads', path)[:body]).not_to be_nil
end
subject
paths.each do |path|
expect { connection.get_object('uploads', path)[:body] }.to raise_error(Excon::Error::NotFound)
end
end
context 'when one of keys is missing' do
let(:keys) { ['unknown'] + super() }
::Fog::Storage.new(FileUploader.object_store_credentials).tap do |connection| it 'deletes only existing keys' do
paths.each do |path| paths.each do |path|
expect(connection.get_object('uploads', path)[:body]).not_to be_nil expect(connection.get_object('uploads', path)[:body]).not_to be_nil
end end
expect_next_instance_of(::Fog::Storage) do |storage|
allow(storage).to receive(:delete_object).and_call_original
expect(storage).to receive(:delete_object).with('uploads', keys.first).and_raise(::Google::Apis::ClientError, 'NotFound')
end end
subject subject
::Fog::Storage.new(FileUploader.object_store_credentials).tap do |connection|
paths.each do |path| paths.each do |path|
expect { connection.get_object('uploads', path)[:body] }.to raise_error(Excon::Error::NotFound) expect { connection.get_object('uploads', path)[:body] }.to raise_error(Excon::Error::NotFound)
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment