Commit 86bde936 authored by Micaël Bergeron's avatar Micaël Bergeron

refactored the specs to the background migration

parent 89012de6
# frozen_string_literal: true
# rubocop:disable Metrics/LineLength
# rubocop:disable Style/Documentation
module Gitlab module Gitlab
module BackgroundMigration module BackgroundMigration
class MigrateUploadsToObjectStorage class MigrateUploadsToObjectStorage
...@@ -103,8 +107,8 @@ module Gitlab ...@@ -103,8 +107,8 @@ module Gitlab
def report(results) def report(results)
success, failures = Gitlab::Utils::BisectEnumerable.bisect(results, &:success?) success, failures = Gitlab::Utils::BisectEnumerable.bisect(results, &:success?)
puts header(success, failures) Rails.logger.info header(success, failures)
puts failures(failures) Rails.logger.warn failures(failures)
end end
def header(success, failures) def header(success, failures)
...@@ -124,7 +128,6 @@ module Gitlab ...@@ -124,7 +128,6 @@ module Gitlab
sanity_check!(uploads, mounted_as) sanity_check!(uploads, mounted_as)
BackgroundMigrationWorker.perform_async('MigrateUploadsToObjectStorage', [uploads.ids, mounted_as, to_store]) BackgroundMigrationWorker.perform_async('MigrateUploadsToObjectStorage', [uploads.ids, mounted_as, to_store])
#BackgroundMigration.perform('MigrateUploadsToObjectStorage', [uploads.map(&:id), mounted_as, to_store])
end end
# We need to be sure all the uploads are for the same uploader and model type # We need to be sure all the uploads are for the same uploader and model type
...@@ -146,7 +149,7 @@ module Gitlab ...@@ -146,7 +149,7 @@ module Gitlab
end end
def perform(ids, mounted_as, to_store) def perform(ids, mounted_as, to_store)
@mounted_as = mounted_as @mounted_as = mounted_as.to_sym
@to_store = to_store @to_store = to_store
uploads = Upload.preload(:model).where(id: ids) uploads = Upload.preload(:model).where(id: ids)
...@@ -155,9 +158,9 @@ module Gitlab ...@@ -155,9 +158,9 @@ module Gitlab
results = migrate(uploads) results = migrate(uploads)
report(results) report(results)
rescue SanityCheckError rescue SanityCheckError => e
# do not retry if the job is insane # do not retry if the job is insane
logger.warn "UploadsToObjectStorage sanity check error: #{e.message}" Rails.logger.warn "UploadsToObjectStorage sanity check error: #{e.message}"
end end
def sanity_check!(uploads) def sanity_check!(uploads)
...@@ -168,16 +171,6 @@ module Gitlab ...@@ -168,16 +171,6 @@ module Gitlab
uploads.map { |upload| upload.build_uploader(@mounted_as) } uploads.map { |upload| upload.build_uploader(@mounted_as) }
end end
def migrate(batch_size, &block)
each_upload_batch(batch_size) do |batch|
results = build_uploaders(batch).map(&method(:process_uploader))
yield results # yield processed batch as [MigrationResult]
@results.concat(results)
end
end
def migrate(uploads) def migrate(uploads)
build_uploaders(uploads).map(&method(:process_uploader)) build_uploaders(uploads).map(&method(:process_uploader))
end end
......
require_relative 'helpers' require_relative 'helpers'
include UploadTaskHelpers
namespace :gitlab do namespace :gitlab do
namespace :uploads do namespace :uploads do
desc 'GitLab | Uploads | Migrate the uploaded files to object storage' desc 'GitLab | Uploads | Migrate the uploaded files to object storage'
task :migrate, [:uploader_class, :model_class, :mounted_as] => :environment do |task, args| task :migrate, [:uploader_class, :model_class, :mounted_as] => :environment do |task, args|
to_store = ObjectStorage::Store::REMOTE include UploadTaskHelpers
@to_store = ObjectStorage::Store::REMOTE
@mounted_as = args.mounted_as&.gsub(':', '')&.to_sym
uploader_class = args.uploader_class.constantize uploader_class = args.uploader_class.constantize
model_class = args.model_class.constantize model_class = args.model_class.constantize
mounted_as = args.mounted_as&.gsub(':', '')&.to_sym
Upload Upload
.where.not(store: to_store) .where.not(store: @to_store)
.where(uploader: uploader_class.to_s, .where(uploader: uploader_class.to_s,
model_type: model_class.to_s) model_type: model_class.to_s)
.in_batches(of: batch_size) do |batch| # rubocop: disable Cop/InBatches .in_batches(of: batch_size, &method(:process)) # rubocop: disable Cop/InBatches
job = Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage.enqueue!(batch, mounted_as, to_store)
puts "Enqueued job: #{job}"
end end
def process(batch)
job = Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage.enqueue!(batch,
@mounted_as,
@to_store)
puts "Enqueued job: #{job}"
rescue Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage::SanityCheckError => e
# continue for the next batch
puts "Could not enqueue batch (#{batch.ids}) #{e.message}".color(:red)
end end
end end
end end
...@@ -7,15 +7,13 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do ...@@ -7,15 +7,13 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do
end end
end end
let!(:projects) { create_list(:project, 10, :with_avatar) }
let(:uploads) { Upload.all } let(:uploads) { Upload.all }
let(:mounted_as) { :avatar } let(:mounted_as) { :avatar }
let(:to_store) { ObjectStorage::Store::REMOTE } let(:to_store) { ObjectStorage::Store::REMOTE }
before do before do
stub_env('BATCH', 1) stub_uploads_object_storage(AvatarUploader)
stub_licensed_features(object_storage: true)
create_list(:upload, 5)
end end
describe '.enqueue!' do describe '.enqueue!' do
...@@ -24,6 +22,7 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do ...@@ -24,6 +22,7 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do
end end
it 'is guarded by .sanity_check!' do it 'is guarded by .sanity_check!' do
expect(BackgroundMigrationWorker).to receive(:perform_async)
expect(described_class).to receive(:sanity_check!) expect(described_class).to receive(:sanity_check!)
enqueue! enqueue!
...@@ -67,4 +66,48 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do ...@@ -67,4 +66,48 @@ describe Gitlab::BackgroundMigration::MigrateUploadsToObjectStorage, :sidekiq do
end end
end end
end end
describe '#perform' do
def perform
described_class.enqueue!(uploads, mounted_as, to_store)
end
# rubocop:disable Style/MultilineIfModifier
shared_examples 'outputs correctly' do |success: 0, failures: 0|
total = success + failures
it 'outputs the reports' do
expect(Rails.logger).to receive(:info).with(%r{Migrated #{success}/#{total} files})
perform
end if success > 0
it 'outputs upload failures' do
expect(Rails.logger).to receive(:warn).with(/Error .* I am a teapot/)
perform
end if failures > 0
end
# rubocop:enable Style/MultilineIfModifier
it_behaves_like 'outputs correctly', success: 10
it 'migrates files' do
perform
aggregate_failures do
projects.each do |project|
expect(project.reload.avatar.upload.local?).to be_falsey
end
end
end
context 'migration is unsuccessful' do
before do
allow_any_instance_of(ObjectStorage::Concern).to receive(:migrate!).and_raise(CarrierWave::UploadError, "I am a teapot.")
end
it_behaves_like 'outputs correctly', failures: 10
end
end
end end
...@@ -36,6 +36,8 @@ describe 'gitlab:uploads rake tasks' do ...@@ -36,6 +36,8 @@ describe 'gitlab:uploads rake tasks' do
stub_env('BATCH', batch_size.to_s) stub_env('BATCH', batch_size.to_s)
stub_uploads_object_storage(uploader_class) stub_uploads_object_storage(uploader_class)
Rake.application.rake_require 'tasks/gitlab/uploads/migrate' Rake.application.rake_require 'tasks/gitlab/uploads/migrate'
allow(BackgroundMigrationWorker).to receive(:perform_async)
end end
def run def run
...@@ -43,42 +45,10 @@ describe 'gitlab:uploads rake tasks' do ...@@ -43,42 +45,10 @@ describe 'gitlab:uploads rake tasks' do
run_rake_task("gitlab:uploads:migrate", *args) run_rake_task("gitlab:uploads:migrate", *args)
end end
shared_examples 'outputs correctly' do |success: 0, failures: 0| it 'enqueue jobs in batch' do
total = success + failures expect(BackgroundMigrationWorker).to receive(:perform_async).exactly(4).times
it 'outputs the results for each batch' do
batch_count = [batch_size, total].min
expect { run }.to output(%r{Migrated #{batch_count}/#{batch_count} files}).to_stdout
end if success > 0 # rubocop:disable Style/MultilineIfModifier
it 'outputs the results for the task' do
expect { run }.to output(%r{Migrated #{success}/#{total} files}).to_stdout
end if success > 0 # rubocop:disable Style/MultilineIfModifier
it 'outputs upload failures' do
expect { run }.to output(/Error .* I am a teapot/).to_stdout
end if failures > 0 # rubocop:disable Style/MultilineIfModifier
end
it_behaves_like 'outputs correctly', success: 10
it 'migrates files' do
run run
aggregate_failures do
projects.each do |project|
expect(project.reload.avatar.upload.local?).to be_falsey
end
end
end
context 'migration is unsuccessful' do
before do
allow_any_instance_of(ObjectStorage::Concern).to receive(:migrate!).and_raise(CarrierWave::UploadError, "I am a teapot.")
end
it_behaves_like 'outputs correctly', failures: 10
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