2019-12-21 20:55:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-06-16 23:09:34 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
require 'rake'
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
RSpec.describe 'gitlab:db namespace rake task', :silence_stdout do
|
2016-06-16 23:09:34 +05:30
|
|
|
before :all do
|
|
|
|
Rake.application.rake_require 'active_record/railties/databases'
|
|
|
|
Rake.application.rake_require 'tasks/seed_fu'
|
|
|
|
Rake.application.rake_require 'tasks/gitlab/db'
|
|
|
|
|
|
|
|
# empty task as env is already loaded
|
|
|
|
Rake::Task.define_task :environment
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
# Stub out db tasks
|
|
|
|
allow(Rake::Task['db:migrate']).to receive(:invoke).and_return(true)
|
2020-04-22 19:07:51 +05:30
|
|
|
allow(Rake::Task['db:structure:load']).to receive(:invoke).and_return(true)
|
2016-06-16 23:09:34 +05:30
|
|
|
allow(Rake::Task['db:seed_fu']).to receive(:invoke).and_return(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'configure' do
|
2016-09-13 17:45:13 +05:30
|
|
|
it 'invokes db:migrate when schema has already been loaded' do
|
2018-11-08 19:23:39 +05:30
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return(%w[table1 table2])
|
2016-06-16 23:09:34 +05:30
|
|
|
expect(Rake::Task['db:migrate']).to receive(:invoke)
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).not_to receive(:invoke)
|
2016-06-16 23:09:34 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.not_to raise_error
|
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
it 'invokes db:shema:load and db:seed_fu when schema is not loaded' do
|
2016-06-16 23:09:34 +05:30
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return([])
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).to receive(:invoke)
|
2016-06-16 23:09:34 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).to receive(:invoke)
|
|
|
|
expect(Rake::Task['db:migrate']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.not_to raise_error
|
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'invokes db:shema:load and db:seed_fu when there is only a single table present' do
|
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return(['default'])
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).to receive(:invoke)
|
2018-11-08 19:23:39 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).to receive(:invoke)
|
|
|
|
expect(Rake::Task['db:migrate']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.not_to raise_error
|
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
it 'does not invoke any other rake tasks during an error' do
|
2016-06-16 23:09:34 +05:30
|
|
|
allow(ActiveRecord::Base).to receive(:connection).and_raise(RuntimeError, 'error')
|
|
|
|
expect(Rake::Task['db:migrate']).not_to receive(:invoke)
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).not_to receive(:invoke)
|
2016-06-16 23:09:34 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.to raise_error(RuntimeError, 'error')
|
|
|
|
# unstub connection so that the database cleaner still works
|
|
|
|
allow(ActiveRecord::Base).to receive(:connection).and_call_original
|
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
it 'does not invoke seed after a failed schema_load' do
|
2016-06-16 23:09:34 +05:30
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return([])
|
2020-04-22 19:07:51 +05:30
|
|
|
allow(Rake::Task['db:structure:load']).to receive(:invoke).and_raise(RuntimeError, 'error')
|
|
|
|
expect(Rake::Task['db:structure:load']).to receive(:invoke)
|
2016-06-16 23:09:34 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).not_to receive(:invoke)
|
|
|
|
expect(Rake::Task['db:migrate']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.to raise_error(RuntimeError, 'error')
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
|
|
|
|
context 'SKIP_POST_DEPLOYMENT_MIGRATIONS environment variable set' do
|
|
|
|
let(:rails_paths) { { 'db' => ['db'], 'db/migrate' => ['db/migrate'] } }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(ENV).to receive(:[]).and_call_original
|
|
|
|
allow(ENV).to receive(:[]).with('SKIP_POST_DEPLOYMENT_MIGRATIONS').and_return true
|
|
|
|
|
|
|
|
# Our environment has already been loaded, so we need to pretend like post_migrations were not
|
|
|
|
allow(Rails.application.config).to receive(:paths).and_return(rails_paths)
|
|
|
|
allow(ActiveRecord::Migrator).to receive(:migrations_paths).and_return(rails_paths['db/migrate'].dup)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'adds post deployment migrations before schema load if the schema is not already loaded' do
|
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return([])
|
|
|
|
expect(Gitlab::Database).to receive(:add_post_migrate_path_to_rails).and_call_original
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).to receive(:invoke)
|
2018-12-05 23:21:45 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).to receive(:invoke)
|
|
|
|
expect(Rake::Task['db:migrate']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.not_to raise_error
|
|
|
|
expect(rails_paths['db/migrate'].include?(File.join(Rails.root, 'db', 'post_migrate'))).to be(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'ignores post deployment migrations when schema has already been loaded' do
|
|
|
|
allow(ActiveRecord::Base.connection).to receive(:tables).and_return(%w[table1 table2])
|
|
|
|
expect(Rake::Task['db:migrate']).to receive(:invoke)
|
|
|
|
expect(Gitlab::Database).not_to receive(:add_post_migrate_path_to_rails)
|
2020-04-22 19:07:51 +05:30
|
|
|
expect(Rake::Task['db:structure:load']).not_to receive(:invoke)
|
2018-12-05 23:21:45 +05:30
|
|
|
expect(Rake::Task['db:seed_fu']).not_to receive(:invoke)
|
|
|
|
expect { run_rake_task('gitlab:db:configure') }.not_to raise_error
|
|
|
|
expect(rails_paths['db/migrate'].include?(File.join(Rails.root, 'db', 'post_migrate'))).to be(false)
|
|
|
|
end
|
|
|
|
end
|
2016-06-16 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
describe 'unattended' do
|
|
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
|
|
|
|
where(:schema_migration_table_exists, :needs_migrations, :rake_output) do
|
|
|
|
false | false | "unattended_migrations_completed"
|
|
|
|
false | true | "unattended_migrations_completed"
|
|
|
|
true | false | "unattended_migrations_static"
|
|
|
|
true | true | "unattended_migrations_completed"
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Rake::Task['gitlab:db:configure']).to receive(:invoke).and_return(true)
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
it 'outputs changed message for automation after operations happen' do
|
|
|
|
allow(ActiveRecord::Base.connection.schema_migration).to receive(:table_exists?).and_return(schema_migration_table_exists)
|
|
|
|
allow_any_instance_of(ActiveRecord::MigrationContext).to receive(:needs_migration?).and_return(needs_migrations)
|
|
|
|
expect { run_rake_task('gitlab:db:unattended') }. to output(/^#{rake_output}$/).to_stdout
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
describe 'clean_structure_sql' do
|
|
|
|
let_it_be(:clean_rake_task) { 'gitlab:db:clean_structure_sql' }
|
|
|
|
let_it_be(:test_task_name) { 'gitlab:db:_test_multiple_structure_cleans' }
|
|
|
|
let_it_be(:input) { 'this is structure data' }
|
2021-09-30 23:02:18 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
let(:output) { StringIO.new }
|
|
|
|
|
|
|
|
before do
|
2021-11-11 11:23:49 +05:30
|
|
|
structure_files = %w[structure.sql ci_structure.sql]
|
2020-04-22 19:07:51 +05:30
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
allow(File).to receive(:open).and_call_original
|
2020-04-22 19:07:51 +05:30
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
structure_files.each do |structure_file_name|
|
|
|
|
structure_file = File.join(ActiveRecord::Tasks::DatabaseTasks.db_dir, structure_file_name)
|
2021-09-30 23:02:18 +05:30
|
|
|
stub_file_read(structure_file, content: input)
|
2021-11-11 11:23:49 +05:30
|
|
|
allow(File).to receive(:open).with(structure_file.to_s, any_args).and_yield(output)
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
Rake::Task[test_task_name].clear if Rake::Task.task_defined?(test_task_name)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'can be executed multiple times within another rake task' do
|
2021-09-30 23:02:18 +05:30
|
|
|
expect_multiple_executions_of_task(test_task_name, clean_rake_task, count: 2) do
|
|
|
|
database_count = ActiveRecord::Base.configurations.configs_for(env_name: Rails.env).size
|
|
|
|
|
|
|
|
expect_next_instances_of(Gitlab::Database::SchemaCleaner, database_count) do |cleaner|
|
|
|
|
expect(cleaner).to receive(:clean).with(output)
|
2020-04-22 19:07:51 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
describe 'drop_tables' do
|
|
|
|
subject { run_rake_task('gitlab:db:drop_tables') }
|
|
|
|
|
|
|
|
let(:tables) { %w(one two) }
|
|
|
|
let(:views) { %w(three four) }
|
|
|
|
let(:connection) { ActiveRecord::Base.connection }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(connection).to receive(:execute).and_return(nil)
|
|
|
|
|
|
|
|
allow(connection).to receive(:tables).and_return(tables)
|
|
|
|
allow(connection).to receive(:views).and_return(views)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'drops all tables, except schema_migrations' do
|
|
|
|
expect(connection).to receive(:execute).with('DROP TABLE IF EXISTS "one" CASCADE')
|
|
|
|
expect(connection).to receive(:execute).with('DROP TABLE IF EXISTS "two" CASCADE')
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'drops all views' do
|
|
|
|
expect(connection).to receive(:execute).with('DROP VIEW IF EXISTS "three" CASCADE')
|
|
|
|
expect(connection).to receive(:execute).with('DROP VIEW IF EXISTS "four" CASCADE')
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'truncates schema_migrations table' do
|
|
|
|
expect(connection).to receive(:execute).with('TRUNCATE schema_migrations')
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'drops extra schemas' do
|
|
|
|
Gitlab::Database::EXTRA_SCHEMAS.each do |schema|
|
|
|
|
expect(connection).to receive(:execute).with("DROP SCHEMA IF EXISTS \"#{schema}\"")
|
|
|
|
end
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
describe 'reindex' do
|
2021-01-03 14:25:43 +05:30
|
|
|
let(:reindex) { double('reindex') }
|
|
|
|
let(:indexes) { double('indexes') }
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it 'cleans up any leftover indexes' do
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:cleanup_leftovers!)
|
|
|
|
|
|
|
|
run_rake_task('gitlab:db:reindex')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when async index creation is enabled' do
|
|
|
|
it 'executes async index creation prior to any reindexing actions' do
|
|
|
|
stub_feature_flags(database_async_index_creation: true)
|
|
|
|
|
|
|
|
expect(Gitlab::Database::AsyncIndexes).to receive(:create_pending_indexes!).ordered
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform).ordered
|
|
|
|
|
|
|
|
run_rake_task('gitlab:db:reindex')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when async index creation is disabled' do
|
|
|
|
it 'does not execute async index creation' do
|
|
|
|
stub_feature_flags(database_async_index_creation: false)
|
|
|
|
|
|
|
|
expect(Gitlab::Database::AsyncIndexes).not_to receive(:create_pending_indexes!)
|
|
|
|
|
|
|
|
run_rake_task('gitlab:db:reindex')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
context 'when no index_name is given' do
|
2021-02-22 17:27:13 +05:30
|
|
|
it 'uses all candidate indexes' do
|
2021-10-27 15:23:28 +05:30
|
|
|
expect(Gitlab::Database::PostgresIndex).to receive(:reindexing_support).and_return(indexes)
|
2021-01-03 14:25:43 +05:30
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform).with(indexes)
|
|
|
|
|
|
|
|
run_rake_task('gitlab:db:reindex')
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
context 'with index name given' do
|
|
|
|
let(:index) { double('index') }
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
before do
|
2021-10-27 15:23:28 +05:30
|
|
|
allow(Gitlab::Database::PostgresIndex).to receive(:reindexing_support).and_return(indexes)
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
it 'calls the index rebuilder with the proper arguments' do
|
2021-02-22 17:27:13 +05:30
|
|
|
allow(indexes).to receive(:where).with(identifier: 'public.foo_idx').and_return([index])
|
2021-01-03 14:25:43 +05:30
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform).with([index])
|
2020-11-24 15:15:51 +05:30
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
run_rake_task('gitlab:db:reindex', '[public.foo_idx]')
|
|
|
|
end
|
2020-11-24 15:15:51 +05:30
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
it 'raises an error if the index does not exist' do
|
2021-02-22 17:27:13 +05:30
|
|
|
allow(indexes).to receive(:where).with(identifier: 'public.absent_index').and_return([])
|
|
|
|
|
|
|
|
expect { run_rake_task('gitlab:db:reindex', '[public.absent_index]') }.to raise_error(/Index not found/)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises an error if the index is not fully qualified with a schema' do
|
|
|
|
expect { run_rake_task('gitlab:db:reindex', '[foo_idx]') }.to raise_error(/Index name is not fully qualified/)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'active' do
|
|
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
|
|
|
|
let(:task) { 'gitlab:db:active' }
|
|
|
|
let(:self_monitoring) { double('self_monitoring') }
|
2020-11-24 15:15:51 +05:30
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
where(:needs_migration, :self_monitoring_project, :project_count, :exit_status, :exit_code) do
|
|
|
|
true | nil | nil | 1 | false
|
|
|
|
false | :self_monitoring | 1 | 1 | false
|
|
|
|
false | nil | 0 | 1 | false
|
|
|
|
false | :self_monitoring | 2 | 0 | true
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
it 'exits 0 or 1 depending on user modifications to the database' do
|
|
|
|
allow_any_instance_of(ActiveRecord::MigrationContext).to receive(:needs_migration?).and_return(needs_migration)
|
|
|
|
allow_any_instance_of(ApplicationSetting).to receive(:self_monitoring_project).and_return(self_monitoring_project)
|
|
|
|
allow(Project).to receive(:count).and_return(project_count)
|
|
|
|
|
|
|
|
expect { run_rake_task(task) }.to raise_error do |error|
|
|
|
|
expect(error).to be_a(SystemExit)
|
|
|
|
expect(error.status).to eq(exit_status)
|
|
|
|
expect(error.success?).to be(exit_code)
|
|
|
|
end
|
2021-01-03 14:25:43 +05:30
|
|
|
end
|
2020-11-24 15:15:51 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-03-11 19:13:27 +05:30
|
|
|
describe '#migrate_with_instrumentation' do
|
2021-11-18 22:05:49 +05:30
|
|
|
describe '#up' do
|
|
|
|
subject { run_rake_task('gitlab:db:migration_testing:up') }
|
2021-03-11 19:13:27 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
it 'delegates to the migration runner' do
|
|
|
|
expect(::Gitlab::Database::Migrations::Runner).to receive_message_chain(:up, :run)
|
2021-03-11 19:13:27 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
subject
|
|
|
|
end
|
2021-03-11 19:13:27 +05:30
|
|
|
end
|
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
describe '#down' do
|
|
|
|
subject { run_rake_task('gitlab:db:migration_testing:down') }
|
2021-03-11 19:13:27 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
it 'delegates to the migration runner' do
|
|
|
|
expect(::Gitlab::Database::Migrations::Runner).to receive_message_chain(:down, :run)
|
2021-03-11 19:13:27 +05:30
|
|
|
|
2021-11-18 22:05:49 +05:30
|
|
|
subject
|
|
|
|
end
|
2021-06-08 01:23:25 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#execute_batched_migrations' do
|
|
|
|
subject { run_rake_task('gitlab:db:execute_batched_migrations') }
|
|
|
|
|
|
|
|
let(:migrations) { create_list(:batched_background_migration, 2) }
|
|
|
|
let(:runner) { instance_double('Gitlab::Database::BackgroundMigration::BatchedMigrationRunner') }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab::Database::BackgroundMigration::BatchedMigration).to receive_message_chain(:active, :queue_order).and_return(migrations)
|
|
|
|
allow(Gitlab::Database::BackgroundMigration::BatchedMigrationRunner).to receive(:new).and_return(runner)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'executes all migrations' do
|
|
|
|
migrations.each do |migration|
|
|
|
|
expect(runner).to receive(:run_entire_migration).with(migration)
|
|
|
|
end
|
|
|
|
|
|
|
|
subject
|
2021-03-11 19:13:27 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
def run_rake_task(task_name, arguments = '')
|
2016-06-16 23:09:34 +05:30
|
|
|
Rake::Task[task_name].reenable
|
2020-11-24 15:15:51 +05:30
|
|
|
Rake.application.invoke_task("#{task_name}#{arguments}")
|
2016-06-16 23:09:34 +05:30
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
|
|
|
|
def expect_multiple_executions_of_task(test_task_name, task_to_invoke, count: 2)
|
|
|
|
Rake::Task.define_task(test_task_name => :environment) do
|
|
|
|
count.times do
|
|
|
|
yield
|
|
|
|
|
|
|
|
Rake::Task[task_to_invoke].invoke
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
run_rake_task(test_task_name)
|
|
|
|
end
|
2016-06-16 23:09:34 +05:30
|
|
|
end
|