Commit 4407bcd2 authored by Douglas Barbosa Alexandre's avatar Douglas Barbosa Alexandre

Merge branch 'improve-file_registry_removal_service' into 'master'

Refactoring FileRegistryRemoval service  and specs

See merge request gitlab-org/gitlab!77539
parents cf96c74e a533ce2d
# frozen_string_literal: true # frozen_string_literal: true
module Geo module Geo
##
## Geo::FileRegistryRemovalService handles blob removal from a secondary node,
## the file itself and the database records.
## It handles all the possible combinations of 4 variables:
## * Whether Model record exists
## * Whether Registry Record exists
## * Whether the file on a storage exists
## * Whether the file_path is passed (RegistryConsistencyWorker doesn't pass one)
## In all the cases the best effort should have to be made.
##
class FileRegistryRemovalService < BaseFileService class FileRegistryRemovalService < BaseFileService
include ::Gitlab::Utils::StrongMemoize include ::Gitlab::Utils::StrongMemoize
...@@ -73,16 +83,10 @@ module Geo ...@@ -73,16 +83,10 @@ module Geo
end end
end end
def blob_path_from_replicator
replicator.blob_path
rescue ActiveRecord::RecordNotFound
nil
end
def file_path def file_path
strong_memoize(:file_path) do strong_memoize(:file_path) do
next @object_file_path if @object_file_path next @object_file_path if @object_file_path
next blob_path_from_replicator if replicator
# When local storage is used, just rely on the existing methods # When local storage is used, just rely on the existing methods
next if file_uploader.nil? next if file_uploader.nil?
next file_uploader.file.path if file_uploader.object_store == ObjectStorage::Store::LOCAL next file_uploader.file.path if file_uploader.object_store == ObjectStorage::Store::LOCAL
...@@ -93,6 +97,8 @@ module Geo ...@@ -93,6 +97,8 @@ module Geo
def file_uploader def file_uploader
strong_memoize(:file_uploader) do strong_memoize(:file_uploader) do
next replicator.carrierwave_uploader if replicator
case object_type case object_type
when :job_artifact when :job_artifact
Ci::JobArtifact.find(object_db_id).file Ci::JobArtifact.find(object_db_id).file
......
...@@ -21,115 +21,43 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do ...@@ -21,115 +21,43 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do
described_class.new(:lfs, 99).execute described_class.new(:lfs, 99).execute
end end
shared_examples 'removes artifact' do context 'with job artifact' do
subject(:service) { described_class.new('job_artifact', registry.artifact_id) } shared_examples 'removes artifact' do
subject(:service) { described_class.new('job_artifact', registry.artifact_id) }
before do
stub_exclusive_lease("file_registry_removal_service:job_artifact:#{registry.artifact_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'file from disk' do
expect do
service.execute
end.to change { File.exist?(file_path) }.from(true).to(false)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::JobArtifactRegistry, :count).by(-1)
end
end
shared_examples 'removes artifact registry' do
subject(:service) { described_class.new('job_artifact', registry.artifact_id) }
before do
stub_exclusive_lease("file_registry_removal_service:job_artifact:#{registry.artifact_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::JobArtifactRegistry, :count).by(-1)
end
end
shared_examples 'removes LFS object' do
subject(:service) { described_class.new('lfs', registry.lfs_object_id) }
before do
stub_exclusive_lease("file_registry_removal_service:lfs:#{registry.lfs_object_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'file from disk' do
expect do
service.execute
end.to change { File.exist?(file_path) }.from(true).to(false)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::LfsObjectRegistry, :count).by(-1)
end
end
shared_examples 'removes LFS object registry' do
subject(:service) { described_class.new('lfs', registry.lfs_object_id) }
before do
stub_exclusive_lease("file_registry_removal_service:lfs:#{registry.lfs_object_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::LfsObjectRegistry, :count).by(-1)
end
end
shared_examples 'removes package file' do
subject(:service) { described_class.new('package_file', registry.package_file_id) }
before do before do
stub_exclusive_lease("file_registry_removal_service:package_file:#{registry.package_file_id}", stub_exclusive_lease("file_registry_removal_service:job_artifact:#{registry.artifact_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT) timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end end
it 'file from disk' do it 'file from disk' do
expect do expect do
service.execute service.execute
end.to change { File.exist?(file_path) }.from(true).to(false) end.to change { File.exist?(file_path) }.from(true).to(false)
end end
it 'deletes registry entry' do it 'deletes registry entry' do
expect do expect do
service.execute service.execute
end.to change(Geo::PackageFileRegistry, :count).by(-1) end.to change(Geo::JobArtifactRegistry, :count).by(-1)
end
end end
end
shared_examples 'removes package file registry' do shared_examples 'removes artifact registry' do
subject(:service) { described_class.new('package_file', registry.package_file_id) } subject(:service) { described_class.new('job_artifact', registry.artifact_id) }
before do before do
stub_exclusive_lease("file_registry_removal_service:package_file:#{registry.package_file_id}", stub_exclusive_lease("file_registry_removal_service:job_artifact:#{registry.artifact_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT) timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end end
it 'deletes registry entry' do it 'deletes registry entry' do
expect do expect do
service.execute service.execute
end.to change(Geo::PackageFileRegistry, :count).by(-1) end.to change(Geo::JobArtifactRegistry, :count).by(-1)
end
end end
end
context 'with job artifact' do
let!(:job_artifact) { create(:ci_job_artifact, :archive) } let!(:job_artifact) { create(:ci_job_artifact, :archive) }
let!(:registry) { create(:geo_job_artifact_registry, artifact_id: job_artifact.id) } let!(:registry) { create(:geo_job_artifact_registry, artifact_id: job_artifact.id) }
let!(:file_path) { job_artifact.file.path } let!(:file_path) { job_artifact.file.path }
...@@ -138,20 +66,14 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do ...@@ -138,20 +66,14 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do
context 'migrated to object storage' do context 'migrated to object storage' do
before do before do
stub_artifacts_object_storage job_artifact.update_column(:file_store, ObjectStorage::Store::REMOTE)
job_artifact.update_column(:file_store, JobArtifactUploader::Store::REMOTE)
end
it_behaves_like 'removes artifact'
end
context 'migrated to object storage' do
before do
stub_artifacts_object_storage
job_artifact.update_column(:file_store, LfsObjectUploader::Store::REMOTE)
end end
context 'with object storage enabled' do context 'with object storage enabled' do
before do
stub_artifacts_object_storage
end
it_behaves_like 'removes artifact' it_behaves_like 'removes artifact'
end end
...@@ -172,30 +94,52 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do ...@@ -172,30 +94,52 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do
it_behaves_like 'removes artifact' do it_behaves_like 'removes artifact' do
subject(:service) { described_class.new('job_artifact', registry.artifact_id, file_path) } subject(:service) { described_class.new('job_artifact', registry.artifact_id, file_path) }
end end
it_behaves_like 'removes artifact registry'
end end
end
context 'with package file' do
shared_examples 'removes package file' do
subject(:service) { described_class.new('package_file', registry.package_file_id) }
context 'with orphaned registry' do
before do before do
job_artifact.delete stub_exclusive_lease("file_registry_removal_service:package_file:#{registry.package_file_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end end
it_behaves_like 'removes artifact registry' do it 'file from disk' do
subject(:service) { described_class.new('job_artifact', registry.artifact_id) } expect do
service.execute
end.to change { File.exist?(file_path) }.from(true).to(false)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::PackageFileRegistry, :count).by(-1)
end end
end end
end
context 'with package file' do shared_examples 'removes package file registry' do
let(:package_file) { create(:package_file_with_file) } subject(:service) { described_class.new('package_file', registry.package_file_id) }
let!(:registry) { create(:geo_package_file_registry, package_file: package_file) }
let(:file_path) { Tempfile.new.path } before do
stub_exclusive_lease("file_registry_removal_service:package_file:#{registry.package_file_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
before do it 'deletes registry entry' do
allow_next_instance_of(Geo::PackageFileReplicator) do |replicator| expect do
allow(replicator).to receive(:blob_path).and_return(file_path) service.execute
end.to change(Geo::PackageFileRegistry, :count).by(-1)
end end
end end
let(:package_file) { create(:package_file_with_file) }
let!(:registry) { create(:geo_package_file_registry, package_file: package_file) }
let(:file_path) { package_file.file.path }
it_behaves_like 'removes package file' it_behaves_like 'removes package file'
context 'no package file record' do context 'no package file record' do
...@@ -220,27 +164,80 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do ...@@ -220,27 +164,80 @@ RSpec.describe Geo::FileRegistryRemovalService, :geo do
end end
context 'with uploads' do context 'with uploads' do
let!(:upload) { create(:user, :with_avatar).avatar.upload } shared_examples 'removes upload' do
subject(:service) { described_class.new('upload', registry.file_id, file_path) }
before do
stub_exclusive_lease("file_registry_removal_service:upload:#{registry.file_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'file from disk' do
expect do
service.execute
end.to change { File.exist?(file_path) }.from(true).to(false)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::UploadRegistry, :count).by(-1)
end
end
shared_examples 'removes upload registry' do
subject(:service) { described_class.new('upload', registry.file_id, file_path) }
before do
stub_exclusive_lease("file_registry_removal_service:upload:#{registry.file_id}",
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT)
end
it 'deletes registry entry' do
expect do
service.execute
end.to change(Geo::UploadRegistry, :count).by(-1)
end
end
let!(:upload) { create(:upload, :with_file) }
let!(:registry) { create(:geo_upload_registry, file_id: upload.id) } let!(:registry) { create(:geo_upload_registry, file_id: upload.id) }
let!(:file_path) { upload.retrieve_uploader.file.path } let!(:file_path) { upload.retrieve_uploader.file.path }
subject(:service) { described_class.new('upload', registry.file_id) } it_behaves_like 'removes upload'
before do context 'migrated to object storage' do
stub_exclusive_lease("file_registry_removal_service:upload:#{registry.file_id}", before do
timeout: Geo::FileRegistryRemovalService::LEASE_TIMEOUT) upload.update_column(:store, ObjectStorage::Store::REMOTE)
end end
context 'with object storage enabled' do
before do
stub_uploads_object_storage(AvatarUploader)
end
it_behaves_like 'removes upload'
end
context 'with object storage disabled' do
before do
stub_uploads_object_storage(AvatarUploader, enabled: false)
end
it 'file from disk' do it_behaves_like 'removes upload registry'
expect do end
service.execute
end.to change { File.exist?(file_path) }.from(true).to(false)
end end
it 'deletes registry entry' do context 'no upload record' do
expect do before do
service.execute upload.delete
end.to change(Geo::UploadRegistry, :count).by(-1) end
it_behaves_like 'removes upload' do
subject(:service) { described_class.new('upload', registry.file_id, file_path) }
end
it_behaves_like 'removes upload registry'
end end
end end
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