2021-12-11 22:18:48 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
RSpec.describe Gitlab::Database::EachDatabase do
|
2022-05-07 20:08:51 +05:30
|
|
|
describe '.each_database_connection', :add_ci_connection do
|
2022-04-04 11:22:00 +05:30
|
|
|
before do
|
|
|
|
allow(Gitlab::Database).to receive(:database_base_models)
|
|
|
|
.and_return({ main: ActiveRecord::Base, ci: Ci::ApplicationRecord }.with_indifferent_access)
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
it 'yields each connection after connecting SharedModel' do
|
2022-04-04 11:22:00 +05:30
|
|
|
expect(Gitlab::Database::SharedModel).to receive(:using_connection)
|
|
|
|
.with(ActiveRecord::Base.connection).ordered.and_yield
|
2021-12-11 22:18:48 +05:30
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
expect(Gitlab::Database::SharedModel).to receive(:using_connection)
|
|
|
|
.with(Ci::ApplicationRecord.connection).ordered.and_yield
|
2021-12-11 22:18:48 +05:30
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
expect { |b| described_class.each_database_connection(&b) }
|
|
|
|
.to yield_successive_args(
|
|
|
|
[ActiveRecord::Base.connection, 'main'],
|
|
|
|
[Ci::ApplicationRecord.connection, 'ci']
|
|
|
|
)
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
2022-05-07 20:08:51 +05:30
|
|
|
|
|
|
|
context 'when only certain databases are selected' do
|
|
|
|
it 'yields the selected connections after connecting SharedModel' do
|
|
|
|
expect(Gitlab::Database::SharedModel).to receive(:using_connection)
|
|
|
|
.with(Ci::ApplicationRecord.connection).ordered.and_yield
|
|
|
|
|
|
|
|
expect { |b| described_class.each_database_connection(only: 'ci', &b) }
|
|
|
|
.to yield_successive_args([Ci::ApplicationRecord.connection, 'ci'])
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the selected names are passed as symbols' do
|
|
|
|
it 'yields the selected connections after connecting SharedModel' do
|
|
|
|
expect(Gitlab::Database::SharedModel).to receive(:using_connection)
|
|
|
|
.with(Ci::ApplicationRecord.connection).ordered.and_yield
|
|
|
|
|
|
|
|
expect { |b| described_class.each_database_connection(only: :ci, &b) }
|
|
|
|
.to yield_successive_args([Ci::ApplicationRecord.connection, 'ci'])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the selected names are invalid' do
|
|
|
|
it 'does not yield any connections' do
|
|
|
|
expect do |b|
|
|
|
|
described_class.each_database_connection(only: :notvalid, &b)
|
|
|
|
rescue ArgumentError => e
|
|
|
|
expect(e.message).to match(/notvalid is not a valid database name/)
|
|
|
|
end.not_to yield_control
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'raises an error' do
|
|
|
|
expect do
|
|
|
|
described_class.each_database_connection(only: :notvalid) {}
|
|
|
|
end.to raise_error(ArgumentError, /notvalid is not a valid database name/)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2022-06-21 17:19:12 +05:30
|
|
|
|
|
|
|
context 'when shared connections are not included' do
|
|
|
|
it 'only yields the unshared connections' do
|
2022-07-23 23:45:48 +05:30
|
|
|
if Gitlab::Database.has_config?(:ci)
|
|
|
|
expect(Gitlab::Database).to receive(:db_config_share_with).exactly(3).times.and_return(nil, 'main', 'main')
|
|
|
|
else
|
|
|
|
expect(Gitlab::Database).to receive(:db_config_share_with).twice.and_return(nil, 'main')
|
|
|
|
end
|
2022-06-21 17:19:12 +05:30
|
|
|
|
|
|
|
expect { |b| described_class.each_database_connection(include_shared: false, &b) }
|
|
|
|
.to yield_successive_args([ActiveRecord::Base.connection, 'main'])
|
|
|
|
end
|
|
|
|
end
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '.each_model_connection' do
|
2022-04-04 11:22:00 +05:30
|
|
|
context 'when the model inherits from SharedModel', :add_ci_connection do
|
|
|
|
let(:model1) { Class.new(Gitlab::Database::SharedModel) }
|
|
|
|
let(:model2) { Class.new(Gitlab::Database::SharedModel) }
|
2021-12-11 22:18:48 +05:30
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
before do
|
|
|
|
allow(Gitlab::Database).to receive(:database_base_models)
|
|
|
|
.and_return({ main: ActiveRecord::Base, ci: Ci::ApplicationRecord }.with_indifferent_access)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'yields each model with SharedModel connected to each database connection' do
|
|
|
|
expect_yielded_models([model1, model2], [
|
|
|
|
{ model: model1, connection: ActiveRecord::Base.connection, name: 'main' },
|
|
|
|
{ model: model1, connection: Ci::ApplicationRecord.connection, name: 'ci' },
|
|
|
|
{ model: model2, connection: ActiveRecord::Base.connection, name: 'main' },
|
|
|
|
{ model: model2, connection: Ci::ApplicationRecord.connection, name: 'ci' }
|
|
|
|
])
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the model limits connection names' do
|
|
|
|
before do
|
|
|
|
model1.limit_connection_names = %i[main]
|
|
|
|
model2.limit_connection_names = %i[ci]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'only yields the model with SharedModel connected to the limited connections' do
|
|
|
|
expect_yielded_models([model1, model2], [
|
|
|
|
{ model: model1, connection: ActiveRecord::Base.connection, name: 'main' },
|
|
|
|
{ model: model2, connection: Ci::ApplicationRecord.connection, name: 'ci' }
|
|
|
|
])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the model does not inherit from SharedModel' do
|
|
|
|
let(:main_model) { Class.new(ActiveRecord::Base) }
|
|
|
|
let(:ci_model) { Class.new(Ci::ApplicationRecord) }
|
|
|
|
|
|
|
|
let(:main_connection) { double(:connection) }
|
|
|
|
let(:ci_connection) { double(:connection) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(main_model).to receive(:connection).and_return(main_connection)
|
|
|
|
allow(ci_model).to receive(:connection).and_return(ci_connection)
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
allow(main_model).to receive_message_chain('connection_db_config.name').and_return('main')
|
|
|
|
allow(ci_model).to receive_message_chain('connection_db_config.name').and_return('ci')
|
2022-04-04 11:22:00 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'yields each model after connecting SharedModel' do
|
|
|
|
expect_yielded_models([main_model, ci_model], [
|
|
|
|
{ model: main_model, connection: main_connection, name: 'main' },
|
|
|
|
{ model: ci_model, connection: ci_connection, name: 'ci' }
|
|
|
|
])
|
|
|
|
end
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
context 'when the database connections are limited by the only_on option' do
|
|
|
|
let(:shared_model) { Class.new(Gitlab::Database::SharedModel) }
|
|
|
|
let(:main_model) { Class.new(ActiveRecord::Base) }
|
|
|
|
let(:ci_model) { Class.new(Ci::ApplicationRecord) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab::Database).to receive(:database_base_models)
|
|
|
|
.and_return({ main: ActiveRecord::Base, ci: Ci::ApplicationRecord }.with_indifferent_access)
|
|
|
|
|
|
|
|
allow(main_model).to receive_message_chain('connection_db_config.name').and_return('main')
|
|
|
|
allow(ci_model).to receive_message_chain('connection_db_config.name').and_return('ci')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when a single name is passed in' do
|
|
|
|
it 'yields models only connected to the given database' do
|
|
|
|
expect_yielded_models([main_model, ci_model, shared_model], [
|
|
|
|
{ model: ci_model, connection: Ci::ApplicationRecord.connection, name: 'ci' },
|
|
|
|
{ model: shared_model, connection: Ci::ApplicationRecord.connection, name: 'ci' }
|
|
|
|
], only_on: 'ci')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when a list of names are passed in' do
|
|
|
|
it 'yields models only connected to the given databases' do
|
|
|
|
expect_yielded_models([main_model, ci_model, shared_model], [
|
|
|
|
{ model: main_model, connection: ActiveRecord::Base.connection, name: 'main' },
|
|
|
|
{ model: ci_model, connection: Ci::ApplicationRecord.connection, name: 'ci' },
|
|
|
|
{ model: shared_model, connection: ActiveRecord::Base.connection, name: 'main' },
|
|
|
|
{ model: shared_model, connection: Ci::ApplicationRecord.connection, name: 'ci' }
|
|
|
|
], only_on: %i[main ci])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def expect_yielded_models(models_to_iterate, expected_values, only_on: nil)
|
2022-04-04 11:22:00 +05:30
|
|
|
times_yielded = 0
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
described_class.each_model_connection(models_to_iterate, only_on: only_on) do |model, name|
|
2022-04-04 11:22:00 +05:30
|
|
|
expected = expected_values[times_yielded]
|
2021-12-11 22:18:48 +05:30
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
expect(model).to be(expected[:model])
|
|
|
|
expect(model.connection).to be(expected[:connection])
|
|
|
|
expect(name).to eq(expected[:name])
|
2021-12-11 22:18:48 +05:30
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
times_yielded += 1
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
expect(times_yielded).to eq(expected_values.size)
|
2021-12-11 22:18:48 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|