Commit 4cd853e7 authored by Douglas Barbosa Alexandre's avatar Douglas Barbosa Alexandre

Merge branch 'tc-geo-log-object-storage' into 'master'

Use Geo log to remove files when migrated to object storage

Closes #7108, #6135, and #6040

See merge request gitlab-org/gitlab-ee!5966
parents a85c4cf4 6f6c3c6f
...@@ -63,6 +63,12 @@ class GitlabUploader < CarrierWave::Uploader::Base ...@@ -63,6 +63,12 @@ class GitlabUploader < CarrierWave::Uploader::Base
super || file&.filename super || file&.filename
end end
def relative_path
return path if pathname.relative?
pathname.relative_path_from(Pathname.new(root))
end
def model_valid? def model_valid?
!!model !!model
end end
...@@ -115,4 +121,8 @@ class GitlabUploader < CarrierWave::Uploader::Base ...@@ -115,4 +121,8 @@ class GitlabUploader < CarrierWave::Uploader::Base
# the cache directory. # the cache directory.
File.join(work_dir, cache_id, version_name.to_s, for_file) File.join(work_dir, cache_id, version_name.to_s, for_file)
end end
def pathname
@pathname ||= Pathname.new(path)
end
end end
...@@ -9,6 +9,8 @@ class JobArtifactUploader < GitlabUploader ...@@ -9,6 +9,8 @@ class JobArtifactUploader < GitlabUploader
storage_options Gitlab.config.artifacts storage_options Gitlab.config.artifacts
alias_method :upload, :model
def cached_size def cached_size
return model.size if model.size.present? && !model.file_changed? return model.size if model.size.present? && !model.file_changed?
......
...@@ -8,6 +8,8 @@ class LegacyArtifactUploader < GitlabUploader ...@@ -8,6 +8,8 @@ class LegacyArtifactUploader < GitlabUploader
storage_options Gitlab.config.artifacts storage_options Gitlab.config.artifacts
alias_method :upload, :model
def store_dir def store_dir
dynamic_segment dynamic_segment
end end
......
...@@ -6,6 +6,8 @@ class LfsObjectUploader < GitlabUploader ...@@ -6,6 +6,8 @@ class LfsObjectUploader < GitlabUploader
storage_options Gitlab.config.lfs storage_options Gitlab.config.lfs
alias_method :upload, :model
def filename def filename
model.oid[4..-1] model.oid[4..-1]
end end
......
...@@ -137,6 +137,8 @@ module ObjectStorage ...@@ -137,6 +137,8 @@ module ObjectStorage
included do |base| included do |base|
base.include(ObjectStorage) base.include(ObjectStorage)
include ::EE::ObjectStorage::Concern
after :migrate, :delete_migrated_file after :migrate, :delete_migrated_file
end end
......
...@@ -81,6 +81,12 @@ module EE ...@@ -81,6 +81,12 @@ module EE
has_artifact?(DAST_FILE) has_artifact?(DAST_FILE)
end end
def log_geo_deleted_event
# It is not needed to generate a Geo deleted event
# since Legacy Artifacts are migrated to multi-build artifacts
# See https://gitlab.com/gitlab-org/gitlab-ce/issues/46652
end
private private
def has_artifact?(name) def has_artifact?(name)
......
...@@ -7,15 +7,13 @@ module EE ...@@ -7,15 +7,13 @@ module EE
extend ActiveSupport::Concern extend ActiveSupport::Concern
prepended do prepended do
after_destroy :log_geo_event after_destroy :log_geo_deleted_event
scope :not_expired, -> { where('expire_at IS NULL OR expire_at > ?', Time.current) } scope :not_expired, -> { where('expire_at IS NULL OR expire_at > ?', Time.current) }
scope :geo_syncable, -> { with_files_stored_locally.not_expired } scope :geo_syncable, -> { with_files_stored_locally.not_expired }
end end
private def log_geo_deleted_event
def log_geo_event
::Geo::JobArtifactDeletedEventStore.new(self).create! ::Geo::JobArtifactDeletedEventStore.new(self).create!
end end
end end
......
...@@ -7,15 +7,13 @@ module EE ...@@ -7,15 +7,13 @@ module EE
extend ActiveSupport::Concern extend ActiveSupport::Concern
prepended do prepended do
after_destroy :log_geo_event after_destroy :log_geo_deleted_event
scope :geo_syncable, -> { with_files_stored_locally } scope :geo_syncable, -> { with_files_stored_locally }
scope :with_files_stored_remotely, -> { where(file_store: LfsObjectUploader::Store::REMOTE) } scope :with_files_stored_remotely, -> { where(file_store: LfsObjectUploader::Store::REMOTE) }
end end
private def log_geo_deleted_event
def log_geo_event
::Geo::LfsObjectDeletedEventStore.new(self).create! ::Geo::LfsObjectDeletedEventStore.new(self).create!
end end
end end
......
...@@ -4,7 +4,7 @@ module EE ...@@ -4,7 +4,7 @@ module EE
extend ::Gitlab::Utils::Override extend ::Gitlab::Utils::Override
prepended do prepended do
include ObjectStorage::BackgroundMove include ::ObjectStorage::BackgroundMove
end end
def for_epic? def for_epic?
......
...@@ -7,14 +7,12 @@ module EE ...@@ -7,14 +7,12 @@ module EE
extend ActiveSupport::Concern extend ActiveSupport::Concern
prepended do prepended do
after_destroy :log_geo_event after_destroy :log_geo_deleted_event
scope :geo_syncable, -> { with_files_stored_locally } scope :geo_syncable, -> { with_files_stored_locally }
end end
private def log_geo_deleted_event
def log_geo_event
::Geo::UploadDeletedEventStore.new(self).create! ::Geo::UploadDeletedEventStore.new(self).create!
end end
end end
......
# frozen_string_literal: true
module Geo module Geo
# This class is responsible for: # This class is responsible for:
# * Finding the appropriate Downloader class for a FileRegistry record # * Finding the appropriate Downloader class for a FileRegistry record
...@@ -50,7 +52,7 @@ module Geo ...@@ -50,7 +52,7 @@ module Geo
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
def update_registry(bytes_downloaded, mark_as_synced:, missing_on_primary: false) def update_registry(bytes_downloaded, mark_as_synced:, missing_on_primary: false)
registry = registry =
if object_type.to_sym == :job_artifact if job_artifact?
Geo::JobArtifactRegistry.find_or_initialize_by(artifact_id: object_db_id) Geo::JobArtifactRegistry.find_or_initialize_by(artifact_id: object_db_id)
else else
Geo::FileRegistry.find_or_initialize_by( Geo::FileRegistry.find_or_initialize_by(
......
# frozen_string_literal: true
module Geo module Geo
class FileRegistryRemovalService < FileService class FileRegistryRemovalService < FileService
include ::Gitlab::Utils::StrongMemoize include ::Gitlab::Utils::StrongMemoize
...@@ -25,7 +27,7 @@ module Geo ...@@ -25,7 +27,7 @@ module Geo
log_info('Local file & registry removed') log_info('Local file & registry removed')
end end
rescue SystemCallError rescue SystemCallError => e
log_error('Could not remove file', e.message) log_error('Could not remove file', e.message)
raise raise
end end
...@@ -35,7 +37,7 @@ module Geo ...@@ -35,7 +37,7 @@ module Geo
# rubocop: disable CodeReuse/ActiveRecord # rubocop: disable CodeReuse/ActiveRecord
def file_registry def file_registry
strong_memoize(:file_registry) do strong_memoize(:file_registry) do
if object_type.to_sym == :job_artifact if job_artifact?
::Geo::JobArtifactRegistry.find_by(artifact_id: object_db_id) ::Geo::JobArtifactRegistry.find_by(artifact_id: object_db_id)
else else
::Geo::FileRegistry.find_by(file_type: object_type, file_id: object_db_id) ::Geo::FileRegistry.find_by(file_type: object_type, file_id: object_db_id)
...@@ -59,16 +61,15 @@ module Geo ...@@ -59,16 +61,15 @@ module Geo
end end
end end
# rubocop: disable CodeReuse/ActiveRecord
def file_uploader def file_uploader
strong_memoize(:file_uploader) do strong_memoize(:file_uploader) do
case object_type.to_s case object_type
when 'lfs' when :lfs
LfsObject.find_by!(id: object_db_id).file LfsObject.find(object_db_id).file
when 'job_artifact' when :job_artifact
Ci::JobArtifact.find_by!(id: object_db_id).file Ci::JobArtifact.find(object_db_id).file
when *Geo::FileService::DEFAULT_OBJECT_TYPES when *Geo::FileService::DEFAULT_OBJECT_TYPES
Upload.find_by!(id: object_db_id).build_uploader Upload.find(object_db_id).build_uploader
else else
raise NameError, "Unrecognized type: #{object_type}" raise NameError, "Unrecognized type: #{object_type}"
end end
...@@ -77,11 +78,6 @@ module Geo ...@@ -77,11 +78,6 @@ module Geo
log_error('Could not build uploader', err.message) log_error('Could not build uploader', err.message)
raise raise
end end
# rubocop: enable CodeReuse/ActiveRecord
def upload?
Geo::FileService::DEFAULT_OBJECT_TYPES.include?(object_type.to_s)
end
def lease_key def lease_key
"file_registry_removal_service:#{object_type}:#{object_db_id}" "file_registry_removal_service:#{object_type}:#{object_db_id}"
......
# frozen_string_literal: true
module Geo module Geo
class FileService class FileService
include ExclusiveLeaseGuard include ExclusiveLeaseGuard
...@@ -5,8 +7,8 @@ module Geo ...@@ -5,8 +7,8 @@ module Geo
attr_reader :object_type, :object_db_id attr_reader :object_type, :object_db_id
DEFAULT_OBJECT_TYPES = %w[attachment avatar file import_export namespace_file personal_file favicon].freeze DEFAULT_OBJECT_TYPES = %i[attachment avatar file import_export namespace_file personal_file favicon].freeze
DEFAULT_SERVICE_TYPE = 'file'.freeze DEFAULT_SERVICE_TYPE = :file
def initialize(object_type, object_db_id) def initialize(object_type, object_db_id)
@object_type = object_type.to_sym @object_type = object_type.to_sym
...@@ -19,9 +21,17 @@ module Geo ...@@ -19,9 +21,17 @@ module Geo
private private
def upload?
DEFAULT_OBJECT_TYPES.include?(object_type)
end
def job_artifact?
object_type == :job_artifact
end
def service_klass_name def service_klass_name
klass_name = klass_name =
if DEFAULT_OBJECT_TYPES.include?(object_type.to_s) if upload?
DEFAULT_SERVICE_TYPE DEFAULT_SERVICE_TYPE
else else
object_type object_type
...@@ -33,7 +43,7 @@ module Geo ...@@ -33,7 +43,7 @@ module Geo
def base_log_data(message) def base_log_data(message)
{ {
class: self.class.name, class: self.class.name,
object_type: object_type.to_s, object_type: object_type,
object_db_id: object_db_id, object_db_id: object_db_id,
message: message message: message
} }
......
...@@ -10,13 +10,6 @@ module Geo ...@@ -10,13 +10,6 @@ module Geo
@job_artifact = job_artifact @job_artifact = job_artifact
end end
override :create!
def create!
return unless job_artifact.local_store?
super
end
private private
def build_event def build_event
...@@ -26,14 +19,8 @@ module Geo ...@@ -26,14 +19,8 @@ module Geo
) )
end end
def local_store_path
Pathname.new(JobArtifactUploader.root)
end
def relative_file_path def relative_file_path
return unless job_artifact.file.present? job_artifact.file.relative_path if job_artifact.file.present?
Pathname.new(job_artifact.file.path).relative_path_from(local_store_path)
end end
# This is called by ProjectLogHelpers to build json log with context info # This is called by ProjectLogHelpers to build json log with context info
......
...@@ -10,13 +10,6 @@ module Geo ...@@ -10,13 +10,6 @@ module Geo
@lfs_object = lfs_object @lfs_object = lfs_object
end end
override :create!
def create!
return unless lfs_object.local_store?
super
end
private private
def build_event def build_event
...@@ -27,14 +20,8 @@ module Geo ...@@ -27,14 +20,8 @@ module Geo
) )
end end
def local_store_path
Pathname.new(LfsObjectUploader.root)
end
def relative_file_path def relative_file_path
return unless lfs_object.file.present? lfs_object.file.relative_path if lfs_object.file.present?
Pathname.new(lfs_object.file.path).relative_path_from(local_store_path)
end end
# This is called by ProjectLogHelpers to build json log with context info # This is called by ProjectLogHelpers to build json log with context info
......
...@@ -10,13 +10,6 @@ module Geo ...@@ -10,13 +10,6 @@ module Geo
@upload = upload @upload = upload
end end
override :create!
def create!
return unless upload.local?
super
end
private private
def build_event def build_event
......
module EE
module ObjectStorage
module Concern
extend ActiveSupport::Concern
included do
after :migrate, :log_geo_deleted_event
end
private
def log_geo_deleted_event(_migrated_file)
upload.log_geo_deleted_event
end
end
end
end
---
title: Use Geo log to remove files when migrated to object storage
merge_request: 5966
author:
type: added
...@@ -6,42 +6,20 @@ module Gitlab ...@@ -6,42 +6,20 @@ module Gitlab
include BaseEvent include BaseEvent
def process def process
return unless file_registry_job_artifacts.any? # avoid race condition # Must always schedule, regardless of shard health
job_id = ::Geo::FileRegistryRemovalWorker.perform_async(:job_artifact, event.job_artifact_id)
# delete synchronously to ensure consistency log_event(job_id)
if File.file?(file_path) && !delete_file(file_path)
return # do not delete file from registry if deletion failed
end
log_event
file_registry_job_artifacts.delete_all
end end
private private
# rubocop: disable CodeReuse/ActiveRecord def log_event(job_id)
def file_registry_job_artifacts
@file_registry_job_artifacts ||= ::Geo::JobArtifactRegistry.where(artifact_id: event.job_artifact_id)
end
# rubocop: enable CodeReuse/ActiveRecord
def file_path
@file_path ||= File.join(::JobArtifactUploader.root, event.file_path)
end
def log_event
logger.event_info( logger.event_info(
created_at, created_at,
'Deleted job artifact', 'Delete job artifact scheduled',
file_id: event.job_artifact_id, file_id: event.job_artifact_id,
file_path: file_path) file_path: event.file_path,
end job_id: job_id)
def delete_file(path)
File.delete(path)
rescue => ex
logger.error("Failed to remove file", exception: ex.class.name, details: ex.message, filename: path)
false
end end
end end
end end
......
...@@ -5,28 +5,21 @@ module Gitlab ...@@ -5,28 +5,21 @@ module Gitlab
class LfsObjectDeletedEvent class LfsObjectDeletedEvent
include BaseEvent include BaseEvent
# rubocop: disable CodeReuse/ActiveRecord
def process def process
# Must always schedule, regardless of shard health # Must always schedule, regardless of shard health
job_id = ::Geo::FileRemovalWorker.perform_async(file_path) job_id = ::Geo::FileRegistryRemovalWorker.perform_async(:lfs, event.lfs_object_id)
log_event(job_id) log_event(job_id)
::Geo::FileRegistry.lfs_objects.where(file_id: event.lfs_object_id).delete_all
end end
# rubocop: enable CodeReuse/ActiveRecord
private private
def file_path
@file_path ||= File.join(LfsObjectUploader.root, event.file_path)
end
def log_event(job_id) def log_event(job_id)
logger.event_info( logger.event_info(
created_at, created_at,
'Deleted LFS object', 'Delete LFS object scheduled',
oid: event.oid, oid: event.oid,
file_id: event.lfs_object_id, file_id: event.lfs_object_id,
file_path: file_path, file_path: event.file_path,
job_id: job_id) job_id: job_id)
end end
end end
......
...@@ -5,24 +5,21 @@ module Gitlab ...@@ -5,24 +5,21 @@ module Gitlab
class UploadDeletedEvent class UploadDeletedEvent
include BaseEvent include BaseEvent
# rubocop: disable CodeReuse/ActiveRecord
def process def process
log_event job_id = ::Geo::FileRegistryRemovalWorker.perform_async(event.upload_type, event.upload_id)
::Geo::FileRegistry.where(file_id: event.upload_id, file_type: event.upload_type).delete_all log_event(job_id)
end end
# rubocop: enable CodeReuse/ActiveRecord
private def log_event(job_id)
def log_event
logger.event_info( logger.event_info(
created_at, created_at,
'Deleted upload file', 'Delete upload file scheduled',
upload_id: event.upload_id, upload_id: event.upload_id,
upload_type: event.upload_type, upload_type: event.upload_type,
file_path: event.file_path, file_path: event.file_path,
model_id: event.model_id, model_id: event.model_id,
model_type: event.model_type) model_type: event.model_type,
job_id: job_id)
end end
end end
end end
......
...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, : ...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
subject { described_class.new(job_artifact_deleted_event, Time.now, logger) } subject { described_class.new(job_artifact_deleted_event, Time.now, logger) }
around do |example| around do |example|
Sidekiq::Testing.fake! { example.run } Sidekiq::Testing.inline! { example.run }
end end
describe '#process' do describe '#process' do
...@@ -32,11 +32,14 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, : ...@@ -32,11 +32,14 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
context 'when the delete fails' do context 'when the delete fails' do
before do before do
expect(File).to receive(:delete).with(job_artifact.file.path).and_raise("Cannot delete") allow(File).to receive(:unlink).and_call_original
allow(File).to receive(:unlink).with(job_artifact.file.path).and_raise(SystemCallError, "Cannot delete")
end end
it 'does not remove the tracking database entry' do it 'does not remove the tracking database entry' do
expect { subject.process }.not_to change(Geo::JobArtifactRegistry, :count) expect do
expect { subject.process }.to raise_error(SystemCallError)
end.not_to change(Geo::JobArtifactRegistry, :count)
end end
end end
end end
......
...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl ...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
subject { described_class.new(lfs_object_deleted_event, Time.now, logger) } subject { described_class.new(lfs_object_deleted_event, Time.now, logger) }
around do |example| around do |example|
Sidekiq::Testing.fake! { example.run } Sidekiq::Testing.inline! { example.run }
end end
describe '#process' do describe '#process' do
...@@ -24,10 +24,8 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl ...@@ -24,10 +24,8 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
expect { subject.process }.to change(Geo::FileRegistry.lfs_objects, :count).by(-1) expect { subject.process }.to change(Geo::FileRegistry.lfs_objects, :count).by(-1)
end end
it 'schedules a Geo::FileRemovalWorker job' do it 'schedules a Geo::FileRegistryRemovalWorker job' do
file_path = File.join(LfsObjectUploader.root, lfs_object_deleted_event.file_path) expect(::Geo::FileRegistryRemovalWorker).to receive(:perform_async).with(:lfs, lfs_object_deleted_event.lfs_object_id)
expect(::Geo::FileRemovalWorker).to receive(:perform_async).with(file_path)
subject.process subject.process
end end
......
...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::UploadDeletedEvent, :postgresql, :clean ...@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::UploadDeletedEvent, :postgresql, :clean
subject { described_class.new(upload_deleted_event, Time.now, logger) } subject { described_class.new(upload_deleted_event, Time.now, logger) }
around do |example| around do |example|
Sidekiq::Testing.fake! { example.run } Sidekiq::Testing.inline! { example.run }
end end
describe '#process' do describe '#process' do
......
require 'spec_helper' require 'spec_helper'
describe Appearance do describe Appearance do
include ::EE::GeoHelpers
subject { build(:appearance) } subject { build(:appearance) }
describe 'validations' do describe 'validations' do
...@@ -17,4 +19,42 @@ describe Appearance do ...@@ -17,4 +19,42 @@ describe Appearance do
it { is_expected.to allow_value(hex).for(:message_font_color) } it { is_expected.to allow_value(hex).for(:message_font_color) }
it { is_expected.not_to allow_value('000').for(:message_font_color) } it { is_expected.not_to allow_value('000').for(:message_font_color) }
end end
context 'object storage with background upload' do
context 'when running in a Geo primary node' do
set(:primary) { create(:geo_node, :primary) }
set(:secondary) { create(:geo_node) }
before do
stub_current_geo_node(primary)
stub_uploads_object_storage(AttachmentUploader, background_upload: true)
end
it 'creates a Geo deleted log event for logo' do
Sidekiq::Testing.inline! do
expect do
create(:appearance, :with_logo)
end.to change(Geo::UploadDeletedEvent, :count).by(1)
end
end
it 'creates a Geo deleted log event for header logo' do
Sidekiq::Testing.inline! do
expect do
create(:appearance, :with_header_logo)
end.to change(Geo::UploadDeletedEvent, :count).by(1)
end
end
it 'creates only a Geo deleted log event for the migrated header logo' do
Sidekiq::Testing.inline! do
appearance = create(:appearance, :with_header_logo, :with_logo)
expect do
appearance.update(header_logo: fixture_file_upload('spec/fixtures/rails_sample.jpg'))
end.to change(Geo::UploadDeletedEvent, :count).by(1)
end
end
end
end
end end
require 'spec_helper'
describe Note do
include ::EE::GeoHelpers
context 'object storage with background upload' do
before do
stub_uploads_object_storage(AttachmentUploader, background_upload: true)
end
context 'when running in a Geo primary node' do
set(:primary) { create(:geo_node, :primary) }
set(:secondary) { create(:geo_node) }
before do
stub_current_geo_node(primary)
end
it 'creates a Geo deleted log event for attachment' do
Sidekiq::Testing.inline! do
expect do
create(:note, :with_attachment)
end.to change(Geo::UploadDeletedEvent, :count).by(1)
end
end
end
end
end
...@@ -129,7 +129,7 @@ describe Geo::FileRegistryRemovalService do ...@@ -129,7 +129,7 @@ describe Geo::FileRegistryRemovalService do
end end
end end
context 'with file' do # TODO context 'with file' do
let!(:upload) { create(:user, :with_avatar).avatar.upload } let!(:upload) { create(:user, :with_avatar).avatar.upload }
let!(:file_registry) { create(:geo_file_registry, :avatar, file_id: upload.id) } let!(:file_registry) { create(:geo_file_registry, :avatar, file_id: upload.id) }
let!(:file_path) { upload.build_uploader.file.path } let!(:file_path) { upload.build_uploader.file.path }
......
...@@ -18,7 +18,7 @@ describe Geo::HashedStorageAttachmentsEventStore do ...@@ -18,7 +18,7 @@ describe Geo::HashedStorageAttachmentsEventStore do
TestEnv.clean_test_path TestEnv.clean_test_path
end end
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::HashedStorageAttachmentsEvent it_behaves_like 'a Geo event store', Geo::HashedStorageAttachmentsEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -17,7 +17,7 @@ describe Geo::HashedStorageMigratedEventStore do ...@@ -17,7 +17,7 @@ describe Geo::HashedStorageMigratedEventStore do
TestEnv.clean_test_path TestEnv.clean_test_path
end end
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::HashedStorageMigratedEvent it_behaves_like 'a Geo event store', Geo::HashedStorageMigratedEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -11,21 +11,17 @@ describe Geo::JobArtifactDeletedEventStore do ...@@ -11,21 +11,17 @@ describe Geo::JobArtifactDeletedEventStore do
subject { described_class.new(job_artifact) } subject { described_class.new(job_artifact) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::JobArtifactDeletedEvent it_behaves_like 'a Geo event store', Geo::JobArtifactDeletedEvent do
let(:file_subject) { job_artifact }
end
context 'when running on a primary node' do context 'when running on a primary node' do
before do before do
stub_primary_node stub_primary_node
end end
it 'does not create an event when LFS object is not on a local store' do it 'tracks artifact attributes' do
allow(job_artifact).to receive(:local_store?).and_return(false)
expect { subject.create! }.not_to change(Geo::JobArtifactDeletedEvent, :count)
end
it 'tracks LFS object attributes' do
subject.create! subject.create!
expect(Geo::JobArtifactDeletedEvent.last).to have_attributes( expect(Geo::JobArtifactDeletedEvent.last).to have_attributes(
......
...@@ -11,20 +11,16 @@ describe Geo::LfsObjectDeletedEventStore do ...@@ -11,20 +11,16 @@ describe Geo::LfsObjectDeletedEventStore do
subject { described_class.new(lfs_object) } subject { described_class.new(lfs_object) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::LfsObjectDeletedEvent it_behaves_like 'a Geo event store', Geo::LfsObjectDeletedEvent do
let(:file_subject) { lfs_object }
end
context 'when running on a primary node' do context 'when running on a primary node' do
before do before do
stub_primary_node stub_primary_node
end end
it 'does not create an event when LFS object is not on a local store' do
allow(lfs_object).to receive(:local_store?).and_return(false)
expect { subject.create! }.not_to change(Geo::LfsObjectDeletedEvent, :count)
end
it 'tracks LFS object attributes' do it 'tracks LFS object attributes' do
subject.create! subject.create!
......
...@@ -9,7 +9,7 @@ describe Geo::RepositoriesChangedEventStore do ...@@ -9,7 +9,7 @@ describe Geo::RepositoriesChangedEventStore do
subject { described_class.new(geo_node) } subject { described_class.new(geo_node) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::RepositoriesChangedEvent it_behaves_like 'a Geo event store', Geo::RepositoriesChangedEvent
end end
end end
...@@ -10,7 +10,7 @@ describe Geo::RepositoryCreatedEventStore do ...@@ -10,7 +10,7 @@ describe Geo::RepositoryCreatedEventStore do
subject { described_class.new(project) } subject { described_class.new(project) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::RepositoryCreatedEvent it_behaves_like 'a Geo event store', Geo::RepositoryCreatedEvent
context 'running on a primary node' do context 'running on a primary node' do
......
...@@ -16,7 +16,7 @@ describe Geo::RepositoryDeletedEventStore do ...@@ -16,7 +16,7 @@ describe Geo::RepositoryDeletedEventStore do
subject { described_class.new(project, repo_path: repo_path, wiki_path: wiki_path) } subject { described_class.new(project, repo_path: repo_path, wiki_path: wiki_path) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::RepositoryDeletedEvent it_behaves_like 'a Geo event store', Geo::RepositoryDeletedEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -13,7 +13,7 @@ describe Geo::RepositoryRenamedEventStore do ...@@ -13,7 +13,7 @@ describe Geo::RepositoryRenamedEventStore do
subject { described_class.new(project, old_path: old_path, old_path_with_namespace: old_path_with_namespace) } subject { described_class.new(project, old_path: old_path, old_path_with_namespace: old_path_with_namespace) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::RepositoryRenamedEvent it_behaves_like 'a Geo event store', Geo::RepositoryRenamedEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -20,7 +20,7 @@ describe Geo::RepositoryUpdatedEventStore do ...@@ -20,7 +20,7 @@ describe Geo::RepositoryUpdatedEventStore do
subject { described_class.new(project, refs: refs, changes: changes) } subject { described_class.new(project, refs: refs, changes: changes) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::RepositoryUpdatedEvent it_behaves_like 'a Geo event store', Geo::RepositoryUpdatedEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -10,7 +10,7 @@ describe Geo::ResetChecksumEventStore do ...@@ -10,7 +10,7 @@ describe Geo::ResetChecksumEventStore do
subject { described_class.new(project) } subject { described_class.new(project) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::ResetChecksumEvent it_behaves_like 'a Geo event store', Geo::ResetChecksumEvent
context 'when running on a primary node' do context 'when running on a primary node' do
......
...@@ -11,20 +11,16 @@ describe Geo::UploadDeletedEventStore do ...@@ -11,20 +11,16 @@ describe Geo::UploadDeletedEventStore do
subject { described_class.new(upload) } subject { described_class.new(upload) }
describe '#create' do describe '#create!' do
it_behaves_like 'a Geo event store', Geo::UploadDeletedEvent it_behaves_like 'a Geo event store', Geo::UploadDeletedEvent do
let(:file_subject) { upload }
end
context 'when running on a primary node' do context 'when running on a primary node' do
before do before do
stub_primary_node stub_primary_node
end end
it 'does not create an event when the upload does not use local storage' do
allow(upload).to receive(:local?).and_return(false)
expect { subject.create! }.not_to change(Geo::UploadDeletedEvent, :count)
end
it 'tracks upload attributes' do it 'tracks upload attributes' do
subject.create! subject.create!
......
...@@ -25,5 +25,17 @@ shared_examples_for 'a Geo event store' do |event_class| ...@@ -25,5 +25,17 @@ shared_examples_for 'a Geo event store' do |event_class|
it 'creates an event' do it 'creates an event' do
expect { subject.create! }.to change(event_class, :count).by(1) expect { subject.create! }.to change(event_class, :count).by(1)
end end
context 'when file subject is not on local store' do
before do
skip 'No file subject defined, skipping' unless defined?(file_subject)
allow(file_subject).to receive(:local?).and_return(false)
end
it 'creates an event' do
expect { subject.create! }.to change(event_class, :count).by(1)
end
end
end end
end end
...@@ -33,6 +33,14 @@ shared_examples "builds correct paths" do |**patterns| ...@@ -33,6 +33,14 @@ shared_examples "builds correct paths" do |**patterns|
it_behaves_like "matches the method pattern", :upload_path it_behaves_like "matches the method pattern", :upload_path
end end
describe "#relative_path" do
it 'is relative' do
skip 'Path not set, skipping.' unless subject.path
expect(Pathname.new(subject.relative_path)).to be_relative
end
end
describe ".absolute_path" do describe ".absolute_path" do
it_behaves_like "matches the method pattern", :absolute_path do it_behaves_like "matches the method pattern", :absolute_path do
let(:target) { subject.class } let(:target) { subject.class }
......
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