Files
gitlab-foss/spec/support/helpers/migrations_helpers.rb
2025-03-10 09:11:53 +00:00

302 lines
8.3 KiB
Ruby

# frozen_string_literal: true
module MigrationsHelpers
FINALIZE_FIRST_ERROR = <<ERROR
Schema should not be specified for background migrations, finalize the migration first.
The schema will be defaulted to the finalizing migration.
See https://docs.gitlab.com/ee/development/database/batched_background_migrations.html#finalize-a-batched-background-migration
ERROR
def migration_out_of_test_window?(migration_class)
return false if ENV.fetch('RUN_ALL_MIGRATION_TESTS', false)
min_milestone = Gitlab::Database.min_schema_gitlab_version
if migration_class < Gitlab::Database::Migration[1.0]
# Skip unless database migration older than min_schema_gitlab_version
milestone = migration_class.try(:milestone)
# Missing milestone indicates that the migration is pre-16.7,
# which is old enough not to execute its tests
return true unless milestone
migration_milestone = Gitlab::VersionInfo.parse_from_milestone(milestone)
migration_milestone < min_milestone
elsif migration_class < Gitlab::BackgroundMigration::BatchedMigrationJob
finalized_by = finalized_by_version.presence
# Execute tests for batched background migrations that are not yet finalized
return false unless finalized_by
finalize_migration_class = migration_class_for(finalized_by)
# Execute tests if we can't find migration class
return false unless finalize_migration_class
milestone = finalize_migration_class.try(:milestone)
# Missing milestone indicates that the migration is pre-16.7,
# which is old enough not to execute its tests
return true unless milestone
migration_milestone = Gitlab::VersionInfo.parse_from_milestone(milestone)
# Skip tests if finalized before the last required stop
migration_milestone < min_milestone
else
false
end
end
def active_record_base(database: nil)
if database.present?
conn = Gitlab::Database.all_database_connections[database]
raise ArgumentError, "#{database} is not a valid argument" unless conn
return conn.klass
end
migration_schema = self.class.metadata[:migration]
Gitlab::Database.schemas_to_base_models.dig(migration_schema, 0) ||
Gitlab::Database.database_base_models[:main]
end
def table(name, database: nil, primary_key: nil)
Class.new(active_record_base(database: database)) do
self.table_name = name
self.inheritance_column = :_type_disabled
self.primary_key = primary_key if primary_key.present?
def self.name
table_name.singularize.camelcase
end
yield self if block_given?
end
end
def partitioned_table(name, database: nil, by: :created_at, strategy: :monthly)
klass = Class.new(active_record_base(database: database)) do
include PartitionedTable
self.table_name = name
self.inheritance_column = :_type_disabled
self.primary_key = :id
partitioned_by by, strategy: strategy
def self.name
table_name.singularize.camelcase
end
end
klass.tap { Gitlab::Database::Partitioning.sync_partitions([klass]) }
end
def migrations_paths
active_record_base.connection.migrations_paths
end
def migration_context
ActiveRecord::MigrationContext.new(migrations_paths)
end
def migrations
migration_context.migrations
end
def clear_schema_cache!
active_record_base.connection_pool.connections.each do |conn|
conn.schema_cache.clear!
end
end
def foreign_key_exists?(source, target = nil, column: nil)
active_record_base.connection.foreign_keys(source).any? do |key|
if column
key.options[:column].to_s == column.to_s
else
key.to_table.to_s == target.to_s
end
end
end
def reset_column_in_all_models
clear_schema_cache!
# Reset column information for the most offending classes **after** we
# migrated the schema up, otherwise, column information could be
# outdated. We have a separate method for this so we can override it in EE.
active_record_base.descendants.each(&method(:reset_column_information))
end
def refresh_attribute_methods
# Without this, we get errors because of missing attributes, e.g.
# super: no superclass method `elasticsearch_indexing' for #<ApplicationSetting:0x00007f85628508d8>
# attr_encrypted also expects ActiveRecord attribute methods to be
# defined, or it will override the accessors:
# https://gitlab.com/gitlab-org/gitlab/issues/8234#note_113976421
[ApplicationSetting, SystemHook].each do |model|
model.define_attribute_methods
end
Gitlab.ee { License.define_attribute_methods }
end
def reset_column_information(klass)
klass.reset_column_information if klass.instance_variable_get(:@table_name)
end
# In some migration tests, we're using factories to create records,
# however those models might be depending on a schema version which
# doesn't have the columns we want in application_settings.
# In these cases, we'll need to use the fake application settings
# as if we have migrations pending
def use_fake_application_settings
# We stub this way because we can't stub on
# `current_application_settings` due to `method_missing` is
# depending on current_application_settings...
allow(Gitlab::Database::Migration::V1_0::MigrationRecord.connection)
.to receive(:active?)
.and_return(false)
allow(Gitlab::Runtime)
.to receive(:rake?)
.and_return(true)
stub_env('IN_MEMORY_APPLICATION_SETTINGS', 'false')
end
def previous_migration(steps_back = 2)
migrations.each_cons(steps_back) do |cons|
break cons.first if cons.last.name == described_class.name
end
end
def finalized_by_version
finalized_by = ::Gitlab::Utils::BatchedBackgroundMigrationsDictionary
.entry(described_class.to_s.demodulize)&.finalized_by
finalized_by.to_i if finalized_by.present?
end
def migration_schema_version
metadata_schema = self.class.metadata[:schema]
if metadata_schema == :latest
migrations.last.version
elsif self.class.metadata[:level] == :background_migration
raise FINALIZE_FIRST_ERROR if ENV['CI'].nil? && !metadata_schema.nil?
metadata_schema || finalized_by_version || migrations.last.version
else
metadata_schema || previous_migration.version
end
end
def schema_migrate_down!
disable_migrations_output do
migration_context.down(migration_schema_version)
end
reset_column_in_all_models
end
def schema_migrate_up!
reset_column_in_all_models
disable_migrations_output do
migration_context.up
end
reset_column_in_all_models
refresh_attribute_methods
end
def disable_migrations_output
ActiveRecord::Migration.verbose = false
yield
ensure
ActiveRecord::Migration.verbose = true
end
def migrate!
open_transactions = Gitlab::Database::Migration::V1_0::MigrationRecord.connection.open_transactions
allow_next_instance_of(described_class) do |migration|
allow(migration).to receive(:transaction_open?) do
Gitlab::Database::Migration::V1_0::MigrationRecord.connection.open_transactions > open_transactions
end
end
migration_context.up do |migration|
migration.name == described_class.name
end
end
class ReversibleMigrationTest
attr_reader :before_up, :after_up
def initialize
@before_up = -> {}
@after_up = -> {}
end
def before(expectations)
@before_up = expectations
self
end
def after(expectations)
@after_up = expectations
self
end
end
def reversible_migration(&block)
tests = yield(ReversibleMigrationTest.new)
tests.before_up.call
migrate!
tests.after_up.call
schema_migrate_down!
tests.before_up.call
end
private
def migration_class_for(timestamp)
migration_file = Dir[
*Rails.application.config.paths['db/migrate'].paths
.map { |dir| File.join(dir, "#{timestamp}_*.rb") }
].first
return unless migration_file
begin
require Rails.root.join(migration_file).to_s
rescue LoadError
return
end
matches = File.basename(migration_file).match(/\A\d+_(?<name>.*)\.rb\z/)
return unless matches[:name]
begin
matches[:name].classify.constantize
rescue NameError
nil
end
end
end
MigrationsHelpers.prepend_mod_with('MigrationsHelpers')