123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- # require all database migrations so we can test them without manual require
- Rails.root.join('db/migrate').children.each do |migration|
- require migration.to_s
- end
- module DbMigrationHelper
- # Provides a helper method to execute a migration for the current class.
- # Make sure to define type: :db_migration in your RSpec.describe call.
- #
- # @param [Symbol] direction the migration should take (:up or :down)
- # @yield [instance] Yields the created instance of the
- # migration to allow expectations or other changes to it
- #
- # @example
- # migrate
- #
- # @example
- # migrate(:down)
- #
- # @return [nil]
- def migrate(direction = :up)
- instance = described_class.new
- yield(instance) if block_given?
- instance.suppress_messages do
- instance.migrate(direction)
- end
- end
- # Provides a helper method to remove foreign_keys if exist.
- # Make sure to define type: :db_migration in your RSpec.describe call
- # and add `self.use_transactional_tests = false` to your context.
- #
- # ATTENTION: We do not use the same arguments as the internally
- # used methods since giving a table name
- # as a second argument as e.g.
- # `remove_foreign_key(:online_notifications, :users)`
- # doesn't remove the index at first execution on at least MySQL
- #
- # @param [Symbol] from_table the name of the table with the foreign_key column
- # @param [Symbol] column the name of the foreign_key column
- #
- # @example
- # without_foreign_key(:online_notifications, column: :user_id)
- #
- # @return [nil]
- def without_foreign_key(from_table, column:)
- suppress_messages do
- break if !foreign_key_exists?(from_table, column: column)
- remove_foreign_key(from_table, column: column)
- end
- end
- # Helper method for setting up specs on DB migrations that add columns.
- # Make sure to define type: :db_migration in your RSpec.describe call
- # and add `self.use_transactional_tests = false` to your context.
- #
- # @param [Symbol] from_table the name of the table with the indexed column
- # @param [Symbol] name(s) of indexed column(s)
- #
- # @example
- # without_column(:online_notifications, column: :user_id)
- #
- # @return [nil]
- def without_column(from_table, column:)
- suppress_messages do
- Array(column).each do |elem|
- next if !column_exists?(from_table, elem)
- remove_column(from_table, elem)
- end
- end
- end
- # Helper method for setting up specs on DB migrations that add indices.
- # Make sure to define type: :db_migration in your RSpec.describe call
- # and add `self.use_transactional_tests = false` to your context.
- #
- # @param [Symbol] from_table the name of the table with the indexed column
- # @param [Symbol] name(s) of indexed column(s)
- #
- # @example
- # without_index(:online_notifications, column: :user_id)
- #
- # @return [nil]
- def without_index(from_table, column:)
- suppress_messages do
- break if !index_exists?(from_table, column)
- remove_index(from_table, column: column)
- end
- end
- # Enables the usage of `ActiveRecord::Migration` methods.
- #
- # @see ActiveRecord::Migration
- #
- # @example
- # remove_foreign_key(:online_notifications, :users)
- #
- # @return [nil]
- def method_missing(method, ...)
- ActiveRecord::Migration.send(method, ...)
- rescue NoMethodError
- super
- end
- # Enables the usage of `ActiveRecord::Migration` methods.
- #
- # @see ActiveRecord::Migration
- #
- # @example
- # remove_foreign_key(:online_notifications, :users)
- #
- # @return [nil]
- def respond_to_missing?(...)
- true
- end
- # Provides a helper method to check if migration class adds a foreign key.
- # Make sure to define type: :db_migration in your RSpec.describe call
- # and add `self.use_transactional_tests = false` to your context.
- #
- # @param [Symbol] from_table the name of the table with the foreign_key column
- # @param [Symbol] column the name of the foreign_key column
- #
- # @example
- # adds_foreign_key(:online_notifications, column: :user_id)
- #
- # @return [nil]
- def adds_foreign_key(from_table, column:)
- without_foreign_key(from_table, column: column)
- suppress_messages do
- expect do
- migrate
- end.to change {
- foreign_key_exists?(from_table, column: column)
- }
- end
- end
- def self.included(base)
- # Execute in RSpec class context
- base.class_exec do
- # This method simulates a system that is is already initialized
- # aka `Setting.exists?(name: 'system_init_done')`
- # It's possible to simulate a not yet initialized system by adding the
- # meta tag `system_init_done` to `false` to the needing example:
- #
- # @example
- # it 'does stuff in an unitialized system', system_init_done: false do
- #
- before do |example|
- initialized = example.metadata.fetch(:system_init_done, true)
- system_init_done(initialized)
- end
- end
- end
- end
- RSpec.configure do |config|
- config.include DbMigrationHelper, type: :db_migration
- end
|