2019-07-25 01:21:37 -04:00
# frozen_string_literal: true
2017-12-07 10:33:30 -05:00
require 'spec_helper'
2020-06-24 05:08:32 -04:00
RSpec . describe Backup :: Repository do
2020-08-12 20:10:06 -04:00
let_it_be ( :project ) { create ( :project , :wiki_repo ) }
2017-12-07 10:33:30 -05:00
let ( :progress ) { StringIO . new }
2019-12-12 07:07:33 -05:00
2018-05-24 10:58:25 -04:00
subject { described_class . new ( progress ) }
2017-12-07 10:33:30 -05:00
before do
allow ( progress ) . to receive ( :puts )
allow ( progress ) . to receive ( :print )
2018-03-05 03:25:02 -05:00
allow ( FileUtils ) . to receive ( :mv ) . and_return ( true )
2017-12-07 10:33:30 -05:00
2019-11-20 01:06:16 -05:00
allow_next_instance_of ( described_class ) do | instance |
allow ( instance ) . to receive ( :progress ) . and_return ( progress )
end
2017-12-07 10:33:30 -05:00
end
2020-08-25 23:10:11 -04:00
describe '#dump' do
2020-08-12 20:10:06 -04:00
before do
allow ( Gitlab . config . repositories . storages ) . to receive ( :keys ) . and_return ( storage_keys )
end
let_it_be ( :projects ) { create_list ( :project , 5 , :wiki_repo ) + [ project ] }
let ( :storage_keys ) { %w[ default test_second_storage ] }
context 'no concurrency' do
it 'creates the expected number of threads' do
expect ( Thread ) . not_to receive ( :new )
projects . each do | project |
expect ( subject ) . to receive ( :dump_project ) . with ( project ) . and_call_original
end
subject . dump ( max_concurrency : 1 , max_storage_concurrency : 1 )
2017-12-07 10:33:30 -05:00
end
2020-08-12 20:10:06 -04:00
describe 'command failure' do
it 'dump_project raises an error' do
allow ( subject ) . to receive ( :dump_project ) . and_raise ( IOError )
expect { subject . dump ( max_concurrency : 1 , max_storage_concurrency : 1 ) } . to raise_error ( IOError )
end
it 'project query raises an error' do
2020-09-04 05:08:38 -04:00
allow ( Project ) . to receive_message_chain ( :includes , :find_each ) . and_raise ( ActiveRecord :: StatementTimeout )
2020-08-12 20:10:06 -04:00
expect { subject . dump ( max_concurrency : 1 , max_storage_concurrency : 1 ) } . to raise_error ( ActiveRecord :: StatementTimeout )
end
end
2020-09-04 05:08:38 -04:00
it 'avoids N+1 database queries' do
control_count = ActiveRecord :: QueryRecorder . new do
subject . dump ( max_concurrency : 1 , max_storage_concurrency : 1 )
end . count
create_list ( :project , 2 , :wiki_repo )
expect do
subject . dump ( max_concurrency : 1 , max_storage_concurrency : 1 )
end . not_to exceed_query_limit ( control_count )
end
2020-08-12 20:10:06 -04:00
end
[ 4 , 10 ] . each do | max_storage_concurrency |
2020-09-04 05:08:38 -04:00
context " max_storage_concurrency #{ max_storage_concurrency } " do
2020-08-12 20:10:06 -04:00
it 'creates the expected number of threads' do
expect ( Thread ) . to receive ( :new )
. exactly ( storage_keys . length * ( max_storage_concurrency + 1 ) ) . times
. and_call_original
projects . each do | project |
expect ( subject ) . to receive ( :dump_project ) . with ( project ) . and_call_original
end
subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency )
end
it 'creates the expected number of threads with extra max concurrency' do
expect ( Thread ) . to receive ( :new )
. exactly ( storage_keys . length * ( max_storage_concurrency + 1 ) ) . times
. and_call_original
projects . each do | project |
expect ( subject ) . to receive ( :dump_project ) . with ( project ) . and_call_original
end
subject . dump ( max_concurrency : 3 , max_storage_concurrency : max_storage_concurrency )
end
describe 'command failure' do
it 'dump_project raises an error' do
allow ( subject ) . to receive ( :dump_project )
. and_raise ( IOError )
expect { subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency ) } . to raise_error ( IOError )
end
it 'project query raises an error' do
2020-09-04 05:08:38 -04:00
allow ( Project ) . to receive_message_chain ( :for_repository_storage , :includes , :find_each ) . and_raise ( ActiveRecord :: StatementTimeout )
2020-08-12 20:10:06 -04:00
expect { subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency ) } . to raise_error ( ActiveRecord :: StatementTimeout )
end
context 'misconfigured storages' do
let ( :storage_keys ) { %w[ test_second_storage ] }
it 'raises an error' do
expect { subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency ) } . to raise_error ( Backup :: Error , 'repositories.storages in gitlab.yml is misconfigured' )
end
end
end
2020-09-04 05:08:38 -04:00
it 'avoids N+1 database queries' do
control_count = ActiveRecord :: QueryRecorder . new do
subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency )
end . count
create_list ( :project , 2 , :wiki_repo )
expect do
subject . dump ( max_concurrency : 1 , max_storage_concurrency : max_storage_concurrency )
end . not_to exceed_query_limit ( control_count )
end
2017-12-07 10:33:30 -05:00
end
end
end
describe '#restore' do
2018-03-02 13:14:19 -05:00
let ( :timestamp ) { Time . utc ( 2017 , 3 , 22 ) }
let ( :temp_dirs ) do
Gitlab . config . repositories . storages . map do | name , storage |
2018-06-05 11:51:14 -04:00
Gitlab :: GitalyClient :: StorageSettings . allow_disk_access do
File . join ( storage . legacy_disk_path , '..' , 'repositories.old.' + timestamp . to_i . to_s )
end
2018-03-02 13:14:19 -05:00
end
end
around do | example |
Timecop . freeze ( timestamp ) { example . run }
end
after do
temp_dirs . each { | path | FileUtils . rm_rf ( path ) }
end
2017-12-07 10:33:30 -05:00
describe 'command failure' do
before do
2020-03-16 08:09:12 -04:00
# Allow us to set expectations on the project directly
expect ( Project ) . to receive ( :find_each ) . and_yield ( project )
expect ( project . repository ) . to receive ( :create_repository ) { raise 'Fail in tests' }
2017-12-07 10:33:30 -05:00
end
2017-12-01 08:58:49 -05:00
context 'hashed storage' do
it 'shows the appropriate error' do
2018-03-02 13:14:19 -05:00
subject . restore
2017-12-07 10:33:30 -05:00
2018-05-16 10:17:05 -04:00
expect ( progress ) . to have_received ( :puts ) . with ( " [Failed] restoring #{ project . full_path } repository " )
2017-12-01 08:58:49 -05:00
end
end
context 'legacy storage' do
let! ( :project ) { create ( :project , :legacy_storage ) }
it 'shows the appropriate error' do
2018-03-02 13:14:19 -05:00
subject . restore
2017-12-01 08:58:49 -05:00
2018-05-16 10:17:05 -04:00
expect ( progress ) . to have_received ( :puts ) . with ( " [Failed] restoring #{ project . full_path } repository " )
2017-12-01 08:58:49 -05:00
end
2017-12-07 10:33:30 -05:00
end
end
2018-12-10 03:31:28 -05:00
context 'restoring object pools' do
2019-10-23 05:06:03 -04:00
it 'schedules restoring of the pool' , :sidekiq_might_not_need_inline do
2018-12-10 03:31:28 -05:00
pool_repository = create ( :pool_repository , :failed )
pool_repository . delete_object_pool
subject . restore
pool_repository . reload
expect ( pool_repository ) . not_to be_failed
expect ( pool_repository . object_pool . exists? ) . to be ( true )
end
end
2020-02-12 19:08:46 -05:00
it 'cleans existing repositories' do
wiki_repository_spy = spy ( :wiki )
allow_next_instance_of ( ProjectWiki ) do | project_wiki |
allow ( project_wiki ) . to receive ( :repository ) . and_return ( wiki_repository_spy )
end
expect_next_instance_of ( Repository ) do | repo |
expect ( repo ) . to receive ( :remove )
end
subject . restore
expect ( wiki_repository_spy ) . to have_received ( :remove )
end
2017-12-07 10:33:30 -05:00
end
describe '#empty_repo?' do
context 'for a wiki' do
let ( :wiki ) { create ( :project_wiki ) }
it 'invalidates the emptiness cache' do
expect ( wiki . repository ) . to receive ( :expire_emptiness_caches ) . once
2018-05-24 10:58:25 -04:00
subject . send ( :empty_repo? , wiki )
2017-12-07 10:33:30 -05:00
end
context 'wiki repo has content' do
let! ( :wiki_page ) { create ( :wiki_page , wiki : wiki ) }
it 'returns true, regardless of bad cache value' do
2018-05-24 10:58:25 -04:00
expect ( subject . send ( :empty_repo? , wiki ) ) . to be ( false )
2017-12-07 10:33:30 -05:00
end
end
context 'wiki repo does not have content' do
it 'returns true, regardless of bad cache value' do
2018-05-24 10:58:25 -04:00
expect ( subject . send ( :empty_repo? , wiki ) ) . to be_truthy
2017-12-07 10:33:30 -05:00
end
end
end
end
end