2021-01-03 14:25:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
RSpec.describe Gitlab::Database::Reindexing do
|
|
|
|
include ExclusiveLeaseHelpers
|
2021-12-11 22:18:48 +05:30
|
|
|
include Database::DatabaseHelpers
|
2021-01-03 14:25:43 +05:30
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
describe '.invoke' do
|
2022-08-13 15:12:31 +05:30
|
|
|
let(:databases) { Gitlab::Database.database_base_models_with_gitlab_shared }
|
2022-01-26 12:08:38 +05:30
|
|
|
let(:databases_count) { databases.count }
|
|
|
|
|
|
|
|
it 'cleans up any leftover indexes' do
|
|
|
|
expect(described_class).to receive(:cleanup_leftovers!).exactly(databases_count).times
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there is an error raised' do
|
|
|
|
it 'logs and re-raise' do
|
|
|
|
expect(described_class).to receive(:automatic_reindexing).and_raise('Unexpected!')
|
|
|
|
expect(Gitlab::AppLogger).to receive(:error)
|
|
|
|
|
|
|
|
expect { described_class.invoke }.to raise_error('Unexpected!')
|
|
|
|
end
|
|
|
|
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.exactly(databases_count).times
|
|
|
|
expect(described_class).to receive(:automatic_reindexing).ordered.exactly(databases_count).times
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
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!)
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-08-27 11:52:29 +05:30
|
|
|
context 'when async index destruction is enabled' do
|
|
|
|
it 'executes async index destruction prior to any reindexing actions' do
|
|
|
|
stub_feature_flags(database_async_index_destruction: true)
|
|
|
|
|
|
|
|
expect(Gitlab::Database::AsyncIndexes).to receive(:drop_pending_indexes!).ordered.exactly(databases_count).times
|
|
|
|
expect(described_class).to receive(:automatic_reindexing).ordered.exactly(databases_count).times
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when async index destruction is disabled' do
|
|
|
|
it 'does not execute async index destruction' do
|
|
|
|
stub_feature_flags(database_async_index_destruction: false)
|
|
|
|
|
|
|
|
expect(Gitlab::Database::AsyncIndexes).not_to receive(:drop_pending_indexes!)
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
context 'calls automatic reindexing' do
|
|
|
|
it 'uses all candidate indexes' do
|
|
|
|
expect(described_class).to receive(:automatic_reindexing).exactly(databases_count).times
|
|
|
|
|
|
|
|
described_class.invoke
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when explicit database is given' do
|
|
|
|
it 'skips other databases' do
|
|
|
|
expect(described_class).to receive(:automatic_reindexing).once
|
|
|
|
|
|
|
|
described_class.invoke(Gitlab::Database::PRIMARY_DATABASE_NAME)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
describe '.automatic_reindexing' do
|
|
|
|
subject { described_class.automatic_reindexing(maximum_records: limit) }
|
2021-01-03 14:25:43 +05:30
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
let(:limit) { 5 }
|
|
|
|
|
|
|
|
before_all do
|
|
|
|
swapout_view_for_table(:postgres_indexes)
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab::Database::Reindexing).to receive(:cleanup_leftovers!)
|
|
|
|
allow(Gitlab::Database::Reindexing).to receive(:perform_from_queue).and_return(0)
|
|
|
|
allow(Gitlab::Database::Reindexing).to receive(:perform_with_heuristic).and_return(0)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'cleans up leftovers, before consuming the queue' do
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:cleanup_leftovers!).ordered
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform_from_queue).ordered
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with records in the queue' do
|
|
|
|
before do
|
|
|
|
create(:reindexing_queued_action)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with enough records in the queue to reach limit' do
|
|
|
|
let(:limit) { 1 }
|
|
|
|
|
|
|
|
it 'does not perform reindexing with heuristic' do
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform_from_queue).and_return(limit)
|
|
|
|
expect(Gitlab::Database::Reindexing).not_to receive(:perform_with_heuristic)
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'without enough records in the queue to reach limit' do
|
|
|
|
let(:limit) { 2 }
|
|
|
|
|
|
|
|
it 'continues if the queue did not have enough records' do
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform_from_queue).ordered.and_return(1)
|
|
|
|
expect(Gitlab::Database::Reindexing).to receive(:perform_with_heuristic).with(maximum_records: 1).ordered
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.perform_with_heuristic' do
|
|
|
|
subject { described_class.perform_with_heuristic(candidate_indexes, maximum_records: limit) }
|
|
|
|
|
|
|
|
let(:limit) { 2 }
|
2021-01-03 14:25:43 +05:30
|
|
|
let(:coordinator) { instance_double(Gitlab::Database::Reindexing::Coordinator) }
|
2021-02-22 17:27:13 +05:30
|
|
|
let(:index_selection) { instance_double(Gitlab::Database::Reindexing::IndexSelection) }
|
|
|
|
let(:candidate_indexes) { double }
|
2021-03-08 18:12:59 +05:30
|
|
|
let(:indexes) { [double, double] }
|
2021-01-03 14:25:43 +05:30
|
|
|
|
|
|
|
it 'delegates to Coordinator' do
|
2021-02-22 17:27:13 +05:30
|
|
|
expect(Gitlab::Database::Reindexing::IndexSelection).to receive(:new).with(candidate_indexes).and_return(index_selection)
|
2021-12-11 22:18:48 +05:30
|
|
|
expect(index_selection).to receive(:take).with(limit).and_return(indexes)
|
2021-03-08 18:12:59 +05:30
|
|
|
|
|
|
|
indexes.each do |index|
|
|
|
|
expect(Gitlab::Database::Reindexing::Coordinator).to receive(:new).with(index).and_return(coordinator)
|
|
|
|
expect(coordinator).to receive(:perform)
|
|
|
|
end
|
2021-01-03 14:25:43 +05:30
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
describe '.perform_from_queue' do
|
|
|
|
subject { described_class.perform_from_queue(maximum_records: limit) }
|
|
|
|
|
|
|
|
before_all do
|
|
|
|
swapout_view_for_table(:postgres_indexes)
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:limit) { 2 }
|
|
|
|
let(:queued_actions) { create_list(:reindexing_queued_action, 3) }
|
|
|
|
let(:coordinator) { instance_double(Gitlab::Database::Reindexing::Coordinator) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
queued_actions.take(limit).each do |action|
|
|
|
|
allow(Gitlab::Database::Reindexing::Coordinator).to receive(:new).with(action.index).and_return(coordinator)
|
|
|
|
allow(coordinator).to receive(:perform)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'consumes the queue in order of created_at and applies the limit' do
|
|
|
|
queued_actions.take(limit).each do |action|
|
|
|
|
expect(Gitlab::Database::Reindexing::Coordinator).to receive(:new).ordered.with(action.index).and_return(coordinator)
|
|
|
|
expect(coordinator).to receive(:perform)
|
|
|
|
end
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates queued action and sets state to done' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
queue = queued_actions
|
|
|
|
|
|
|
|
queue.shift(limit).each do |action|
|
|
|
|
expect(action.reload.state).to eq('done')
|
|
|
|
end
|
|
|
|
|
|
|
|
queue.each do |action|
|
|
|
|
expect(action.reload.state).to eq('queued')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'updates queued action upon error and sets state to failed' do
|
|
|
|
expect(Gitlab::Database::Reindexing::Coordinator).to receive(:new).ordered.with(queued_actions.first.index).and_return(coordinator)
|
|
|
|
expect(coordinator).to receive(:perform).and_raise('something went wrong')
|
|
|
|
|
|
|
|
subject
|
|
|
|
|
|
|
|
states = queued_actions.map(&:reload).map(&:state)
|
|
|
|
|
|
|
|
expect(states).to eq(%w(failed done queued))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
describe '.cleanup_leftovers!' do
|
2022-01-26 12:08:38 +05:30
|
|
|
subject(:cleanup_leftovers) { described_class.cleanup_leftovers! }
|
|
|
|
|
|
|
|
let(:model) { Gitlab::Database.database_base_models[Gitlab::Database::PRIMARY_DATABASE_NAME] }
|
|
|
|
let(:connection) { model.connection }
|
|
|
|
|
|
|
|
around do |example|
|
|
|
|
Gitlab::Database::SharedModel.using_connection(connection) do
|
|
|
|
example.run
|
|
|
|
end
|
|
|
|
end
|
2021-10-27 15:23:28 +05:30
|
|
|
|
|
|
|
before do
|
2022-01-26 12:08:38 +05:30
|
|
|
connection.execute(<<~SQL)
|
2021-10-27 15:23:28 +05:30
|
|
|
CREATE INDEX foobar_ccnew ON users (id);
|
|
|
|
CREATE INDEX foobar_ccnew1 ON users (id);
|
|
|
|
SQL
|
|
|
|
end
|
2021-01-03 14:25:43 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it 'drops both leftover indexes' do
|
|
|
|
expect_query("SET lock_timeout TO '60000ms'")
|
|
|
|
expect_query("DROP INDEX CONCURRENTLY IF EXISTS \"public\".\"foobar_ccnew\"")
|
|
|
|
expect_query("RESET idle_in_transaction_session_timeout; RESET lock_timeout")
|
|
|
|
expect_query("SET lock_timeout TO '60000ms'")
|
|
|
|
expect_query("DROP INDEX CONCURRENTLY IF EXISTS \"public\".\"foobar_ccnew1\"")
|
|
|
|
expect_query("RESET idle_in_transaction_session_timeout; RESET lock_timeout")
|
2021-01-03 14:25:43 +05:30
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
cleanup_leftovers
|
2021-10-27 15:23:28 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def expect_query(sql)
|
2022-01-26 12:08:38 +05:30
|
|
|
expect(connection).to receive(:execute).ordered.with(sql).and_wrap_original do |method, sql|
|
2021-10-27 15:23:28 +05:30
|
|
|
method.call(sql.sub(/CONCURRENTLY/, ''))
|
|
|
|
end
|
2021-01-03 14:25:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|