Commit 081d95c0 authored by Yorick Peterse's avatar Yorick Peterse

Disallow the use of X.establish_connection

This adds a RuboCop rule that disallows the usage of
`SomeModel.establish_connection`, as this pattern can break database
configurations and slows down the tests.

See https://gitlab.com/gitlab-org/gitlab/-/issues/338653 for more
information.
parent 55d31b13
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
if defined?(ActiveRecord::Base) && !Gitlab::Runtime.sidekiq? if defined?(ActiveRecord::Base) && !Gitlab::Runtime.sidekiq?
Gitlab::Cluster::LifecycleEvents.on_worker_start do Gitlab::Cluster::LifecycleEvents.on_worker_start do
ActiveSupport.on_load(:active_record) do ActiveSupport.on_load(:active_record) do
ActiveRecord::Base.establish_connection ActiveRecord::Base.establish_connection # rubocop: disable Database/EstablishConnection
Gitlab::AppLogger.debug("ActiveRecord connection established") Gitlab::AppLogger.debug("ActiveRecord connection established")
end end
......
...@@ -13,6 +13,6 @@ Gitlab.ee do ...@@ -13,6 +13,6 @@ Gitlab.ee do
# The Geo::TrackingBase model does not yet use connects_to. So, # The Geo::TrackingBase model does not yet use connects_to. So,
# this will not properly support geo: from config/databse.yml # this will not properly support geo: from config/databse.yml
# file yet. This is ACK of the current state and will be fixed. # file yet. This is ACK of the current state and will be fixed.
Geo::TrackingBase.establish_connection(Gitlab::Database.geo_db_config_with_default_pool_size) Geo::TrackingBase.establish_connection(Gitlab::Database.geo_db_config_with_default_pool_size) # rubocop: disable Database/EstablishConnection
end end
end end
...@@ -132,7 +132,7 @@ module Gitlab ...@@ -132,7 +132,7 @@ module Gitlab
ActiveRecord::Tasks::DatabaseTasks.load_schema(ActiveRecord::Base.configurations.configs_for(env_name: 'test').first, :sql, ENV['SCHEMA']) ActiveRecord::Tasks::DatabaseTasks.load_schema(ActiveRecord::Base.configurations.configs_for(env_name: 'test').first, :sql, ENV['SCHEMA'])
ensure ensure
if should_reconnect if should_reconnect
ActiveRecord::Base.establish_connection(Gitlab::Geo::DatabaseTasks.db_config) ActiveRecord::Base.establish_connection(Gitlab::Geo::DatabaseTasks.db_config) # rubocop: disable Database/EstablishConnection
end end
end end
end end
...@@ -204,7 +204,7 @@ module Gitlab ...@@ -204,7 +204,7 @@ module Gitlab
ActiveRecord::Migrator.migrations_paths = ActiveRecord::Tasks::DatabaseTasks.migrations_paths ActiveRecord::Migrator.migrations_paths = ActiveRecord::Tasks::DatabaseTasks.migrations_paths
Gitlab::Database::SchemaMigrations::Context.default_schema_migrations_path = settings[:schema_migrations_path] Gitlab::Database::SchemaMigrations::Context.default_schema_migrations_path = settings[:schema_migrations_path]
ActiveRecord::Base.establish_connection(db_config) ActiveRecord::Base.establish_connection(db_config) # rubocop: disable Database/EstablishConnection
end end
class SeedLoader class SeedLoader
......
...@@ -170,7 +170,7 @@ namespace :gitlab do ...@@ -170,7 +170,7 @@ namespace :gitlab do
# the `ActiveRecord::Base.connection` might be switched to another one # the `ActiveRecord::Base.connection` might be switched to another one
# This is due to `if should_reconnect`: # This is due to `if should_reconnect`:
# https://github.com/rails/rails/blob/a81aeb63a007ede2fe606c50539417dada9030c7/activerecord/lib/active_record/railties/databases.rake#L622 # https://github.com/rails/rails/blob/a81aeb63a007ede2fe606c50539417dada9030c7/activerecord/lib/active_record/railties/databases.rake#L622
ActiveRecord::Base.establish_connection :main ActiveRecord::Base.establish_connection :main # rubocop: disable Database/EstablishConnection
Rake::Task['gitlab:db:create_dynamic_partitions'].invoke Rake::Task['gitlab:db:create_dynamic_partitions'].invoke
end end
......
# frozen_string_literal: true
module RuboCop
module Cop
module Database
class EstablishConnection < RuboCop::Cop::Cop
MSG = "Don't establish new database connections, as this slows down " \
'tests and may result in new connections using an incorrect configuration'
def_node_matcher :establish_connection?, <<~PATTERN
(send (const ...) :establish_connection ...)
PATTERN
def on_send(node)
add_offense(node, location: :expression) if establish_connection?(node)
end
end
end
end
end
...@@ -12,7 +12,7 @@ module RspecProfiling ...@@ -12,7 +12,7 @@ module RspecProfiling
# This disables the automatic creation of the database and # This disables the automatic creation of the database and
# table. In the future, we may want a way to specify the host of # table. In the future, we may want a way to specify the host of
# the database to connect so that we can call #install. # the database to connect so that we can call #install.
Result.establish_connection(results_url) Result.establish_connection(results_url) # rubocop: disable Database/EstablishConnection
end end
def results_url def results_url
......
...@@ -101,7 +101,7 @@ RSpec.describe Gitlab::Database::BulkUpdate do ...@@ -101,7 +101,7 @@ RSpec.describe Gitlab::Database::BulkUpdate do
before do before do
configuration_hash = ActiveRecord::Base.connection_db_config.configuration_hash configuration_hash = ActiveRecord::Base.connection_db_config.configuration_hash
ActiveRecord::Base.establish_connection( ActiveRecord::Base.establish_connection( # rubocop: disable Database/EstablishConnection
configuration_hash.merge(prepared_statements: prepared_statements) configuration_hash.merge(prepared_statements: prepared_statements)
) )
end end
......
# frozen_string_literal: true
require 'spec_helper'
require_relative '../../../../rubocop/cop/database/establish_connection'
RSpec.describe RuboCop::Cop::Database::EstablishConnection do
subject(:cop) { described_class.new }
it 'flags the use of ActiveRecord::Base.establish_connection' do
expect_offense(<<~CODE)
ActiveRecord::Base.establish_connection
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Don't establish new database [...]
CODE
end
it 'flags the use of ActiveRecord::Base.establish_connection with arguments' do
expect_offense(<<~CODE)
ActiveRecord::Base.establish_connection(:foo)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Don't establish new database [...]
CODE
end
it 'flags the use of SomeModel.establish_connection' do
expect_offense(<<~CODE)
SomeModel.establish_connection
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Don't establish new database [...]
CODE
end
end
...@@ -67,7 +67,7 @@ module DbCleaner ...@@ -67,7 +67,7 @@ module DbCleaner
# Migrate each database individually # Migrate each database individually
with_reestablished_active_record_base do with_reestablished_active_record_base do
all_connection_classes.each do |connection_class| all_connection_classes.each do |connection_class|
ActiveRecord::Base.establish_connection(connection_class.connection_db_config) ActiveRecord::Base.establish_connection(connection_class.connection_db_config) # rubocop: disable Database/EstablishConnection
ActiveRecord::Tasks::DatabaseTasks.migrate ActiveRecord::Tasks::DatabaseTasks.migrate
end end
......
...@@ -7,13 +7,13 @@ RSpec.describe 'Database::MultipleDatabases' do ...@@ -7,13 +7,13 @@ RSpec.describe 'Database::MultipleDatabases' do
context 'when doing establish_connection' do context 'when doing establish_connection' do
context 'on ActiveRecord::Base' do context 'on ActiveRecord::Base' do
it 'raises exception' do it 'raises exception' do
expect { ActiveRecord::Base.establish_connection(:main) }.to raise_error /Cannot re-establish/ expect { ActiveRecord::Base.establish_connection(:main) }.to raise_error /Cannot re-establish/ # rubocop: disable Database/EstablishConnection
end end
context 'when using with_reestablished_active_record_base' do context 'when using with_reestablished_active_record_base' do
it 'does not raise exception' do it 'does not raise exception' do
with_reestablished_active_record_base do with_reestablished_active_record_base do
expect { ActiveRecord::Base.establish_connection(:main) }.not_to raise_error expect { ActiveRecord::Base.establish_connection(:main) }.not_to raise_error # rubocop: disable Database/EstablishConnection
end end
end end
end end
...@@ -25,13 +25,13 @@ RSpec.describe 'Database::MultipleDatabases' do ...@@ -25,13 +25,13 @@ RSpec.describe 'Database::MultipleDatabases' do
end end
it 'raises exception' do it 'raises exception' do
expect { Ci::ApplicationRecord.establish_connection(:ci) }.to raise_error /Cannot re-establish/ expect { Ci::ApplicationRecord.establish_connection(:ci) }.to raise_error /Cannot re-establish/ # rubocop: disable Database/EstablishConnection
end end
context 'when using with_reestablished_active_record_base' do context 'when using with_reestablished_active_record_base' do
it 'does not raise exception' do it 'does not raise exception' do
with_reestablished_active_record_base do with_reestablished_active_record_base do
expect { Ci::ApplicationRecord.establish_connection(:main) }.not_to raise_error expect { Ci::ApplicationRecord.establish_connection(:main) }.not_to raise_error # rubocop: disable Database/EstablishConnection
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