2019-10-12 21:52:04 +05:30
# frozen_string_literal: true
2016-06-02 11:05:42 +05:30
require 'spec_helper'
2020-07-28 23:09:34 +05:30
RSpec . describe Gitlab :: Database :: MigrationHelpers do
2021-03-08 18:12:59 +05:30
include Database :: TableSchemaHelpers
2021-06-08 01:23:25 +05:30
include Database :: TriggerHelpers
2021-03-08 18:12:59 +05:30
2016-06-02 11:05:42 +05:30
let ( :model ) do
2018-03-17 18:26:18 +05:30
ActiveRecord :: Migration . new . extend ( described_class )
2016-06-02 11:05:42 +05:30
end
2017-09-10 17:25:29 +05:30
before do
allow ( model ) . to receive ( :puts )
end
2022-04-04 11:22:00 +05:30
describe 'overridden dynamic model helpers' do
let ( :test_table ) { '__test_batching_table' }
before do
model . connection . execute ( << ~ SQL )
CREATE TABLE #{test_table} (
id integer NOT NULL PRIMARY KEY ,
name text NOT NULL
) ;
INSERT INTO #{test_table} (id, name)
VALUES ( 1 , 'bob' ) , ( 2 , 'mary' ) , ( 3 , 'amy' ) ;
SQL
end
describe '#define_batchable_model' do
it 'defines a batchable model with the migration connection' do
expect ( model . define_batchable_model ( test_table ) . count ) . to eq ( 3 )
end
end
describe '#each_batch' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
it 'calls each_batch with the migration connection' do
each_batch_name = - > ( & block ) do
model . each_batch ( test_table , of : 2 ) do | batch |
block . call ( batch . pluck ( :name ) )
end
end
expect { | b | each_batch_name . call ( & b ) } . to yield_successive_args ( %w[ bob mary ] , %w[ amy ] )
end
end
describe '#each_batch_range' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
it 'calls each_batch with the migration connection' do
expect { | b | model . each_batch_range ( test_table , of : 2 , & b ) } . to yield_successive_args ( [ 1 , 2 ] , [ 3 , 3 ] )
end
end
end
2019-10-12 21:52:04 +05:30
describe '#remove_timestamps' do
it 'can remove the default timestamps' do
Gitlab :: Database :: MigrationHelpers :: DEFAULT_TIMESTAMP_COLUMNS . each do | column_name |
expect ( model ) . to receive ( :remove_column ) . with ( :foo , column_name )
end
model . remove_timestamps ( :foo )
end
it 'can remove custom timestamps' do
expect ( model ) . to receive ( :remove_column ) . with ( :foo , :bar )
model . remove_timestamps ( :foo , columns : [ :bar ] )
end
end
2017-09-10 17:25:29 +05:30
describe '#add_timestamps_with_timezone' do
2019-10-12 21:52:04 +05:30
it 'adds "created_at" and "updated_at" fields with the "datetime_with_timezone" data type' do
Gitlab :: Database :: MigrationHelpers :: DEFAULT_TIMESTAMP_COLUMNS . each do | column_name |
2021-12-11 22:18:48 +05:30
expect ( model ) . to receive ( :add_column )
. with ( :foo , column_name , :datetime_with_timezone , { default : nil , null : false } )
2017-09-10 17:25:29 +05:30
end
2019-10-12 21:52:04 +05:30
model . add_timestamps_with_timezone ( :foo )
end
2017-09-10 17:25:29 +05:30
2019-10-12 21:52:04 +05:30
it 'can disable the NOT NULL constraint' do
Gitlab :: Database :: MigrationHelpers :: DEFAULT_TIMESTAMP_COLUMNS . each do | column_name |
2021-12-11 22:18:48 +05:30
expect ( model ) . to receive ( :add_column )
. with ( :foo , column_name , :datetime_with_timezone , { default : nil , null : true } )
2017-09-10 17:25:29 +05:30
end
2019-10-12 21:52:04 +05:30
model . add_timestamps_with_timezone ( :foo , null : true )
2017-09-10 17:25:29 +05:30
end
2019-10-12 21:52:04 +05:30
it 'can add just one column' do
expect ( model ) . to receive ( :add_column ) . with ( :foo , :created_at , :datetime_with_timezone , anything )
expect ( model ) . not_to receive ( :add_column ) . with ( :foo , :updated_at , :datetime_with_timezone , anything )
model . add_timestamps_with_timezone ( :foo , columns : [ :created_at ] )
end
it 'can add choice of acceptable columns' do
expect ( model ) . to receive ( :add_column ) . with ( :foo , :created_at , :datetime_with_timezone , anything )
expect ( model ) . to receive ( :add_column ) . with ( :foo , :deleted_at , :datetime_with_timezone , anything )
2021-12-11 22:18:48 +05:30
expect ( model ) . to receive ( :add_column ) . with ( :foo , :processed_at , :datetime_with_timezone , anything )
2019-10-12 21:52:04 +05:30
expect ( model ) . not_to receive ( :add_column ) . with ( :foo , :updated_at , :datetime_with_timezone , anything )
2021-12-11 22:18:48 +05:30
model . add_timestamps_with_timezone ( :foo , columns : [ :created_at , :deleted_at , :processed_at ] )
2019-10-12 21:52:04 +05:30
end
it 'cannot add unacceptable column names' do
expect do
model . add_timestamps_with_timezone ( :foo , columns : [ :bar ] )
end . to raise_error %r/Illegal timestamp column name/
end
2017-09-10 17:25:29 +05:30
end
2016-06-02 11:05:42 +05:30
2021-03-08 18:12:59 +05:30
describe '#create_table_with_constraints' do
let ( :table_name ) { :test_table }
let ( :column_attributes ) do
[
{ name : 'id' , sql_type : 'bigint' , null : false , default : nil } ,
{ name : 'created_at' , sql_type : 'timestamp with time zone' , null : false , default : nil } ,
{ name : 'updated_at' , sql_type : 'timestamp with time zone' , null : false , default : nil } ,
{ name : 'some_id' , sql_type : 'integer' , null : false , default : nil } ,
{ name : 'active' , sql_type : 'boolean' , null : false , default : 'true' } ,
{ name : 'name' , sql_type : 'text' , null : true , default : nil }
]
end
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( true )
end
context 'when no check constraints are defined' do
it 'creates the table as expected' do
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
end
expect_table_columns_to_match ( column_attributes , table_name )
end
end
context 'when check constraints are defined' do
context 'when the text_limit is explicity named' do
it 'creates the table as expected' do
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
t . text_limit :name , 255 , name : 'check_name_length'
t . check_constraint :some_id_is_positive , 'some_id > 0'
end
expect_table_columns_to_match ( column_attributes , table_name )
expect_check_constraint ( table_name , 'check_name_length' , 'char_length(name) <= 255' )
expect_check_constraint ( table_name , 'some_id_is_positive' , 'some_id > 0' )
end
end
context 'when the text_limit is not named' do
it 'creates the table as expected, naming the text limit' do
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
t . text_limit :name , 255
t . check_constraint :some_id_is_positive , 'some_id > 0'
end
expect_table_columns_to_match ( column_attributes , table_name )
expect_check_constraint ( table_name , 'check_cda6f69506' , 'char_length(name) <= 255' )
expect_check_constraint ( table_name , 'some_id_is_positive' , 'some_id > 0' )
end
end
it 'runs the change within a with_lock_retries' do
expect ( model ) . to receive ( :with_lock_retries ) . ordered . and_yield
expect ( model ) . to receive ( :create_table ) . ordered . and_call_original
expect ( model ) . to receive ( :execute ) . with ( << ~ SQL ) . ordered
ALTER TABLE " #{ table_name } " \ nADD CONSTRAINT " check_cda6f69506 " CHECK ( char_length ( " name " ) < = 255 )
SQL
model . create_table_with_constraints table_name do | t |
t . text :name
t . text_limit :name , 255
end
end
2021-04-17 20:07:23 +05:30
context 'when with_lock_retries re-runs the block' do
it 'only creates constraint for unique definitions' do
expected_sql = << ~ SQL
ALTER TABLE " #{ table_name } " \ nADD CONSTRAINT " check_cda6f69506 " CHECK ( char_length ( " name " ) < = 255 )
SQL
expect ( model ) . to receive ( :create_table ) . twice . and_call_original
expect ( model ) . to receive ( :execute ) . with ( expected_sql ) . and_raise ( ActiveRecord :: LockWaitTimeout )
expect ( model ) . to receive ( :execute ) . with ( expected_sql ) . and_call_original
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
t . text_limit :name , 255
end
expect_table_columns_to_match ( column_attributes , table_name )
expect_check_constraint ( table_name , 'check_cda6f69506' , 'char_length(name) <= 255' )
end
end
2021-03-08 18:12:59 +05:30
context 'when constraints are given invalid names' do
let ( :expected_max_length ) { described_class :: MAX_IDENTIFIER_NAME_LENGTH }
let ( :expected_error_message ) { " The maximum allowed constraint name is #{ expected_max_length } characters " }
context 'when the explicit text limit name is not valid' do
it 'raises an error' do
too_long_length = expected_max_length + 1
expect do
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
t . text_limit :name , 255 , name : ( 'a' * too_long_length )
t . check_constraint :some_id_is_positive , 'some_id > 0'
end
end . to raise_error ( expected_error_message )
end
end
context 'when a check constraint name is not valid' do
it 'raises an error' do
too_long_length = expected_max_length + 1
expect do
model . create_table_with_constraints table_name do | t |
t . timestamps_with_timezone
t . integer :some_id , null : false
t . boolean :active , null : false , default : true
t . text :name
t . text_limit :name , 255
t . check_constraint ( 'a' * too_long_length ) , 'some_id > 0'
end
end . to raise_error ( expected_error_message )
end
end
end
end
end
2016-06-02 11:05:42 +05:30
describe '#add_concurrent_index' do
context 'outside a transaction' do
before do
2017-08-17 22:00:37 +05:30
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
2019-10-12 21:52:04 +05:30
allow ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2016-06-02 11:05:42 +05:30
end
2019-10-12 21:52:04 +05:30
it 'creates the index concurrently' do
expect ( model ) . to receive ( :add_index )
. with ( :users , :foo , algorithm : :concurrently )
2018-05-09 12:01:36 +05:30
2019-10-12 21:52:04 +05:30
model . add_concurrent_index ( :users , :foo )
2016-06-02 11:05:42 +05:30
end
2019-10-12 21:52:04 +05:30
it 'creates unique index concurrently' do
expect ( model ) . to receive ( :add_index )
. with ( :users , :foo , { algorithm : :concurrently , unique : true } )
2016-06-02 11:05:42 +05:30
2019-10-12 21:52:04 +05:30
model . add_concurrent_index ( :users , :foo , unique : true )
end
2018-05-09 12:01:36 +05:30
2021-12-11 22:18:48 +05:30
context 'when the index exists and is valid' do
before do
model . add_index :users , :id , unique : true
end
2018-05-09 12:01:36 +05:30
2021-12-11 22:18:48 +05:30
it 'does leaves the existing index' do
expect ( model ) . to receive ( :index_exists? )
. with ( :users , :id , { algorithm : :concurrently , unique : true } ) . and_call_original
expect ( model ) . not_to receive ( :remove_index )
expect ( model ) . not_to receive ( :add_index )
model . add_concurrent_index ( :users , :id , unique : true )
end
end
context 'when an invalid copy of the index exists' do
before do
model . add_index :users , :id , unique : true , name : index_name
model . connection . execute ( << ~ SQL )
UPDATE pg_index
SET indisvalid = false
WHERE indexrelid = '#{index_name}' :: regclass
SQL
end
context 'when the default name is used' do
let ( :index_name ) { model . index_name ( :users , :id ) }
it 'drops and recreates the index' do
expect ( model ) . to receive ( :index_exists? )
. with ( :users , :id , { algorithm : :concurrently , unique : true } ) . and_call_original
expect ( model ) . to receive ( :index_invalid? ) . with ( index_name , schema : nil ) . and_call_original
expect ( model ) . to receive ( :remove_concurrent_index_by_name ) . with ( :users , index_name )
expect ( model ) . to receive ( :add_index )
. with ( :users , :id , { algorithm : :concurrently , unique : true } )
model . add_concurrent_index ( :users , :id , unique : true )
end
end
context 'when a custom name is used' do
let ( :index_name ) { 'my_test_index' }
it 'drops and recreates the index' do
expect ( model ) . to receive ( :index_exists? )
. with ( :users , :id , { algorithm : :concurrently , unique : true , name : index_name } ) . and_call_original
expect ( model ) . to receive ( :index_invalid? ) . with ( index_name , schema : nil ) . and_call_original
expect ( model ) . to receive ( :remove_concurrent_index_by_name ) . with ( :users , index_name )
expect ( model ) . to receive ( :add_index )
. with ( :users , :id , { algorithm : :concurrently , unique : true , name : index_name } )
model . add_concurrent_index ( :users , :id , unique : true , name : index_name )
end
end
context 'when a qualified table name is used' do
let ( :other_schema ) { 'foo_schema' }
let ( :index_name ) { 'my_test_index' }
let ( :table_name ) { " #{ other_schema } .users " }
before do
model . connection . execute ( << ~ SQL )
CREATE SCHEMA #{other_schema};
ALTER TABLE users SET SCHEMA #{other_schema};
SQL
end
it 'drops and recreates the index' do
expect ( model ) . to receive ( :index_exists? )
. with ( table_name , :id , { algorithm : :concurrently , unique : true , name : index_name } ) . and_call_original
expect ( model ) . to receive ( :index_invalid? ) . with ( index_name , schema : other_schema ) . and_call_original
expect ( model ) . to receive ( :remove_concurrent_index_by_name ) . with ( table_name , index_name )
expect ( model ) . to receive ( :add_index )
. with ( table_name , :id , { algorithm : :concurrently , unique : true , name : index_name } )
model . add_concurrent_index ( table_name , :id , unique : true , name : index_name )
end
end
2016-06-02 11:05:42 +05:30
end
2021-10-27 15:23:28 +05:30
it 'unprepares the async index creation' do
expect ( model ) . to receive ( :add_index )
. with ( :users , :foo , algorithm : :concurrently )
expect ( model ) . to receive ( :unprepare_async_index )
. with ( :users , :foo , algorithm : :concurrently )
model . add_concurrent_index ( :users , :foo )
end
2016-06-02 11:05:42 +05:30
end
context 'inside a transaction' do
it 'raises RuntimeError' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
2017-09-10 17:25:29 +05:30
expect { model . add_concurrent_index ( :users , :foo ) }
. to raise_error ( RuntimeError )
2016-06-02 11:05:42 +05:30
end
end
end
2017-08-17 22:00:37 +05:30
describe '#remove_concurrent_index' do
context 'outside a transaction' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
2018-05-09 12:01:36 +05:30
allow ( model ) . to receive ( :index_exists? ) . and_return ( true )
2018-11-20 20:47:30 +05:30
allow ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2017-08-17 22:00:37 +05:30
end
2019-10-12 21:52:04 +05:30
describe 'by column name' do
it 'removes the index concurrently' do
expect ( model ) . to receive ( :remove_index )
. with ( :users , { algorithm : :concurrently , column : :foo } )
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
model . remove_concurrent_index ( :users , :foo )
end
2018-05-09 12:01:36 +05:30
2019-10-12 21:52:04 +05:30
it 'does nothing if the index does not exist' do
expect ( model ) . to receive ( :index_exists? )
. with ( :users , :foo , { algorithm : :concurrently , unique : true } ) . and_return ( false )
expect ( model ) . not_to receive ( :remove_index )
2018-05-09 12:01:36 +05:30
2019-10-12 21:52:04 +05:30
model . remove_concurrent_index ( :users , :foo , unique : true )
2017-08-17 22:00:37 +05:30
end
2017-09-10 17:25:29 +05:30
2021-10-27 15:23:28 +05:30
it 'unprepares the async index creation' do
expect ( model ) . to receive ( :remove_index )
. with ( :users , { algorithm : :concurrently , column : :foo } )
expect ( model ) . to receive ( :unprepare_async_index )
. with ( :users , :foo , { algorithm : :concurrently } )
model . remove_concurrent_index ( :users , :foo )
end
2018-05-09 12:01:36 +05:30
describe 'by index name' do
before do
allow ( model ) . to receive ( :index_exists_by_name? ) . with ( :users , " index_x_by_y " ) . and_return ( true )
end
it 'removes the index concurrently by index name' do
expect ( model ) . to receive ( :remove_index )
. with ( :users , { algorithm : :concurrently , name : " index_x_by_y " } )
model . remove_concurrent_index_by_name ( :users , " index_x_by_y " )
end
it 'does nothing if the index does not exist' do
expect ( model ) . to receive ( :index_exists_by_name? ) . with ( :users , " index_x_by_y " ) . and_return ( false )
expect ( model ) . not_to receive ( :remove_index )
2017-09-10 17:25:29 +05:30
2018-05-09 12:01:36 +05:30
model . remove_concurrent_index_by_name ( :users , " index_x_by_y " )
end
2020-07-28 23:09:34 +05:30
it 'removes the index with keyword arguments' do
expect ( model ) . to receive ( :remove_index )
. with ( :users , { algorithm : :concurrently , name : " index_x_by_y " } )
model . remove_concurrent_index_by_name ( :users , name : " index_x_by_y " )
end
it 'raises an error if the index is blank' do
expect do
model . remove_concurrent_index_by_name ( :users , wrong_key : " index_x_by_y " )
end . to raise_error 'remove_concurrent_index_by_name must get an index name as the second argument'
end
2021-10-27 15:23:28 +05:30
it 'unprepares the async index creation' do
expect ( model ) . to receive ( :remove_index )
. with ( :users , { algorithm : :concurrently , name : " index_x_by_y " } )
expect ( model ) . to receive ( :unprepare_async_index_by_name )
. with ( :users , " index_x_by_y " , { algorithm : :concurrently } )
model . remove_concurrent_index_by_name ( :users , " index_x_by_y " )
end
2017-09-10 17:25:29 +05:30
end
2017-08-17 22:00:37 +05:30
end
end
context 'inside a transaction' do
it 'raises RuntimeError' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
2017-09-10 17:25:29 +05:30
expect { model . remove_concurrent_index ( :users , :foo ) }
. to raise_error ( RuntimeError )
2017-08-17 22:00:37 +05:30
end
end
end
2022-04-04 11:22:00 +05:30
describe '#remove_foreign_key_if_exists' do
context 'when the foreign key does not exist' do
before do
allow ( model ) . to receive ( :foreign_key_exists? ) . and_return ( false )
end
it 'does nothing' do
expect ( model ) . not_to receive ( :remove_foreign_key )
model . remove_foreign_key_if_exists ( :projects , :users , column : :user_id )
end
end
context 'when the foreign key exists' do
before do
allow ( model ) . to receive ( :foreign_key_exists? ) . and_return ( true )
end
it 'removes the foreign key' do
expect ( model ) . to receive ( :remove_foreign_key ) . with ( :projects , :users , { column : :user_id } )
model . remove_foreign_key_if_exists ( :projects , :users , column : :user_id )
end
context 'when the target table is not given' do
it 'passes the options as the second parameter' do
expect ( model ) . to receive ( :remove_foreign_key ) . with ( :projects , { column : :user_id } )
model . remove_foreign_key_if_exists ( :projects , column : :user_id )
end
end
context 'when the reverse_lock_order option is given' do
it 'requests for lock before removing the foreign key' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect ( model ) . to receive ( :execute ) . with ( / LOCK TABLE users, projects / )
expect ( model ) . not_to receive ( :remove_foreign_key ) . with ( :projects , :users )
model . remove_foreign_key_if_exists ( :projects , :users , column : :user_id , reverse_lock_order : true )
end
context 'when not inside a transaction' do
it 'does not lock' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( false )
expect ( model ) . not_to receive ( :execute ) . with ( / LOCK TABLE users, projects / )
expect ( model ) . to receive ( :remove_foreign_key ) . with ( :projects , :users , { column : :user_id } )
model . remove_foreign_key_if_exists ( :projects , :users , column : :user_id , reverse_lock_order : true )
end
end
end
end
end
2017-08-17 22:00:37 +05:30
describe '#add_concurrent_foreign_key' do
2018-05-09 12:01:36 +05:30
before do
allow ( model ) . to receive ( :foreign_key_exists? ) . and_return ( false )
end
2017-08-17 22:00:37 +05:30
context 'inside a transaction' do
it 'raises an error' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect do
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id )
end . to raise_error ( RuntimeError )
end
end
context 'outside a transaction' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
2021-09-30 23:02:18 +05:30
context 'target column' do
it 'defaults to (id) when no custom target column is provided' do
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2021-09-30 23:02:18 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2021-09-30 23:02:18 +05:30
expect ( model ) . to receive ( :execute ) . with ( / REFERENCES users \ (id \ ) / )
model . add_concurrent_foreign_key ( :projects , :users ,
column : :user_id )
end
it 'references the custom taget column when provided' do
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2021-09-30 23:02:18 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2021-09-30 23:02:18 +05:30
expect ( model ) . to receive ( :execute ) . with ( / REFERENCES users \ (id_convert_to_bigint \ ) / )
model . add_concurrent_foreign_key ( :projects , :users ,
column : :user_id ,
target_column : :id_convert_to_bigint )
end
end
2020-01-01 13:55:28 +05:30
context 'ON DELETE statements' do
context 'on_delete: :nullify' do
it 'appends ON DELETE SET NULL statement' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / ON DELETE SET NULL / )
model . add_concurrent_foreign_key ( :projects , :users ,
column : :user_id ,
on_delete : :nullify )
end
end
2017-08-17 22:00:37 +05:30
2020-01-01 13:55:28 +05:30
context 'on_delete: :cascade' do
it 'appends ON DELETE CASCADE statement' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / ON DELETE CASCADE / )
model . add_concurrent_foreign_key ( :projects , :users ,
column : :user_id ,
on_delete : :cascade )
end
end
2017-08-17 22:00:37 +05:30
2020-01-01 13:55:28 +05:30
context 'on_delete: nil' do
it 'appends no ON DELETE statement' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2017-09-10 17:25:29 +05:30
2020-01-01 13:55:28 +05:30
expect ( model ) . not_to receive ( :execute ) . with ( / ON DELETE / )
model . add_concurrent_foreign_key ( :projects , :users ,
column : :user_id ,
on_delete : nil )
end
end
2019-10-12 21:52:04 +05:30
end
2018-05-09 12:01:36 +05:30
2020-01-01 13:55:28 +05:30
context 'when no custom key name is supplied' do
it 'creates a concurrent foreign key and validates it' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / NOT VALID / )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-01-01 13:55:28 +05:30
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id )
end
it 'does not create a foreign key if it exists already' do
name = model . concurrent_foreign_key_name ( :projects , :user_id )
expect ( model ) . to receive ( :foreign_key_exists? ) . with ( :projects , :users ,
column : :user_id ,
on_delete : :cascade ,
2021-09-30 23:02:18 +05:30
name : name ,
primary_key : :id ) . and_return ( true )
2020-01-01 13:55:28 +05:30
expect ( model ) . not_to receive ( :execute ) . with ( / ADD CONSTRAINT / )
expect ( model ) . to receive ( :execute ) . with ( / VALIDATE CONSTRAINT / )
2018-05-09 12:01:36 +05:30
2020-01-01 13:55:28 +05:30
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id )
end
2019-10-12 21:52:04 +05:30
end
2019-09-04 21:01:54 +05:30
2020-01-01 13:55:28 +05:30
context 'when a custom key name is supplied' do
context 'for creating a new foreign key for a column that does not presently exist' do
it 'creates a new foreign key' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / NOT VALID / )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT.+foo / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-01-01 13:55:28 +05:30
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id , name : :foo )
end
end
context 'for creating a duplicate foreign key for a column that presently exists' do
context 'when the supplied key name is the same as the existing foreign key name' do
it 'does not create a new foreign key' do
expect ( model ) . to receive ( :foreign_key_exists? ) . with ( :projects , :users ,
name : :foo ,
2021-09-30 23:02:18 +05:30
primary_key : :id ,
2020-01-01 13:55:28 +05:30
on_delete : :cascade ,
column : :user_id ) . and_return ( true )
expect ( model ) . not_to receive ( :execute ) . with ( / ADD CONSTRAINT / )
expect ( model ) . to receive ( :execute ) . with ( / VALIDATE CONSTRAINT / )
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id , name : :foo )
end
end
context 'when the supplied key name is different from the existing foreign key name' do
it 'creates a new foreign key' do
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / NOT VALID / )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT.+bar / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2019-09-04 21:01:54 +05:30
2020-01-01 13:55:28 +05:30
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id , name : :bar )
end
end
end
2017-08-17 22:00:37 +05:30
end
2020-03-13 15:44:24 +05:30
describe 'validate option' do
let ( :args ) { [ :projects , :users ] }
let ( :options ) { { column : :user_id , on_delete : nil } }
context 'when validate is supplied with a falsey value' do
it_behaves_like 'skips validation' , validate : false
it_behaves_like 'skips validation' , validate : nil
end
context 'when validate is supplied with a truthy value' do
it_behaves_like 'performs validation' , validate : true
it_behaves_like 'performs validation' , validate : :whatever
end
context 'when validate is not supplied' do
it_behaves_like 'performs validation' , { }
end
end
2021-10-27 15:23:28 +05:30
context 'when the reverse_lock_order flag is set' do
it 'explicitly locks the tables in target-source order' , :aggregate_failures do
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
expect ( model ) . to receive ( :execute ) . with ( 'LOCK TABLE users, projects IN SHARE ROW EXCLUSIVE MODE' )
expect ( model ) . to receive ( :execute ) . with ( / REFERENCES users \ (id \ ) / )
model . add_concurrent_foreign_key ( :projects , :users , column : :user_id , reverse_lock_order : true )
end
end
2020-03-13 15:44:24 +05:30
end
end
describe '#validate_foreign_key' do
context 'when name is provided' do
it 'does not infer the foreign key constraint name' do
expect ( model ) . to receive ( :foreign_key_exists? ) . with ( :projects , name : :foo ) . and_return ( true )
aggregate_failures do
expect ( model ) . not_to receive ( :concurrent_foreign_key_name )
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-03-13 15:44:24 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / ALTER TABLE projects VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-03-13 15:44:24 +05:30
end
model . validate_foreign_key ( :projects , :user_id , name : :foo )
end
end
context 'when name is not provided' do
it 'infers the foreign key constraint name' do
expect ( model ) . to receive ( :foreign_key_exists? ) . with ( :projects , name : anything ) . and_return ( true )
aggregate_failures do
expect ( model ) . to receive ( :concurrent_foreign_key_name )
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-03-13 15:44:24 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / ALTER TABLE projects VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-03-13 15:44:24 +05:30
end
model . validate_foreign_key ( :projects , :user_id )
end
context 'when the inferred foreign key constraint does not exist' do
it 'raises an error' do
expect ( model ) . to receive ( :foreign_key_exists? ) . and_return ( false )
2020-04-08 14:13:33 +05:30
error_message = / Could not find foreign key "fk_name" on table "projects" /
expect { model . validate_foreign_key ( :projects , :user_id , name : :fk_name ) } . to raise_error ( error_message )
2020-03-13 15:44:24 +05:30
end
end
2017-08-17 22:00:37 +05:30
end
end
describe '#concurrent_foreign_key_name' do
it 'returns the name for a foreign key' do
name = model . concurrent_foreign_key_name ( :this_is_a_very_long_table_name ,
:with_a_very_long_column_name )
expect ( name ) . to be_an_instance_of ( String )
expect ( name . length ) . to eq ( 13 )
end
end
2018-05-09 12:01:36 +05:30
describe '#foreign_key_exists?' do
before do
2021-09-30 23:02:18 +05:30
key = ActiveRecord :: ConnectionAdapters :: ForeignKeyDefinition . new (
:projects , :users ,
{
column : :non_standard_id ,
name : :fk_projects_users_non_standard_id ,
on_delete : :cascade ,
primary_key : :id
}
)
2018-05-09 12:01:36 +05:30
allow ( model ) . to receive ( :foreign_keys ) . with ( :projects ) . and_return ( [ key ] )
end
2020-01-01 13:55:28 +05:30
shared_examples_for 'foreign key checks' do
it 'finds existing foreign keys by column' do
expect ( model . foreign_key_exists? ( :projects , target_table , column : :non_standard_id ) ) . to be_truthy
end
it 'finds existing foreign keys by name' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :fk_projects_users_non_standard_id ) ) . to be_truthy
end
it 'finds existing foreign_keys by name and column' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :fk_projects_users_non_standard_id , column : :non_standard_id ) ) . to be_truthy
end
it 'finds existing foreign_keys by name, column and on_delete' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :fk_projects_users_non_standard_id , column : :non_standard_id , on_delete : :cascade ) ) . to be_truthy
end
it 'finds existing foreign keys by target table only' do
expect ( model . foreign_key_exists? ( :projects , target_table ) ) . to be_truthy
end
it 'compares by column name if given' do
expect ( model . foreign_key_exists? ( :projects , target_table , column : :user_id ) ) . to be_falsey
end
2021-09-30 23:02:18 +05:30
it 'compares by target column name if given' do
expect ( model . foreign_key_exists? ( :projects , target_table , primary_key : :user_id ) ) . to be_falsey
expect ( model . foreign_key_exists? ( :projects , target_table , primary_key : :id ) ) . to be_truthy
end
2020-01-01 13:55:28 +05:30
it 'compares by foreign key name if given' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :non_existent_foreign_key_name ) ) . to be_falsey
end
it 'compares by foreign key name and column if given' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :non_existent_foreign_key_name , column : :non_standard_id ) ) . to be_falsey
end
it 'compares by foreign key name, column and on_delete if given' do
expect ( model . foreign_key_exists? ( :projects , target_table , name : :fk_projects_users_non_standard_id , column : :non_standard_id , on_delete : :nullify ) ) . to be_falsey
end
2018-05-09 12:01:36 +05:30
end
2020-01-01 13:55:28 +05:30
context 'without specifying a target table' do
let ( :target_table ) { nil }
it_behaves_like 'foreign key checks'
2018-05-09 12:01:36 +05:30
end
2020-01-01 13:55:28 +05:30
context 'specifying a target table' do
let ( :target_table ) { :users }
it_behaves_like 'foreign key checks'
2018-05-09 12:01:36 +05:30
end
2020-01-01 13:55:28 +05:30
it 'compares by target table if no column given' do
2018-05-09 12:01:36 +05:30
expect ( model . foreign_key_exists? ( :projects , :other_table ) ) . to be_falsey
end
end
2017-08-17 22:00:37 +05:30
describe '#disable_statement_timeout' do
2019-10-12 21:52:04 +05:30
it 'disables statement timeouts to current transaction only' do
expect ( model ) . to receive ( :execute ) . with ( 'SET LOCAL statement_timeout TO 0' )
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
model . disable_statement_timeout
end
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
# this specs runs without an enclosing transaction (:delete truncation method for db_cleaner)
context 'with real environment' , :delete do
before do
model . execute ( " SET statement_timeout TO '20000' " )
2017-08-17 22:00:37 +05:30
end
2018-11-20 20:47:30 +05:30
2019-10-12 21:52:04 +05:30
after do
2021-10-27 15:23:28 +05:30
model . execute ( 'RESET statement_timeout' )
2019-10-12 21:52:04 +05:30
end
2018-11-20 20:47:30 +05:30
2019-10-12 21:52:04 +05:30
it 'defines statement to 0 only for current transaction' do
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '20s' )
2018-11-20 20:47:30 +05:30
2019-10-12 21:52:04 +05:30
model . connection . transaction do
model . disable_statement_timeout
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '0' )
2018-11-20 20:47:30 +05:30
end
2019-10-12 21:52:04 +05:30
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '20s' )
2018-11-20 20:47:30 +05:30
end
context 'when passing a blocks' do
it 'disables statement timeouts on session level and executes the block' do
expect ( model ) . to receive ( :execute ) . with ( 'SET statement_timeout TO 0' )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( 'RESET statement_timeout' ) . at_least ( :once )
2018-11-20 20:47:30 +05:30
expect { | block | model . disable_statement_timeout ( & block ) } . to yield_control
end
# this specs runs without an enclosing transaction (:delete truncation method for db_cleaner)
2019-10-12 21:52:04 +05:30
context 'with real environment' , :delete do
2018-11-20 20:47:30 +05:30
before do
model . execute ( " SET statement_timeout TO '20000' " )
end
after do
2021-10-27 15:23:28 +05:30
model . execute ( 'RESET statement_timeout' )
2018-11-20 20:47:30 +05:30
end
it 'defines statement to 0 for any code run inside the block' do
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '20s' )
model . disable_statement_timeout do
model . connection . transaction do
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '0' )
end
expect ( model . execute ( 'SHOW statement_timeout' ) . first [ 'statement_timeout' ] ) . to eq ( '0' )
end
end
end
end
2017-08-17 22:00:37 +05:30
end
2020-05-24 23:13:21 +05:30
# This spec runs without an enclosing transaction (:delete truncation method for db_cleaner)
context 'when the statement_timeout is already disabled' , :delete do
before do
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute ( 'SET statement_timeout TO 0' )
2020-05-24 23:13:21 +05:30
end
after do
2021-11-11 11:23:49 +05:30
# Use ActiveRecord::Migration.connection instead of model.execute
2020-05-24 23:13:21 +05:30
# so that this call is not counted below
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute ( 'RESET statement_timeout' )
2020-05-24 23:13:21 +05:30
end
it 'yields control without disabling the timeout or resetting' do
expect ( model ) . not_to receive ( :execute ) . with ( 'SET statement_timeout TO 0' )
2021-10-27 15:23:28 +05:30
expect ( model ) . not_to receive ( :execute ) . with ( 'RESET statement_timeout' )
2020-05-24 23:13:21 +05:30
expect { | block | model . disable_statement_timeout ( & block ) } . to yield_control
end
end
2017-08-17 22:00:37 +05:30
end
describe '#true_value' do
2019-10-12 21:52:04 +05:30
it 'returns the appropriate value' do
expect ( model . true_value ) . to eq ( " 't' " )
2017-08-17 22:00:37 +05:30
end
end
describe '#false_value' do
2019-10-12 21:52:04 +05:30
it 'returns the appropriate value' do
expect ( model . false_value ) . to eq ( " 'f' " )
2017-08-17 22:00:37 +05:30
end
end
2016-06-02 11:05:42 +05:30
describe '#update_column_in_batches' do
2017-09-10 17:25:29 +05:30
context 'when running outside of a transaction' do
before do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( false )
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
create_list ( :project , 5 )
end
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
it 'updates all the rows in a table' do
2019-09-30 21:07:59 +05:30
model . update_column_in_batches ( :projects , :description_html , 'foo' )
2017-09-10 17:25:29 +05:30
2019-09-30 21:07:59 +05:30
expect ( Project . where ( description_html : 'foo' ) . count ) . to eq ( 5 )
2017-09-10 17:25:29 +05:30
end
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
it 'updates boolean values correctly' do
model . update_column_in_batches ( :projects , :archived , true )
2016-06-02 11:05:42 +05:30
2017-09-10 17:25:29 +05:30
expect ( Project . where ( archived : true ) . count ) . to eq ( 5 )
end
context 'when a block is supplied' do
it 'yields an Arel table and query object to the supplied block' do
first_id = Project . first . id
model . update_column_in_batches ( :projects , :archived , true ) do | t , query |
query . where ( t [ :id ] . eq ( first_id ) )
end
expect ( Project . where ( archived : true ) . count ) . to eq ( 1 )
end
end
2016-06-22 15:30:34 +05:30
2017-09-10 17:25:29 +05:30
context 'when the value is Arel.sql (Arel::Nodes::SqlLiteral)' do
it 'updates the value as a SQL expression' do
model . update_column_in_batches ( :projects , :star_count , Arel . sql ( '1+1' ) )
2016-06-22 15:30:34 +05:30
2017-09-10 17:25:29 +05:30
expect ( Project . sum ( :star_count ) ) . to eq ( 2 * Project . count )
2016-06-22 15:30:34 +05:30
end
2017-09-10 17:25:29 +05:30
end
end
2016-06-22 15:30:34 +05:30
2017-09-10 17:25:29 +05:30
context 'when running inside the transaction' do
it 'raises RuntimeError' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect do
model . update_column_in_batches ( :projects , :star_count , Arel . sql ( '1+1' ) )
end . to raise_error ( RuntimeError )
2016-06-22 15:30:34 +05:30
end
end
2016-06-02 11:05:42 +05:30
end
describe '#add_column_with_default' do
2020-04-08 14:13:33 +05:30
let ( :column ) { Project . columns . find { | c | c . name == " id " } }
2020-05-24 23:13:21 +05:30
it 'delegates to #add_column' do
expect ( model ) . to receive ( :add_column ) . with ( :projects , :foo , :integer , default : 10 , limit : nil , null : true )
2020-04-22 19:07:51 +05:30
2020-05-24 23:13:21 +05:30
model . add_column_with_default ( :projects , :foo , :integer ,
default : 10 ,
allow_null : true )
2016-06-02 11:05:42 +05:30
end
end
2017-08-17 22:00:37 +05:30
describe '#rename_column_concurrently' do
context 'in a transaction' do
it 'raises RuntimeError' do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( true )
2017-09-10 17:25:29 +05:30
expect { model . rename_column_concurrently ( :users , :old , :new ) }
. to raise_error ( RuntimeError )
2017-08-17 22:00:37 +05:30
end
end
context 'outside a transaction' do
let ( :old_column ) do
double ( :column ,
type : :integer ,
limit : 8 ,
default : 0 ,
null : false ,
precision : 5 ,
scale : 1 )
end
let ( :trigger_name ) { model . rename_trigger_name ( :users , :old , :new ) }
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
2020-04-08 14:13:33 +05:30
context 'when the column to rename exists' do
before do
allow ( model ) . to receive ( :column_for ) . and_return ( old_column )
end
2018-03-17 18:26:18 +05:30
2020-04-08 14:13:33 +05:30
it 'renames a column concurrently' do
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2017-08-17 22:00:37 +05:30
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers )
2021-04-29 21:17:54 +05:30
. with ( :users , :old , :new )
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
expect ( model ) . to receive ( :add_column )
. with ( :users , :new , :integer ,
limit : old_column . limit ,
precision : old_column . precision ,
scale : old_column . scale )
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
expect ( model ) . to receive ( :change_column_default )
. with ( :users , :new , old_column . default )
expect ( model ) . to receive ( :update_column_in_batches )
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :add_not_null_constraint ) . with ( :users , :new )
2020-04-08 14:13:33 +05:30
expect ( model ) . to receive ( :copy_indexes ) . with ( :users , :old , :new )
expect ( model ) . to receive ( :copy_foreign_keys ) . with ( :users , :old , :new )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :copy_check_constraints ) . with ( :users , :old , :new )
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
model . rename_column_concurrently ( :users , :old , :new )
end
2017-08-17 22:00:37 +05:30
2020-07-28 23:09:34 +05:30
context 'with existing records and type casting' do
let ( :trigger_name ) { model . rename_trigger_name ( :users , :id , :new ) }
let ( :user ) { create ( :user ) }
2021-04-29 21:17:54 +05:30
let ( :copy_trigger ) { double ( 'copy trigger' ) }
2021-11-11 11:23:49 +05:30
let ( :connection ) { ActiveRecord :: Migration . connection }
2021-04-29 21:17:54 +05:30
before do
expect ( Gitlab :: Database :: UnidirectionalCopyTrigger ) . to receive ( :on_table )
2021-11-11 11:23:49 +05:30
. with ( :users , connection : connection ) . and_return ( copy_trigger )
2021-04-29 21:17:54 +05:30
end
2020-07-28 23:09:34 +05:30
it 'copies the value to the new column using the type_cast_function' , :aggregate_failures do
expect ( model ) . to receive ( :copy_indexes ) . with ( :users , :id , :new )
expect ( model ) . to receive ( :add_not_null_constraint ) . with ( :users , :new )
expect ( model ) . to receive ( :execute ) . with ( " UPDATE \" users \" SET \" new \" = cast_to_jsonb_with_default( \" users \" . \" id \" ) WHERE \" users \" . \" id \" >= #{ user . id } " )
2021-04-29 21:17:54 +05:30
expect ( copy_trigger ) . to receive ( :create ) . with ( :id , :new , trigger_name : nil )
2020-07-28 23:09:34 +05:30
model . rename_column_concurrently ( :users , :id , :new , type_cast_function : 'cast_to_jsonb_with_default' )
end
end
2020-05-24 23:13:21 +05:30
it 'passes the batch_column_name' do
expect ( model ) . to receive ( :column_exists? ) . with ( :users , :other_batch_column ) . and_return ( true )
expect ( model ) . to receive ( :check_trigger_permissions! ) . and_return ( true )
expect ( model ) . to receive ( :create_column_from ) . with (
2020-07-28 23:09:34 +05:30
:users , :old , :new , type : nil , batch_column_name : :other_batch_column , type_cast_function : nil
2020-05-24 23:13:21 +05:30
) . and_return ( true )
expect ( model ) . to receive ( :install_rename_triggers ) . and_return ( true )
model . rename_column_concurrently ( :users , :old , :new , batch_column_name : :other_batch_column )
end
2020-07-28 23:09:34 +05:30
it 'passes the type_cast_function' do
expect ( model ) . to receive ( :create_column_from ) . with (
:users , :old , :new , type : nil , batch_column_name : :id , type_cast_function : 'JSON'
) . and_return ( true )
model . rename_column_concurrently ( :users , :old , :new , type_cast_function : 'JSON' )
end
2020-05-24 23:13:21 +05:30
it 'raises an error with invalid batch_column_name' do
expect do
model . rename_column_concurrently ( :users , :old , :new , batch_column_name : :invalid )
end . to raise_error ( RuntimeError , / Column invalid does not exist on users / )
end
2020-04-08 14:13:33 +05:30
context 'when default is false' do
let ( :old_column ) do
double ( :column ,
type : :boolean ,
limit : nil ,
default : false ,
null : false ,
precision : nil ,
scale : nil )
end
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
it 'copies the default to the new column' do
expect ( model ) . to receive ( :change_column_default )
. with ( :users , :new , old_column . default )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :copy_check_constraints )
. with ( :users , :old , :new )
2020-04-08 14:13:33 +05:30
model . rename_column_concurrently ( :users , :old , :new )
end
end
2017-08-17 22:00:37 +05:30
end
2020-04-08 14:13:33 +05:30
context 'when the column to be renamed does not exist' do
before do
allow ( model ) . to receive ( :columns ) . and_return ( [ ] )
2019-10-12 21:52:04 +05:30
end
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
it 'raises an error with appropriate message' do
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2017-08-17 22:00:37 +05:30
2020-04-08 14:13:33 +05:30
error_message = / Could not find column "missing_column" on table "users" /
expect { model . rename_column_concurrently ( :users , :missing_column , :new ) } . to raise_error ( error_message )
2017-08-17 22:00:37 +05:30
end
end
end
end
2019-10-12 21:52:04 +05:30
describe '#undo_rename_column_concurrently' do
it 'reverses the operations of rename_column_concurrently' do
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :remove_rename_triggers )
2019-10-12 21:52:04 +05:30
. with ( :users , / trigger_.{12} / )
expect ( model ) . to receive ( :remove_column ) . with ( :users , :new )
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
model . undo_rename_column_concurrently ( :users , :old , :new )
end
end
describe '#cleanup_concurrent_column_rename' do
it 'cleans up the renaming procedure' do
2018-03-17 18:26:18 +05:30
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :remove_rename_triggers )
2017-09-10 17:25:29 +05:30
. with ( :users , / trigger_.{12} / )
2017-08-17 22:00:37 +05:30
expect ( model ) . to receive ( :remove_column ) . with ( :users , :old )
model . cleanup_concurrent_column_rename ( :users , :old , :new )
end
2019-10-12 21:52:04 +05:30
end
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
describe '#undo_cleanup_concurrent_column_rename' do
context 'in a transaction' do
it 'raises RuntimeError' do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( true )
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
expect { model . undo_cleanup_concurrent_column_rename ( :users , :old , :new ) }
. to raise_error ( RuntimeError )
end
end
2018-03-17 18:26:18 +05:30
2019-10-12 21:52:04 +05:30
context 'outside a transaction' do
let ( :new_column ) do
double ( :column ,
type : :integer ,
limit : 8 ,
default : 0 ,
null : false ,
precision : 5 ,
scale : 1 )
end
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
let ( :trigger_name ) { model . rename_trigger_name ( :users , :old , :new ) }
2017-08-17 22:00:37 +05:30
2019-10-12 21:52:04 +05:30
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
allow ( model ) . to receive ( :column_for ) . and_return ( new_column )
end
it 'reverses the operations of cleanup_concurrent_column_rename' do
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers )
2021-04-29 21:17:54 +05:30
. with ( :users , :old , :new )
2019-10-12 21:52:04 +05:30
expect ( model ) . to receive ( :add_column )
. with ( :users , :old , :integer ,
limit : new_column . limit ,
precision : new_column . precision ,
scale : new_column . scale )
expect ( model ) . to receive ( :change_column_default )
. with ( :users , :old , new_column . default )
expect ( model ) . to receive ( :update_column_in_batches )
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :add_not_null_constraint ) . with ( :users , :old )
2019-10-12 21:52:04 +05:30
expect ( model ) . to receive ( :copy_indexes ) . with ( :users , :new , :old )
expect ( model ) . to receive ( :copy_foreign_keys ) . with ( :users , :new , :old )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :copy_check_constraints ) . with ( :users , :new , :old )
2019-10-12 21:52:04 +05:30
model . undo_cleanup_concurrent_column_rename ( :users , :old , :new )
end
2020-05-24 23:13:21 +05:30
it 'passes the batch_column_name' do
expect ( model ) . to receive ( :column_exists? ) . with ( :users , :other_batch_column ) . and_return ( true )
expect ( model ) . to receive ( :check_trigger_permissions! ) . and_return ( true )
expect ( model ) . to receive ( :create_column_from ) . with (
:users , :new , :old , type : nil , batch_column_name : :other_batch_column
) . and_return ( true )
expect ( model ) . to receive ( :install_rename_triggers ) . and_return ( true )
model . undo_cleanup_concurrent_column_rename ( :users , :old , :new , batch_column_name : :other_batch_column )
end
it 'raises an error with invalid batch_column_name' do
expect do
model . undo_cleanup_concurrent_column_rename ( :users , :old , :new , batch_column_name : :invalid )
end . to raise_error ( RuntimeError , / Column invalid does not exist on users / )
end
2019-10-12 21:52:04 +05:30
context 'when default is false' do
let ( :new_column ) do
double ( :column ,
type : :boolean ,
limit : nil ,
default : false ,
null : false ,
precision : nil ,
scale : nil )
end
it 'copies the default to the old column' do
expect ( model ) . to receive ( :change_column_default )
. with ( :users , :old , new_column . default )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :copy_check_constraints )
. with ( :users , :new , :old )
2019-10-12 21:52:04 +05:30
model . undo_cleanup_concurrent_column_rename ( :users , :old , :new )
end
end
2017-08-17 22:00:37 +05:30
end
end
describe '#change_column_type_concurrently' do
it 'changes the column type' do
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :rename_column_concurrently )
2020-11-24 15:15:51 +05:30
. with ( 'users' , 'username' , 'username_for_type_change' , type : :text , type_cast_function : nil , batch_column_name : :id )
2017-08-17 22:00:37 +05:30
model . change_column_type_concurrently ( 'users' , 'username' , :text )
end
2020-07-28 23:09:34 +05:30
2020-11-24 15:15:51 +05:30
it 'passed the batch column name' do
expect ( model ) . to receive ( :rename_column_concurrently )
. with ( 'users' , 'username' , 'username_for_type_change' , type : :text , type_cast_function : nil , batch_column_name : :user_id )
model . change_column_type_concurrently ( 'users' , 'username' , :text , batch_column_name : :user_id )
end
2020-07-28 23:09:34 +05:30
context 'with type cast' do
it 'changes the column type with casting the value to the new type' do
expect ( model ) . to receive ( :rename_column_concurrently )
2020-11-24 15:15:51 +05:30
. with ( 'users' , 'username' , 'username_for_type_change' , type : :text , type_cast_function : 'JSON' , batch_column_name : :id )
2020-07-28 23:09:34 +05:30
model . change_column_type_concurrently ( 'users' , 'username' , :text , type_cast_function : 'JSON' )
end
end
2017-08-17 22:00:37 +05:30
end
2021-01-03 14:25:43 +05:30
describe '#undo_change_column_type_concurrently' do
it 'reverses the operations of change_column_type_concurrently' do
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :remove_rename_triggers )
2021-01-03 14:25:43 +05:30
. with ( :users , / trigger_.{12} / )
expect ( model ) . to receive ( :remove_column ) . with ( :users , " old_for_type_change " )
model . undo_change_column_type_concurrently ( :users , :old )
end
end
2017-08-17 22:00:37 +05:30
describe '#cleanup_concurrent_column_type_change' do
it 'cleans up the type changing procedure' do
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :cleanup_concurrent_column_rename )
. with ( 'users' , 'username' , 'username_for_type_change' )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :rename_column )
. with ( 'users' , 'username_for_type_change' , 'username' )
2017-08-17 22:00:37 +05:30
model . cleanup_concurrent_column_type_change ( 'users' , 'username' )
end
end
2021-01-03 14:25:43 +05:30
describe '#undo_cleanup_concurrent_column_type_change' do
context 'in a transaction' do
it 'raises RuntimeError' do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect { model . undo_cleanup_concurrent_column_type_change ( :users , :old , :new ) }
. to raise_error ( RuntimeError )
end
end
context 'outside a transaction' do
let ( :temp_column ) { " old_for_type_change " }
let ( :temp_undo_cleanup_column ) do
identifier = " users_old_for_type_change "
hashed_identifier = Digest :: SHA256 . hexdigest ( identifier ) . first ( 10 )
" tmp_undo_cleanup_column_ #{ hashed_identifier } "
end
let ( :trigger_name ) { model . rename_trigger_name ( :users , :old , :old_for_type_change ) }
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
it 'reverses the operations of cleanup_concurrent_column_type_change' do
2022-06-21 17:19:12 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_ddl_mode! )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
expect ( model ) . to receive ( :create_column_from ) . with (
:users ,
:old ,
temp_undo_cleanup_column ,
type : :string ,
batch_column_name : :id ,
2021-02-22 17:27:13 +05:30
type_cast_function : nil ,
limit : nil
2021-01-03 14:25:43 +05:30
) . and_return ( true )
expect ( model ) . to receive ( :rename_column )
. with ( :users , :old , temp_column )
expect ( model ) . to receive ( :rename_column )
. with ( :users , temp_undo_cleanup_column , :old )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers )
2021-04-29 21:17:54 +05:30
. with ( :users , :old , 'old_for_type_change' )
2021-01-03 14:25:43 +05:30
model . undo_cleanup_concurrent_column_type_change ( :users , :old , :string )
end
2021-02-22 17:27:13 +05:30
it 'passes the type_cast_function, batch_column_name and limit' do
2022-06-21 17:19:12 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_ddl_mode! )
2021-01-03 14:25:43 +05:30
expect ( model ) . to receive ( :column_exists? ) . with ( :users , :other_batch_column ) . and_return ( true )
expect ( model ) . to receive ( :check_trigger_permissions! ) . with ( :users )
expect ( model ) . to receive ( :create_column_from ) . with (
:users ,
:old ,
temp_undo_cleanup_column ,
type : :string ,
batch_column_name : :other_batch_column ,
2021-02-22 17:27:13 +05:30
type_cast_function : :custom_type_cast_function ,
limit : 8
2021-01-03 14:25:43 +05:30
) . and_return ( true )
expect ( model ) . to receive ( :rename_column )
. with ( :users , :old , temp_column )
expect ( model ) . to receive ( :rename_column )
. with ( :users , temp_undo_cleanup_column , :old )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers )
2021-04-29 21:17:54 +05:30
. with ( :users , :old , 'old_for_type_change' )
2021-01-03 14:25:43 +05:30
model . undo_cleanup_concurrent_column_type_change (
:users ,
:old ,
:string ,
type_cast_function : :custom_type_cast_function ,
2021-02-22 17:27:13 +05:30
batch_column_name : :other_batch_column ,
limit : 8
2021-01-03 14:25:43 +05:30
)
end
it 'raises an error with invalid batch_column_name' do
expect do
model . undo_cleanup_concurrent_column_type_change ( :users , :old , :new , batch_column_name : :invalid )
end . to raise_error ( RuntimeError , / Column invalid does not exist on users / )
end
end
end
2021-06-08 01:23:25 +05:30
describe '#install_rename_triggers' do
2021-11-11 11:23:49 +05:30
let ( :connection ) { ActiveRecord :: Migration . connection }
2021-06-08 01:23:25 +05:30
it 'installs the triggers' do
2021-04-29 21:17:54 +05:30
copy_trigger = double ( 'copy trigger' )
2017-08-17 22:00:37 +05:30
2021-04-29 21:17:54 +05:30
expect ( Gitlab :: Database :: UnidirectionalCopyTrigger ) . to receive ( :on_table )
2021-11-11 11:23:49 +05:30
. with ( :users , connection : connection ) . and_return ( copy_trigger )
2019-12-04 20:38:33 +05:30
2021-04-29 21:17:54 +05:30
expect ( copy_trigger ) . to receive ( :create ) . with ( :old , :new , trigger_name : 'foo' )
2017-08-17 22:00:37 +05:30
2021-06-08 01:23:25 +05:30
model . install_rename_triggers ( :users , :old , :new , trigger_name : 'foo' )
2019-12-04 20:38:33 +05:30
end
2017-08-17 22:00:37 +05:30
end
2021-06-08 01:23:25 +05:30
describe '#remove_rename_triggers' do
2021-11-11 11:23:49 +05:30
let ( :connection ) { ActiveRecord :: Migration . connection }
2017-08-17 22:00:37 +05:30
it 'removes the function and trigger' do
2021-04-29 21:17:54 +05:30
copy_trigger = double ( 'copy trigger' )
expect ( Gitlab :: Database :: UnidirectionalCopyTrigger ) . to receive ( :on_table )
2021-11-11 11:23:49 +05:30
. with ( 'bar' , connection : connection ) . and_return ( copy_trigger )
2021-04-29 21:17:54 +05:30
expect ( copy_trigger ) . to receive ( :drop ) . with ( 'foo' )
2017-08-17 22:00:37 +05:30
2021-06-08 01:23:25 +05:30
model . remove_rename_triggers ( 'bar' , 'foo' )
2017-08-17 22:00:37 +05:30
end
end
describe '#rename_trigger_name' do
it 'returns a String' do
2017-09-10 17:25:29 +05:30
expect ( model . rename_trigger_name ( :users , :foo , :bar ) )
. to match ( / trigger_.{12} / )
2017-08-17 22:00:37 +05:30
end
end
describe '#indexes_for' do
it 'returns the indexes for a column' do
idx1 = double ( :idx , columns : %w( project_id ) )
idx2 = double ( :idx , columns : %w( user_id ) )
allow ( model ) . to receive ( :indexes ) . with ( 'table' ) . and_return ( [ idx1 , idx2 ] )
expect ( model . indexes_for ( 'table' , :user_id ) ) . to eq ( [ idx2 ] )
end
end
describe '#foreign_keys_for' do
it 'returns the foreign keys for a column' do
fk1 = double ( :fk , column : 'project_id' )
fk2 = double ( :fk , column : 'user_id' )
allow ( model ) . to receive ( :foreign_keys ) . with ( 'table' ) . and_return ( [ fk1 , fk2 ] )
expect ( model . foreign_keys_for ( 'table' , :user_id ) ) . to eq ( [ fk2 ] )
end
end
describe '#copy_indexes' do
context 'using a regular index using a single column' do
it 'copies the index' do
index = double ( :index ,
columns : %w( project_id ) ,
name : 'index_on_issues_project_id' ,
using : nil ,
where : nil ,
opclasses : { } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
2017-08-17 22:00:37 +05:30
%w( gl_project_id ) ,
2022-07-16 23:28:13 +05:30
{
2017-08-17 22:00:37 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id' ,
length : [ ] ,
2022-07-16 23:28:13 +05:30
order : [ ]
} )
2017-08-17 22:00:37 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using a regular index with multiple columns' do
it 'copies the index' do
index = double ( :index ,
columns : %w( project_id foobar ) ,
name : 'index_on_issues_project_id_foobar' ,
using : nil ,
where : nil ,
opclasses : { } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
2017-08-17 22:00:37 +05:30
%w( gl_project_id foobar ) ,
2022-07-16 23:28:13 +05:30
{
2017-08-17 22:00:37 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id_foobar' ,
length : [ ] ,
2022-07-16 23:28:13 +05:30
order : [ ]
} )
2017-08-17 22:00:37 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using an index with a WHERE clause' do
it 'copies the index' do
index = double ( :index ,
columns : %w( project_id ) ,
name : 'index_on_issues_project_id' ,
using : nil ,
where : 'foo' ,
opclasses : { } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
2017-08-17 22:00:37 +05:30
%w( gl_project_id ) ,
2022-07-16 23:28:13 +05:30
{
2017-08-17 22:00:37 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id' ,
length : [ ] ,
order : [ ] ,
2022-07-16 23:28:13 +05:30
where : 'foo'
} )
2017-08-17 22:00:37 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using an index with a USING clause' do
it 'copies the index' do
index = double ( :index ,
columns : %w( project_id ) ,
name : 'index_on_issues_project_id' ,
where : nil ,
using : 'foo' ,
opclasses : { } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
2017-08-17 22:00:37 +05:30
%w( gl_project_id ) ,
2022-07-16 23:28:13 +05:30
{
2017-08-17 22:00:37 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id' ,
length : [ ] ,
order : [ ] ,
2022-07-16 23:28:13 +05:30
using : 'foo'
} )
2017-08-17 22:00:37 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using an index with custom operator classes' do
it 'copies the index' do
index = double ( :index ,
columns : %w( project_id ) ,
name : 'index_on_issues_project_id' ,
using : nil ,
where : nil ,
opclasses : { 'project_id' = > 'bar' } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
2017-08-17 22:00:37 +05:30
%w( gl_project_id ) ,
2022-07-16 23:28:13 +05:30
{
2017-08-17 22:00:37 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id' ,
length : [ ] ,
order : [ ] ,
2022-07-16 23:28:13 +05:30
opclass : { 'gl_project_id' = > 'bar' }
} )
2021-01-03 14:25:43 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using an index with multiple columns and custom operator classes' do
it 'copies the index' do
index = double ( :index ,
2022-07-16 23:28:13 +05:30
{
columns : %w( project_id foobar ) ,
name : 'index_on_issues_project_id_foobar' ,
using : :gin ,
where : nil ,
opclasses : { 'project_id' = > 'bar' , 'foobar' = > :gin_trgm_ops } ,
unique : false ,
lengths : [ ] ,
orders : [ ]
} )
2021-01-03 14:25:43 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
%w( gl_project_id foobar ) ,
2022-07-16 23:28:13 +05:30
{
2021-01-03 14:25:43 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id_foobar' ,
length : [ ] ,
order : [ ] ,
opclass : { 'gl_project_id' = > 'bar' , 'foobar' = > :gin_trgm_ops } ,
2022-07-16 23:28:13 +05:30
using : :gin
} )
2021-01-03 14:25:43 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
context 'using an index with multiple columns and a custom operator class on the non affected column' do
it 'copies the index' do
index = double ( :index ,
2022-07-16 23:28:13 +05:30
{
columns : %w( project_id foobar ) ,
name : 'index_on_issues_project_id_foobar' ,
using : :gin ,
where : nil ,
opclasses : { 'foobar' = > :gin_trgm_ops } ,
unique : false ,
lengths : [ ] ,
orders : [ ]
} )
2021-01-03 14:25:43 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
expect ( model ) . to receive ( :add_concurrent_index )
. with ( :issues ,
%w( gl_project_id foobar ) ,
2022-07-16 23:28:13 +05:30
{
2021-01-03 14:25:43 +05:30
unique : false ,
name : 'index_on_issues_gl_project_id_foobar' ,
length : [ ] ,
order : [ ] ,
opclass : { 'foobar' = > :gin_trgm_ops } ,
2022-07-16 23:28:13 +05:30
using : :gin
} )
2017-08-17 22:00:37 +05:30
model . copy_indexes ( :issues , :project_id , :gl_project_id )
end
end
describe 'using an index of which the name does not contain the source column' do
it 'raises RuntimeError' do
index = double ( :index ,
columns : %w( project_id ) ,
name : 'index_foobar_index' ,
using : nil ,
where : nil ,
opclasses : { } ,
unique : false ,
lengths : [ ] ,
orders : [ ] )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :indexes_for ) . with ( :issues , 'project_id' )
. and_return ( [ index ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect { model . copy_indexes ( :issues , :project_id , :gl_project_id ) }
. to raise_error ( RuntimeError )
2017-08-17 22:00:37 +05:30
end
end
end
describe '#copy_foreign_keys' do
it 'copies foreign keys from one column to another' do
fk = double ( :fk ,
from_table : 'issues' ,
to_table : 'projects' ,
on_delete : :cascade )
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :foreign_keys_for ) . with ( :issues , :project_id )
. and_return ( [ fk ] )
2017-08-17 22:00:37 +05:30
2017-09-10 17:25:29 +05:30
expect ( model ) . to receive ( :add_concurrent_foreign_key )
. with ( 'issues' , 'projects' , column : :gl_project_id , on_delete : :cascade )
2017-08-17 22:00:37 +05:30
model . copy_foreign_keys ( :issues , :project_id , :gl_project_id )
end
end
describe '#column_for' do
it 'returns a column object for an existing column' do
column = model . column_for ( :users , :id )
expect ( column . name ) . to eq ( 'id' )
end
2020-04-08 14:13:33 +05:30
it 'raises an error when a column does not exist' do
error_message = / Could not find column "kittens" on table "users" /
expect { model . column_for ( :users , :kittens ) } . to raise_error ( error_message )
2017-08-17 22:00:37 +05:30
end
end
describe '#replace_sql' do
2019-10-12 21:52:04 +05:30
it 'builds the sql with correct functions' do
expect ( model . replace_sql ( Arel :: Table . new ( :users ) [ :first_name ] , " Alice " , " Eve " ) . to_s )
. to include ( 'regexp_replace' )
2017-08-17 22:00:37 +05:30
end
describe 'results' do
let! ( :user ) { create ( :user , name : 'Kathy Alice Aliceson' ) }
it 'replaces the correct part of the string' do
2017-09-10 17:25:29 +05:30
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
query = model . replace_sql ( Arel :: Table . new ( :users ) [ :name ] , 'Alice' , 'Eve' )
model . update_column_in_batches ( :users , :name , query )
2017-08-17 22:00:37 +05:30
expect ( user . reload . name ) . to eq ( 'Kathy Eve Aliceson' )
end
end
end
2018-03-17 18:26:18 +05:30
2020-03-13 15:44:24 +05:30
describe 'sidekiq migration helpers' , :redis do
2018-03-17 18:26:18 +05:30
let ( :worker ) do
Class . new do
include Sidekiq :: Worker
2021-11-18 22:05:49 +05:30
2018-03-17 18:26:18 +05:30
sidekiq_options queue : 'test'
2021-11-18 22:05:49 +05:30
def self . name
'WorkerClass'
end
2018-03-17 18:26:18 +05:30
end
end
2021-11-18 22:05:49 +05:30
before do
stub_const ( worker . name , worker )
end
2018-03-17 18:26:18 +05:30
describe '#sidekiq_queue_length' do
context 'when queue is empty' do
it 'returns zero' do
Sidekiq :: Testing . disable! do
expect ( model . sidekiq_queue_length ( 'test' ) ) . to eq 0
end
end
end
context 'when queue contains jobs' do
it 'returns correct size of the queue' do
Sidekiq :: Testing . disable! do
worker . perform_async ( 'Something' , [ 1 ] )
worker . perform_async ( 'Something' , [ 2 ] )
expect ( model . sidekiq_queue_length ( 'test' ) ) . to eq 2
end
end
end
end
describe '#migrate_sidekiq_queue' do
it 'migrates jobs from one sidekiq queue to another' do
Sidekiq :: Testing . disable! do
worker . perform_async ( 'Something' , [ 1 ] )
worker . perform_async ( 'Something' , [ 2 ] )
expect ( model . sidekiq_queue_length ( 'test' ) ) . to eq 2
expect ( model . sidekiq_queue_length ( 'new_test' ) ) . to eq 0
model . sidekiq_queue_migrate ( 'test' , to : 'new_test' )
expect ( model . sidekiq_queue_length ( 'test' ) ) . to eq 0
expect ( model . sidekiq_queue_length ( 'new_test' ) ) . to eq 2
end
end
end
end
describe '#check_trigger_permissions!' do
it 'does nothing when the user has the correct permissions' do
expect { model . check_trigger_permissions! ( 'users' ) }
. not_to raise_error
end
it 'raises RuntimeError when the user does not have the correct permissions' do
allow ( Gitlab :: Database :: Grant ) . to receive ( :create_and_execute_trigger? )
. with ( 'kittens' )
. and_return ( false )
expect { model . check_trigger_permissions! ( 'kittens' ) }
. to raise_error ( RuntimeError , / Your database user is not allowed / )
end
end
2021-06-08 01:23:25 +05:30
describe '#convert_to_bigint_column' do
it 'returns the name of the temporary column used to convert to bigint' do
expect ( model . convert_to_bigint_column ( :id ) ) . to eq ( 'id_convert_to_bigint' )
end
end
2021-03-08 18:12:59 +05:30
describe '#initialize_conversion_of_integer_to_bigint' do
2021-04-29 21:17:54 +05:30
let ( :table ) { :test_table }
let ( :column ) { :id }
2021-06-08 01:23:25 +05:30
let ( :tmp_column ) { model . convert_to_bigint_column ( column ) }
2021-04-29 21:17:54 +05:30
before do
model . create_table table , id : false do | t |
t . integer :id , primary_key : true
t . integer :non_nullable_column , null : false
t . integer :nullable_column
t . timestamps
end
2021-03-08 18:12:59 +05:30
end
2021-04-29 21:17:54 +05:30
context 'when the target table does not exist' do
it 'raises an error' do
expect { model . initialize_conversion_of_integer_to_bigint ( :this_table_is_not_real , column ) }
. to raise_error ( 'Table this_table_is_not_real does not exist' )
end
end
2021-03-08 18:12:59 +05:30
2021-04-29 21:17:54 +05:30
context 'when the primary key does not exist' do
it 'raises an error' do
expect { model . initialize_conversion_of_integer_to_bigint ( table , column , primary_key : :foobar ) }
. to raise_error ( " Column foobar does not exist on #{ table } " )
2021-03-08 18:12:59 +05:30
end
end
2021-06-08 01:23:25 +05:30
context 'when the column to migrate does not exist' do
2021-04-29 21:17:54 +05:30
it 'raises an error' do
2021-06-08 01:23:25 +05:30
expect { model . initialize_conversion_of_integer_to_bigint ( table , :this_column_is_not_real ) }
. to raise_error ( ArgumentError , " Column this_column_is_not_real does not exist on #{ table } " )
2021-04-29 21:17:54 +05:30
end
end
context 'when the column to convert is the primary key' do
it 'creates a not-null bigint column and installs triggers' do
expect ( model ) . to receive ( :add_column ) . with ( table , tmp_column , :bigint , default : 0 , null : false )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , [ column ] , [ tmp_column ] )
2021-04-29 21:17:54 +05:30
model . initialize_conversion_of_integer_to_bigint ( table , column )
end
end
context 'when the column to convert is not the primary key, but non-nullable' do
let ( :column ) { :non_nullable_column }
it 'creates a not-null bigint column and installs triggers' do
expect ( model ) . to receive ( :add_column ) . with ( table , tmp_column , :bigint , default : 0 , null : false )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , [ column ] , [ tmp_column ] )
2021-04-29 21:17:54 +05:30
model . initialize_conversion_of_integer_to_bigint ( table , column )
end
end
context 'when the column to convert is not the primary key, but nullable' do
let ( :column ) { :nullable_column }
it 'creates a nullable bigint column and installs triggers' do
expect ( model ) . to receive ( :add_column ) . with ( table , tmp_column , :bigint , default : nil )
2021-06-08 01:23:25 +05:30
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , [ column ] , [ tmp_column ] )
2021-04-29 21:17:54 +05:30
model . initialize_conversion_of_integer_to_bigint ( table , column )
end
end
2021-06-08 01:23:25 +05:30
context 'when multiple columns are given' do
it 'creates the correct columns and installs the trigger' do
columns_to_convert = % i [ id non_nullable_column nullable_column ]
temporary_columns = columns_to_convert . map { | column | model . convert_to_bigint_column ( column ) }
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_columns [ 0 ] , :bigint , default : 0 , null : false )
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_columns [ 1 ] , :bigint , default : 0 , null : false )
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_columns [ 2 ] , :bigint , default : nil )
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , columns_to_convert , temporary_columns )
model . initialize_conversion_of_integer_to_bigint ( table , columns_to_convert )
end
end
end
2021-11-11 11:23:49 +05:30
describe '#restore_conversion_of_integer_to_bigint' do
let ( :table ) { :test_table }
let ( :column ) { :id }
let ( :tmp_column ) { model . convert_to_bigint_column ( column ) }
before do
model . create_table table , id : false do | t |
t . bigint :id , primary_key : true
t . bigint :build_id , null : false
t . timestamps
end
end
context 'when the target table does not exist' do
it 'raises an error' do
expect { model . restore_conversion_of_integer_to_bigint ( :this_table_is_not_real , column ) }
. to raise_error ( 'Table this_table_is_not_real does not exist' )
end
end
context 'when the column to migrate does not exist' do
it 'raises an error' do
expect { model . restore_conversion_of_integer_to_bigint ( table , :this_column_is_not_real ) }
. to raise_error ( ArgumentError , " Column this_column_is_not_real does not exist on #{ table } " )
end
end
context 'when a single column is given' do
let ( :column_to_convert ) { 'id' }
let ( :temporary_column ) { model . convert_to_bigint_column ( column_to_convert ) }
it 'creates the correct columns and installs the trigger' do
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_column , :int , default : 0 , null : false )
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , [ column_to_convert ] , [ temporary_column ] )
model . restore_conversion_of_integer_to_bigint ( table , column_to_convert )
end
end
context 'when multiple columns are given' do
let ( :columns_to_convert ) { % i [ id build_id ] }
let ( :temporary_columns ) { columns_to_convert . map { | column | model . convert_to_bigint_column ( column ) } }
it 'creates the correct columns and installs the trigger' do
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_columns [ 0 ] , :int , default : 0 , null : false )
expect ( model ) . to receive ( :add_column ) . with ( table , temporary_columns [ 1 ] , :int , default : 0 , null : false )
expect ( model ) . to receive ( :install_rename_triggers ) . with ( table , columns_to_convert , temporary_columns )
model . restore_conversion_of_integer_to_bigint ( table , columns_to_convert )
end
end
end
2021-06-08 01:23:25 +05:30
describe '#revert_initialize_conversion_of_integer_to_bigint' do
let ( :table ) { :test_table }
before do
model . create_table table , id : false do | t |
t . integer :id , primary_key : true
t . integer :other_id
end
model . initialize_conversion_of_integer_to_bigint ( table , columns )
end
context 'when single column is given' do
let ( :columns ) { :id }
it 'removes column, trigger, and function' do
temporary_column = model . convert_to_bigint_column ( :id )
trigger_name = model . rename_trigger_name ( table , :id , temporary_column )
model . revert_initialize_conversion_of_integer_to_bigint ( table , columns )
expect ( model . column_exists? ( table , temporary_column ) ) . to eq ( false )
expect_trigger_not_to_exist ( table , trigger_name )
expect_function_not_to_exist ( trigger_name )
end
end
context 'when multiple columns are given' do
let ( :columns ) { [ :id , :other_id ] }
it 'removes column, trigger, and function' do
temporary_columns = columns . map { | column | model . convert_to_bigint_column ( column ) }
trigger_name = model . rename_trigger_name ( table , columns , temporary_columns )
model . revert_initialize_conversion_of_integer_to_bigint ( table , columns )
temporary_columns . each do | column |
expect ( model . column_exists? ( table , column ) ) . to eq ( false )
end
expect_trigger_not_to_exist ( table , trigger_name )
expect_function_not_to_exist ( trigger_name )
end
end
2021-04-29 21:17:54 +05:30
end
describe '#backfill_conversion_of_integer_to_bigint' do
let ( :table ) { :_test_backfill_table }
let ( :column ) { :id }
2021-06-08 01:23:25 +05:30
let ( :tmp_column ) { model . convert_to_bigint_column ( column ) }
2021-04-29 21:17:54 +05:30
before do
model . create_table table , id : false do | t |
t . integer :id , primary_key : true
t . text :message , null : false
2021-06-08 01:23:25 +05:30
t . integer :other_id
2021-04-29 21:17:54 +05:30
t . timestamps
end
2022-07-23 23:45:48 +05:30
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
2021-04-29 21:17:54 +05:30
end
context 'when the target table does not exist' do
it 'raises an error' do
expect { model . backfill_conversion_of_integer_to_bigint ( :this_table_is_not_real , column ) }
. to raise_error ( 'Table this_table_is_not_real does not exist' )
end
end
context 'when the primary key does not exist' do
it 'raises an error' do
expect { model . backfill_conversion_of_integer_to_bigint ( table , column , primary_key : :foobar ) }
. to raise_error ( " Column foobar does not exist on #{ table } " )
end
end
context 'when the column to convert does not exist' do
let ( :column ) { :foobar }
it 'raises an error' do
expect { model . backfill_conversion_of_integer_to_bigint ( table , column ) }
2021-06-08 01:23:25 +05:30
. to raise_error ( ArgumentError , " Column #{ column } does not exist on #{ table } " )
2021-04-29 21:17:54 +05:30
end
end
context 'when the temporary column does not exist' do
it 'raises an error' do
expect { model . backfill_conversion_of_integer_to_bigint ( table , column ) }
2021-06-08 01:23:25 +05:30
. to raise_error ( ArgumentError , " Column #{ tmp_column } does not exist on #{ table } " )
2021-04-29 21:17:54 +05:30
end
end
context 'when the conversion is properly initialized' do
let ( :model_class ) do
Class . new ( ActiveRecord :: Base ) do
self . table_name = :_test_backfill_table
end
end
2022-06-21 17:19:12 +05:30
let ( :migration_relation ) { Gitlab :: Database :: BackgroundMigration :: BatchedMigration . with_status ( :active ) }
2021-04-29 21:17:54 +05:30
2021-03-08 18:12:59 +05:30
before do
2021-06-08 01:23:25 +05:30
model . initialize_conversion_of_integer_to_bigint ( table , columns )
2021-04-29 21:17:54 +05:30
model_class . create! ( message : 'hello' )
model_class . create! ( message : 'so long' )
2021-03-08 18:12:59 +05:30
end
2021-06-08 01:23:25 +05:30
context 'when a single column is being converted' do
let ( :columns ) { column }
2021-04-29 21:17:54 +05:30
2021-06-08 01:23:25 +05:30
it 'creates the batched migration tracking record' do
last_record = model_class . create! ( message : 'goodbye' )
expect do
model . backfill_conversion_of_integer_to_bigint ( table , column , batch_size : 2 , sub_batch_size : 1 )
end . to change { migration_relation . count } . by ( 1 )
expect ( migration_relation . last ) . to have_attributes (
job_class_name : 'CopyColumnUsingBackgroundMigrationJob' ,
table_name : table . to_s ,
column_name : column . to_s ,
min_value : 1 ,
max_value : last_record . id ,
interval : 120 ,
batch_size : 2 ,
sub_batch_size : 1 ,
job_arguments : [ [ column . to_s ] , [ model . convert_to_bigint_column ( column ) ] ]
)
end
2021-03-08 18:12:59 +05:30
end
2021-04-29 21:17:54 +05:30
2021-06-08 01:23:25 +05:30
context 'when multiple columns are being converted' do
let ( :other_column ) { :other_id }
let ( :other_tmp_column ) { model . convert_to_bigint_column ( other_column ) }
let ( :columns ) { [ column , other_column ] }
2021-04-29 21:17:54 +05:30
2021-06-08 01:23:25 +05:30
it 'creates the batched migration tracking record' do
last_record = model_class . create! ( message : 'goodbye' , other_id : 50 )
2021-04-29 21:17:54 +05:30
2021-06-08 01:23:25 +05:30
expect do
model . backfill_conversion_of_integer_to_bigint ( table , columns , batch_size : 2 , sub_batch_size : 1 )
end . to change { migration_relation . count } . by ( 1 )
expect ( migration_relation . last ) . to have_attributes (
job_class_name : 'CopyColumnUsingBackgroundMigrationJob' ,
table_name : table . to_s ,
column_name : column . to_s ,
min_value : 1 ,
max_value : last_record . id ,
interval : 120 ,
batch_size : 2 ,
sub_batch_size : 1 ,
job_arguments : [ [ column . to_s , other_column . to_s ] , [ tmp_column , other_tmp_column ] ]
)
2021-04-29 21:17:54 +05:30
end
end
2021-03-08 18:12:59 +05:30
end
end
2021-06-08 01:23:25 +05:30
describe '#revert_backfill_conversion_of_integer_to_bigint' do
let ( :table ) { :_test_backfill_table }
let ( :primary_key ) { :id }
before do
model . create_table table , id : false do | t |
t . integer primary_key , primary_key : true
t . text :message , null : false
t . integer :other_id
t . timestamps
end
2022-07-23 23:45:48 +05:30
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
2021-06-08 01:23:25 +05:30
model . initialize_conversion_of_integer_to_bigint ( table , columns , primary_key : primary_key )
model . backfill_conversion_of_integer_to_bigint ( table , columns , primary_key : primary_key )
end
context 'when a single column is being converted' do
let ( :columns ) { :id }
it 'deletes the batched migration tracking record' do
expect do
model . revert_backfill_conversion_of_integer_to_bigint ( table , columns )
end . to change { Gitlab :: Database :: BackgroundMigration :: BatchedMigration . count } . by ( - 1 )
end
end
context 'when a multiple columns are being converted' do
let ( :columns ) { [ :id , :other_id ] }
it 'deletes the batched migration tracking record' do
expect do
model . revert_backfill_conversion_of_integer_to_bigint ( table , columns )
end . to change { Gitlab :: Database :: BackgroundMigration :: BatchedMigration . count } . by ( - 1 )
end
end
context 'when primary key column has custom name' do
let ( :primary_key ) { :other_pk }
let ( :columns ) { :other_id }
it 'deletes the batched migration tracking record' do
expect do
model . revert_backfill_conversion_of_integer_to_bigint ( table , columns , primary_key : primary_key )
end . to change { Gitlab :: Database :: BackgroundMigration :: BatchedMigration . count } . by ( - 1 )
end
end
end
2021-09-04 01:27:46 +05:30
describe '#ensure_batched_background_migration_is_finished' do
2022-06-21 17:19:12 +05:30
let ( :job_class_name ) { 'CopyColumnUsingBackgroundMigrationJob' }
let ( :table ) { :events }
let ( :column_name ) { :id }
let ( :job_arguments ) { [ [ " id " ] , [ " id_convert_to_bigint " ] , nil ] }
2021-09-04 01:27:46 +05:30
let ( :configuration ) do
{
2022-06-21 17:19:12 +05:30
job_class_name : job_class_name ,
table_name : table ,
column_name : column_name ,
job_arguments : job_arguments
2021-09-04 01:27:46 +05:30
}
end
2022-07-23 23:45:48 +05:30
let ( :migration_attributes ) do
configuration . merge ( gitlab_schema : Gitlab :: Database . gitlab_schemas_for_connection ( model . connection ) . first )
end
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
2021-09-04 01:27:46 +05:30
subject ( :ensure_batched_background_migration_is_finished ) { model . ensure_batched_background_migration_is_finished ( ** configuration ) }
it 'raises an error when migration exists and is not marked as finished' do
2022-07-23 23:45:48 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_dml_mode! ) . twice
create ( :batched_background_migration , :active , migration_attributes )
2022-06-21 17:19:12 +05:30
allow_next_instance_of ( Gitlab :: Database :: BackgroundMigration :: BatchedMigrationRunner ) do | runner |
allow ( runner ) . to receive ( :finalize ) . with ( job_class_name , table , column_name , job_arguments ) . and_return ( false )
end
2021-09-04 01:27:46 +05:30
expect { ensure_batched_background_migration_is_finished }
2021-09-30 23:02:18 +05:30
. to raise_error " Expected batched background migration for the given configuration to be marked as 'finished', but it is 'active': " \
" \t #{ configuration } " \
" \n \n " \
2022-07-23 23:45:48 +05:30
" Finalize it manually by running the following command in a `bash` or `sh` shell: " \
2021-09-30 23:02:18 +05:30
" \n \n " \
2022-05-07 20:08:51 +05:30
" \t sudo gitlab-rake gitlab:background_migrations:finalize[CopyColumnUsingBackgroundMigrationJob,events,id,'[[ \" id \" ] \\ ,[ \" id_convert_to_bigint \" ] \\ ,null]'] " \
2021-09-30 23:02:18 +05:30
" \n \n " \
" For more information, check the documentation " \
" \n \n " \
" \t https://docs.gitlab.com/ee/user/admin_area/monitoring/background_migrations.html # database-migrations-failing-because-of-batched-background-migration-not-finished "
2021-09-04 01:27:46 +05:30
end
it 'does not raise error when migration exists and is marked as finished' do
2022-07-23 23:45:48 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_dml_mode! )
create ( :batched_background_migration , :finished , migration_attributes )
2021-09-04 01:27:46 +05:30
expect { ensure_batched_background_migration_is_finished }
. not_to raise_error
end
it 'logs a warning when migration does not exist' do
2022-07-23 23:45:48 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_dml_mode! )
create ( :batched_background_migration , :active , migration_attributes . merge ( gitlab_schema : :gitlab_something_else ) )
2021-09-04 01:27:46 +05:30
expect ( Gitlab :: AppLogger ) . to receive ( :warn )
. with ( " Could not find batched background migration for the given configuration: #{ configuration } " )
expect { ensure_batched_background_migration_is_finished }
. not_to raise_error
end
2022-06-21 17:19:12 +05:30
it 'finalizes the migration' do
2022-07-23 23:45:48 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_dml_mode! ) . twice
2022-06-21 17:19:12 +05:30
migration = create ( :batched_background_migration , :active , configuration )
allow_next_instance_of ( Gitlab :: Database :: BackgroundMigration :: BatchedMigrationRunner ) do | runner |
expect ( runner ) . to receive ( :finalize ) . with ( job_class_name , table , column_name , job_arguments ) . and_return ( migration . finish! )
end
ensure_batched_background_migration_is_finished
end
context 'when the flag finalize is false' do
it 'does not finalize the migration' do
2022-07-23 23:45:48 +05:30
expect ( Gitlab :: Database :: QueryAnalyzers :: RestrictAllowedSchemas ) . to receive ( :require_dml_mode! )
2022-06-21 17:19:12 +05:30
create ( :batched_background_migration , :active , configuration )
allow_next_instance_of ( Gitlab :: Database :: BackgroundMigration :: BatchedMigrationRunner ) do | runner |
expect ( runner ) . not_to receive ( :finalize ) . with ( job_class_name , table , column_name , job_arguments )
end
expect { model . ensure_batched_background_migration_is_finished ( ** configuration . merge ( finalize : false ) ) } . to raise_error ( RuntimeError )
end
end
2021-09-04 01:27:46 +05:30
end
2018-05-09 12:01:36 +05:30
describe '#index_exists_by_name?' do
2019-01-03 12:48:30 +05:30
it 'returns true if an index exists' do
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2020-03-13 15:44:24 +05:30
'CREATE INDEX test_index_for_index_exists ON projects (path);'
)
expect ( model . index_exists_by_name? ( :projects , 'test_index_for_index_exists' ) )
2018-05-09 12:01:36 +05:30
. to be_truthy
end
it 'returns false if the index does not exist' do
expect ( model . index_exists_by_name? ( :projects , 'this_does_not_exist' ) )
. to be_falsy
end
2019-10-12 21:52:04 +05:30
context 'when an index with a function exists' do
2018-05-09 12:01:36 +05:30
before do
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2018-05-09 12:01:36 +05:30
'CREATE INDEX test_index ON projects (LOWER(path));'
)
end
it 'returns true if an index exists' do
expect ( model . index_exists_by_name? ( :projects , 'test_index' ) )
. to be_truthy
end
end
2021-01-03 14:25:43 +05:30
context 'when an index exists for a table with the same name in another schema' do
before do
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'CREATE SCHEMA new_test_schema'
)
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'CREATE TABLE new_test_schema.projects (id integer, name character varying)'
)
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'CREATE INDEX test_index_on_name ON new_test_schema.projects (LOWER(name));'
)
end
it 'returns false if the index does not exist in the current schema' do
expect ( model . index_exists_by_name? ( :projects , 'test_index_on_name' ) )
. to be_falsy
end
end
2018-05-09 12:01:36 +05:30
end
2020-01-01 13:55:28 +05:30
describe '#create_or_update_plan_limit' do
2020-06-23 00:09:42 +05:30
before do
stub_const ( 'Plan' , Class . new ( ActiveRecord :: Base ) )
stub_const ( 'PlanLimits' , Class . new ( ActiveRecord :: Base ) )
Plan . class_eval do
self . table_name = 'plans'
end
2020-04-22 19:07:51 +05:30
2020-06-23 00:09:42 +05:30
PlanLimits . class_eval do
self . table_name = 'plan_limits'
end
2020-04-22 19:07:51 +05:30
end
it 'properly escapes names' do
2020-01-01 13:55:28 +05:30
expect ( model ) . to receive ( :execute ) . with << ~ SQL
INSERT INTO plan_limits ( plan_id , " project_hooks " )
2020-04-22 19:07:51 +05:30
SELECT id , '10' FROM plans WHERE name = 'free' LIMIT 1
2020-01-01 13:55:28 +05:30
ON CONFLICT ( plan_id ) DO UPDATE SET " project_hooks " = EXCLUDED . " project_hooks " ;
SQL
model . create_or_update_plan_limit ( 'project_hooks' , 'free' , 10 )
end
2020-04-22 19:07:51 +05:30
context 'when plan does not exist' do
it 'does not create any plan limits' do
expect { model . create_or_update_plan_limit ( 'project_hooks' , 'plan_name' , 10 ) }
2020-06-23 00:09:42 +05:30
. not_to change { PlanLimits . count }
2020-04-22 19:07:51 +05:30
end
end
context 'when plan does exist' do
2020-06-23 00:09:42 +05:30
let! ( :plan ) { Plan . create! ( name : 'plan_name' ) }
2020-04-22 19:07:51 +05:30
context 'when limit does not exist' do
it 'inserts a new plan limits' do
expect { model . create_or_update_plan_limit ( 'project_hooks' , 'plan_name' , 10 ) }
2020-06-23 00:09:42 +05:30
. to change { PlanLimits . count } . by ( 1 )
2020-04-22 19:07:51 +05:30
2020-06-23 00:09:42 +05:30
expect ( PlanLimits . pluck ( :project_hooks ) ) . to contain_exactly ( 10 )
2020-04-22 19:07:51 +05:30
end
end
context 'when limit does exist' do
2020-06-23 00:09:42 +05:30
let! ( :plan_limit ) { PlanLimits . create! ( plan_id : plan . id ) }
2020-04-22 19:07:51 +05:30
it 'updates an existing plan limits' do
expect { model . create_or_update_plan_limit ( 'project_hooks' , 'plan_name' , 999 ) }
2020-06-23 00:09:42 +05:30
. not_to change { PlanLimits . count }
2020-04-22 19:07:51 +05:30
expect ( plan_limit . reload . project_hooks ) . to eq ( 999 )
end
end
end
2020-01-01 13:55:28 +05:30
end
2020-03-13 15:44:24 +05:30
describe '#with_lock_retries' do
let ( :buffer ) { StringIO . new }
let ( :in_memory_logger ) { Gitlab :: JsonLogger . new ( buffer ) }
let ( :env ) { { 'DISABLE_LOCK_RETRIES' = > 'true' } }
it 'sets the migration class name in the logs' do
model . with_lock_retries ( env : env , logger : in_memory_logger ) { }
buffer . rewind
expect ( buffer . read ) . to include ( " \" class \" : \" #{ model . class } \" " )
end
2021-09-30 23:02:18 +05:30
where ( raise_on_exhaustion : [ true , false ] )
with_them do
it 'sets raise_on_exhaustion as requested' do
with_lock_retries = double
expect ( Gitlab :: Database :: WithLockRetries ) . to receive ( :new ) . and_return ( with_lock_retries )
expect ( with_lock_retries ) . to receive ( :run ) . with ( raise_on_exhaustion : raise_on_exhaustion )
model . with_lock_retries ( env : env , logger : in_memory_logger , raise_on_exhaustion : raise_on_exhaustion ) { }
end
end
it 'does not raise on exhaustion by default' do
with_lock_retries = double
expect ( Gitlab :: Database :: WithLockRetries ) . to receive ( :new ) . and_return ( with_lock_retries )
expect ( with_lock_retries ) . to receive ( :run ) . with ( raise_on_exhaustion : false )
model . with_lock_retries ( env : env , logger : in_memory_logger ) { }
end
2021-11-11 11:23:49 +05:30
it 'defaults to allowing subtransactions' do
with_lock_retries = double
expect ( Gitlab :: Database :: WithLockRetries ) . to receive ( :new ) . with ( hash_including ( allow_savepoints : true ) ) . and_return ( with_lock_retries )
expect ( with_lock_retries ) . to receive ( :run ) . with ( raise_on_exhaustion : false )
model . with_lock_retries ( env : env , logger : in_memory_logger ) { }
end
2020-03-13 15:44:24 +05:30
end
describe '#backfill_iids' do
include MigrationsHelpers
2022-08-13 15:12:31 +05:30
let_it_be ( :issue_base_type_enum ) { 0 }
let_it_be ( :issue_type ) { table ( :work_item_types ) . find_by ( base_type : issue_base_type_enum ) }
2021-09-30 23:02:18 +05:30
let ( :issue_class ) do
Class . new ( ActiveRecord :: Base ) do
2020-06-23 00:09:42 +05:30
include AtomicInternalId
2020-03-13 15:44:24 +05:30
2020-06-23 00:09:42 +05:30
self . table_name = 'issues'
self . inheritance_column = :_type_disabled
2020-03-13 15:44:24 +05:30
2021-09-30 23:02:18 +05:30
belongs_to :project , class_name : " ::Project " , inverse_of : nil
2020-03-13 15:44:24 +05:30
2020-06-23 00:09:42 +05:30
has_internal_id :iid ,
scope : :project ,
2021-01-29 00:20:46 +05:30
init : - > ( s , _scope ) { s & . project & . issues & . maximum ( :iid ) } ,
2020-06-23 00:09:42 +05:30
presence : false
2022-08-13 15:12:31 +05:30
before_validation - > { self . work_item_type_id = :: WorkItems :: Type . default_issue_type . id }
2020-06-23 00:09:42 +05:30
end
2020-03-13 15:44:24 +05:30
end
let ( :namespaces ) { table ( :namespaces ) }
let ( :projects ) { table ( :projects ) }
let ( :issues ) { table ( :issues ) }
def setup
2022-01-26 12:08:38 +05:30
namespace = namespaces . create! ( name : 'foo' , path : 'foo' , type : Namespaces :: UserNamespace . sti_name )
2022-06-21 17:19:12 +05:30
project_namespace = namespaces . create! ( name : 'project-foo' , path : 'project-foo' , type : 'Project' , parent_id : namespace . id , visibility_level : 20 )
projects . create! ( namespace_id : namespace . id , project_namespace_id : project_namespace . id )
2020-03-13 15:44:24 +05:30
end
it 'generates iids properly for models created after the migration' do
project = setup
model . backfill_iids ( 'issues' )
2021-09-30 23:02:18 +05:30
issue = issue_class . create! ( project_id : project . id )
2020-03-13 15:44:24 +05:30
expect ( issue . iid ) . to eq ( 1 )
end
it 'generates iids properly for models created after the migration when iids are backfilled' do
project = setup
2022-08-13 15:12:31 +05:30
issue_a = issues . create! ( project_id : project . id , work_item_type_id : issue_type . id )
2020-03-13 15:44:24 +05:30
model . backfill_iids ( 'issues' )
2021-09-30 23:02:18 +05:30
issue_b = issue_class . create! ( project_id : project . id )
2020-03-13 15:44:24 +05:30
expect ( issue_a . reload . iid ) . to eq ( 1 )
expect ( issue_b . iid ) . to eq ( 2 )
end
it 'generates iids properly for models created after the migration across multiple projects' do
project_a = setup
project_b = setup
2022-08-13 15:12:31 +05:30
issues . create! ( project_id : project_a . id , work_item_type_id : issue_type . id )
issues . create! ( project_id : project_b . id , work_item_type_id : issue_type . id )
issues . create! ( project_id : project_b . id , work_item_type_id : issue_type . id )
2020-03-13 15:44:24 +05:30
model . backfill_iids ( 'issues' )
2022-08-13 15:12:31 +05:30
issue_a = issue_class . create! ( project_id : project_a . id , work_item_type_id : issue_type . id )
issue_b = issue_class . create! ( project_id : project_b . id , work_item_type_id : issue_type . id )
2020-03-13 15:44:24 +05:30
expect ( issue_a . iid ) . to eq ( 2 )
expect ( issue_b . iid ) . to eq ( 3 )
end
context 'when the first model is created for a project after the migration' do
it 'generates an iid' do
project_a = setup
project_b = setup
2022-08-13 15:12:31 +05:30
issue_a = issues . create! ( project_id : project_a . id , work_item_type_id : issue_type . id )
2020-03-13 15:44:24 +05:30
model . backfill_iids ( 'issues' )
2021-09-30 23:02:18 +05:30
issue_b = issue_class . create! ( project_id : project_b . id )
2020-03-13 15:44:24 +05:30
expect ( issue_a . reload . iid ) . to eq ( 1 )
expect ( issue_b . reload . iid ) . to eq ( 1 )
end
end
context 'when a row already has an iid set in the database' do
it 'backfills iids' do
project = setup
2022-08-13 15:12:31 +05:30
issue_a = issues . create! ( project_id : project . id , work_item_type_id : issue_type . id , iid : 1 )
issue_b = issues . create! ( project_id : project . id , work_item_type_id : issue_type . id , iid : 2 )
2020-03-13 15:44:24 +05:30
model . backfill_iids ( 'issues' )
expect ( issue_a . reload . iid ) . to eq ( 1 )
expect ( issue_b . reload . iid ) . to eq ( 2 )
end
it 'backfills for multiple projects' do
project_a = setup
project_b = setup
2022-08-13 15:12:31 +05:30
issue_a = issues . create! ( project_id : project_a . id , work_item_type_id : issue_type . id , iid : 1 )
issue_b = issues . create! ( project_id : project_b . id , work_item_type_id : issue_type . id , iid : 1 )
issue_c = issues . create! ( project_id : project_a . id , work_item_type_id : issue_type . id , iid : 2 )
2020-03-13 15:44:24 +05:30
model . backfill_iids ( 'issues' )
expect ( issue_a . reload . iid ) . to eq ( 1 )
expect ( issue_b . reload . iid ) . to eq ( 1 )
expect ( issue_c . reload . iid ) . to eq ( 2 )
end
end
end
2020-04-22 19:07:51 +05:30
describe '#check_constraint_name' do
it 'returns a valid constraint name' do
name = model . check_constraint_name ( :this_is_a_very_long_table_name ,
:with_a_very_long_column_name ,
:with_a_very_long_type )
expect ( name ) . to be_an_instance_of ( String )
expect ( name ) . to start_with ( 'check_' )
expect ( name . length ) . to eq ( 16 )
end
end
describe '#check_constraint_exists?' do
before do
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2020-04-22 19:07:51 +05:30
'ALTER TABLE projects ADD CONSTRAINT check_1 CHECK (char_length(path) <= 5) NOT VALID'
)
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'CREATE SCHEMA new_test_schema'
)
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'CREATE TABLE new_test_schema.projects (id integer, name character varying)'
)
2021-11-11 11:23:49 +05:30
ActiveRecord :: Migration . connection . execute (
2021-01-03 14:25:43 +05:30
'ALTER TABLE new_test_schema.projects ADD CONSTRAINT check_2 CHECK (char_length(name) <= 5)'
2020-04-22 19:07:51 +05:30
)
end
it 'returns true if a constraint exists' do
expect ( model . check_constraint_exists? ( :projects , 'check_1' ) )
. to be_truthy
end
it 'returns false if a constraint does not exist' do
expect ( model . check_constraint_exists? ( :projects , 'this_does_not_exist' ) )
. to be_falsy
end
it 'returns false if a constraint with the same name exists in another table' do
expect ( model . check_constraint_exists? ( :users , 'check_1' ) )
. to be_falsy
end
2021-01-03 14:25:43 +05:30
it 'returns false if a constraint with the same name exists for the same table in another schema' do
expect ( model . check_constraint_exists? ( :projects , 'check_2' ) )
. to be_falsy
end
2020-04-22 19:07:51 +05:30
end
describe '#add_check_constraint' do
before do
allow ( model ) . to receive ( :check_constraint_exists? ) . and_return ( false )
end
2020-06-23 00:09:42 +05:30
context 'constraint name validation' do
it 'raises an error when too long' do
expect do
model . add_check_constraint (
:test_table ,
'name IS NOT NULL' ,
'a' * ( Gitlab :: Database :: MigrationHelpers :: MAX_IDENTIFIER_NAME_LENGTH + 1 )
)
end . to raise_error ( RuntimeError )
end
it 'does not raise error when the length is acceptable' do
constraint_name = 'a' * Gitlab :: Database :: MigrationHelpers :: MAX_IDENTIFIER_NAME_LENGTH
expect ( model ) . to receive ( :transaction_open? ) . and_return ( false )
expect ( model ) . to receive ( :check_constraint_exists? ) . and_return ( false )
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( / ADD CONSTRAINT / )
model . add_check_constraint (
:test_table ,
'name IS NOT NULL' ,
constraint_name ,
validate : false
)
end
end
2020-04-22 19:07:51 +05:30
context 'inside a transaction' do
it 'raises an error' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect do
model . add_check_constraint (
:test_table ,
'name IS NOT NULL' ,
'check_name_not_null'
)
end . to raise_error ( RuntimeError )
end
end
context 'outside a transaction' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
context 'when the constraint is already defined in the database' do
it 'does not create a constraint' do
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , 'check_name_not_null' )
. and_return ( true )
expect ( model ) . not_to receive ( :execute ) . with ( / ADD CONSTRAINT / )
# setting validate: false to only focus on the ADD CONSTRAINT command
model . add_check_constraint (
:test_table ,
'name IS NOT NULL' ,
'check_name_not_null' ,
validate : false
)
end
end
context 'when the constraint is not defined in the database' do
it 'creates the constraint' do
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( / ADD CONSTRAINT check_name_not_null / )
# setting validate: false to only focus on the ADD CONSTRAINT command
model . add_check_constraint (
:test_table ,
'char_length(name) <= 255' ,
'check_name_not_null' ,
validate : false
)
end
end
context 'when validate is not provided' do
it 'performs validation' do
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , 'check_name_not_null' )
. and_return ( false ) . exactly ( 1 )
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( / ADD CONSTRAINT check_name_not_null / )
# we need the check constraint to exist so that the validation proceeds
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , 'check_name_not_null' )
. and_return ( true ) . exactly ( 1 )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-04-22 19:07:51 +05:30
model . add_check_constraint (
:test_table ,
'char_length(name) <= 255' ,
'check_name_not_null'
)
end
end
context 'when validate is provided with a falsey value' do
it 'skips validation' do
expect ( model ) . not_to receive ( :disable_statement_timeout )
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( / ADD CONSTRAINT / )
expect ( model ) . not_to receive ( :execute ) . with ( / VALIDATE CONSTRAINT / )
model . add_check_constraint (
:test_table ,
'char_length(name) <= 255' ,
'check_name_not_null' ,
validate : false
)
end
end
context 'when validate is provided with a truthy value' do
it 'performs validation' do
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , 'check_name_not_null' )
. and_return ( false ) . exactly ( 1 )
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( / ADD CONSTRAINT check_name_not_null / )
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , 'check_name_not_null' )
. and_return ( true ) . exactly ( 1 )
expect ( model ) . to receive ( :execute ) . ordered . with ( / VALIDATE CONSTRAINT / )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-04-22 19:07:51 +05:30
model . add_check_constraint (
:test_table ,
'char_length(name) <= 255' ,
'check_name_not_null' ,
validate : true
)
end
end
end
end
describe '#validate_check_constraint' do
context 'when the constraint does not exist' do
it 'raises an error' do
error_message = / Could not find check constraint "check_1" on table "test_table" /
expect ( model ) . to receive ( :check_constraint_exists? ) . and_return ( false )
expect do
model . validate_check_constraint ( :test_table , 'check_1' )
end . to raise_error ( RuntimeError , error_message )
end
end
context 'when the constraint exists' do
it 'performs validation' do
validate_sql = / ALTER TABLE test_table VALIDATE CONSTRAINT check_name /
expect ( model ) . to receive ( :check_constraint_exists? ) . and_return ( true )
expect ( model ) . to receive ( :disable_statement_timeout ) . and_call_original
2020-05-24 23:13:21 +05:30
expect ( model ) . to receive ( :statement_timeout_disabled? ) . and_return ( false )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . with ( / SET statement_timeout TO / )
2020-04-22 19:07:51 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( validate_sql )
2021-10-27 15:23:28 +05:30
expect ( model ) . to receive ( :execute ) . ordered . with ( / RESET statement_timeout / )
2020-04-22 19:07:51 +05:30
model . validate_check_constraint ( :test_table , 'check_name' )
end
end
end
describe '#remove_check_constraint' do
2021-11-11 11:23:49 +05:30
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
end
2020-04-22 19:07:51 +05:30
it 'removes the constraint' do
drop_sql = / ALTER TABLE test_table \ s+DROP CONSTRAINT IF EXISTS check_name /
expect ( model ) . to receive ( :with_lock_retries ) . and_call_original
expect ( model ) . to receive ( :execute ) . with ( drop_sql )
model . remove_check_constraint ( :test_table , 'check_name' )
end
end
2021-01-03 14:25:43 +05:30
describe '#copy_check_constraints' do
context 'inside a transaction' do
it 'raises an error' do
expect ( model ) . to receive ( :transaction_open? ) . and_return ( true )
expect do
model . copy_check_constraints ( :test_table , :old_column , :new_column )
end . to raise_error ( RuntimeError )
end
end
context 'outside a transaction' do
before do
allow ( model ) . to receive ( :transaction_open? ) . and_return ( false )
allow ( model ) . to receive ( :column_exists? ) . and_return ( true )
end
let ( :old_column_constraints ) do
[
{
'schema_name' = > 'public' ,
'table_name' = > 'test_table' ,
'column_name' = > 'old_column' ,
'constraint_name' = > 'check_d7d49d475d' ,
'constraint_def' = > 'CHECK ((old_column IS NOT NULL))'
} ,
{
'schema_name' = > 'public' ,
'table_name' = > 'test_table' ,
'column_name' = > 'old_column' ,
'constraint_name' = > 'check_48560e521e' ,
'constraint_def' = > 'CHECK ((char_length(old_column) <= 255))'
} ,
{
'schema_name' = > 'public' ,
'table_name' = > 'test_table' ,
'column_name' = > 'old_column' ,
'constraint_name' = > 'custom_check_constraint' ,
'constraint_def' = > 'CHECK (((old_column IS NOT NULL) AND (another_column IS NULL)))'
} ,
{
'schema_name' = > 'public' ,
'table_name' = > 'test_table' ,
'column_name' = > 'old_column' ,
'constraint_name' = > 'not_valid_check_constraint' ,
'constraint_def' = > 'CHECK ((old_column IS NOT NULL)) NOT VALID'
}
]
end
it 'copies check constraints from one column to another' do
allow ( model ) . to receive ( :check_constraints_for )
. with ( :test_table , :old_column , schema : nil )
. and_return ( old_column_constraints )
allow ( model ) . to receive ( :not_null_constraint_name ) . with ( :test_table , :new_column )
. and_return ( 'check_1' )
allow ( model ) . to receive ( :text_limit_name ) . with ( :test_table , :new_column )
. and_return ( 'check_2' )
allow ( model ) . to receive ( :check_constraint_name )
. with ( :test_table , :new_column , 'copy_check_constraint' )
. and_return ( 'check_3' )
expect ( model ) . to receive ( :add_check_constraint )
. with (
:test_table ,
'(new_column IS NOT NULL)' ,
'check_1' ,
validate : true
) . once
expect ( model ) . to receive ( :add_check_constraint )
. with (
:test_table ,
'(char_length(new_column) <= 255)' ,
'check_2' ,
validate : true
) . once
expect ( model ) . to receive ( :add_check_constraint )
. with (
:test_table ,
'((new_column IS NOT NULL) AND (another_column IS NULL))' ,
'check_3' ,
validate : true
) . once
expect ( model ) . to receive ( :add_check_constraint )
. with (
:test_table ,
'(new_column IS NOT NULL)' ,
'check_1' ,
validate : false
) . once
model . copy_check_constraints ( :test_table , :old_column , :new_column )
end
it 'does nothing if there are no constraints defined for the old column' do
allow ( model ) . to receive ( :check_constraints_for )
. with ( :test_table , :old_column , schema : nil )
. and_return ( [ ] )
expect ( model ) . not_to receive ( :add_check_constraint )
model . copy_check_constraints ( :test_table , :old_column , :new_column )
end
it 'raises an error when the orginating column does not exist' do
allow ( model ) . to receive ( :column_exists? ) . with ( :test_table , :old_column ) . and_return ( false )
error_message = / Column old_column does not exist on test_table /
expect do
model . copy_check_constraints ( :test_table , :old_column , :new_column )
end . to raise_error ( RuntimeError , error_message )
end
it 'raises an error when the target column does not exist' do
allow ( model ) . to receive ( :column_exists? ) . with ( :test_table , :new_column ) . and_return ( false )
error_message = / Column new_column does not exist on test_table /
expect do
model . copy_check_constraints ( :test_table , :old_column , :new_column )
end . to raise_error ( RuntimeError , error_message )
end
end
end
2020-04-22 19:07:51 +05:30
describe '#add_text_limit' do
context 'when it is called with the default options' do
it 'calls add_check_constraint with an infered constraint name and validate: true' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'max_length' )
check = " char_length(name) <= 255 "
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :add_check_constraint )
. with ( :test_table , check , constraint_name , validate : true )
model . add_text_limit ( :test_table , :name , 255 )
end
end
context 'when all parameters are provided' do
it 'calls add_check_constraint with the correct parameters' do
constraint_name = 'check_name_limit'
check = " char_length(name) <= 255 "
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :add_check_constraint )
. with ( :test_table , check , constraint_name , validate : false )
model . add_text_limit (
:test_table ,
:name ,
255 ,
constraint_name : constraint_name ,
validate : false
)
end
end
end
describe '#validate_text_limit' do
context 'when constraint_name is not provided' do
it 'calls validate_check_constraint with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'max_length' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :validate_check_constraint )
. with ( :test_table , constraint_name )
model . validate_text_limit ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls validate_check_constraint with the correct parameters' do
constraint_name = 'check_name_limit'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :validate_check_constraint )
. with ( :test_table , constraint_name )
model . validate_text_limit ( :test_table , :name , constraint_name : constraint_name )
end
end
end
describe '#remove_text_limit' do
context 'when constraint_name is not provided' do
it 'calls remove_check_constraint with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'max_length' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :remove_check_constraint )
. with ( :test_table , constraint_name )
model . remove_text_limit ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls remove_check_constraint with the correct parameters' do
constraint_name = 'check_name_limit'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :remove_check_constraint )
. with ( :test_table , constraint_name )
model . remove_text_limit ( :test_table , :name , constraint_name : constraint_name )
end
end
end
describe '#check_text_limit_exists?' do
context 'when constraint_name is not provided' do
it 'calls check_constraint_exists? with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'max_length' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , constraint_name )
model . check_text_limit_exists? ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls check_constraint_exists? with the correct parameters' do
constraint_name = 'check_name_limit'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , constraint_name )
model . check_text_limit_exists? ( :test_table , :name , constraint_name : constraint_name )
end
end
end
2020-05-24 23:13:21 +05:30
describe '#add_not_null_constraint' do
context 'when it is called with the default options' do
it 'calls add_check_constraint with an infered constraint name and validate: true' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'not_null' )
check = " name IS NOT NULL "
expect ( model ) . to receive ( :column_is_nullable? ) . and_return ( true )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :add_check_constraint )
. with ( :test_table , check , constraint_name , validate : true )
model . add_not_null_constraint ( :test_table , :name )
end
end
context 'when all parameters are provided' do
it 'calls add_check_constraint with the correct parameters' do
constraint_name = 'check_name_not_null'
check = " name IS NOT NULL "
expect ( model ) . to receive ( :column_is_nullable? ) . and_return ( true )
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :add_check_constraint )
. with ( :test_table , check , constraint_name , validate : false )
model . add_not_null_constraint (
:test_table ,
:name ,
constraint_name : constraint_name ,
validate : false
)
end
end
context 'when the column is defined as NOT NULL' do
it 'does not add a check constraint' do
expect ( model ) . to receive ( :column_is_nullable? ) . and_return ( false )
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . not_to receive ( :add_check_constraint )
model . add_not_null_constraint ( :test_table , :name )
end
end
end
describe '#validate_not_null_constraint' do
context 'when constraint_name is not provided' do
it 'calls validate_check_constraint with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'not_null' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :validate_check_constraint )
. with ( :test_table , constraint_name )
model . validate_not_null_constraint ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls validate_check_constraint with the correct parameters' do
constraint_name = 'check_name_not_null'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :validate_check_constraint )
. with ( :test_table , constraint_name )
model . validate_not_null_constraint ( :test_table , :name , constraint_name : constraint_name )
end
end
end
describe '#remove_not_null_constraint' do
context 'when constraint_name is not provided' do
it 'calls remove_check_constraint with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'not_null' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :remove_check_constraint )
. with ( :test_table , constraint_name )
model . remove_not_null_constraint ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls remove_check_constraint with the correct parameters' do
constraint_name = 'check_name_not_null'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :remove_check_constraint )
. with ( :test_table , constraint_name )
model . remove_not_null_constraint ( :test_table , :name , constraint_name : constraint_name )
end
end
end
describe '#check_not_null_constraint_exists?' do
context 'when constraint_name is not provided' do
it 'calls check_constraint_exists? with an infered constraint name' do
constraint_name = model . check_constraint_name ( :test_table ,
:name ,
'not_null' )
expect ( model ) . to receive ( :check_constraint_name ) . and_call_original
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , constraint_name )
model . check_not_null_constraint_exists? ( :test_table , :name )
end
end
context 'when constraint_name is provided' do
it 'calls check_constraint_exists? with the correct parameters' do
constraint_name = 'check_name_not_null'
expect ( model ) . not_to receive ( :check_constraint_name )
expect ( model ) . to receive ( :check_constraint_exists? )
. with ( :test_table , constraint_name )
model . check_not_null_constraint_exists? ( :test_table , :name , constraint_name : constraint_name )
end
end
end
2020-10-24 23:57:45 +05:30
describe '#create_extension' do
subject { model . create_extension ( extension ) }
let ( :extension ) { :btree_gist }
it 'executes CREATE EXTENSION statement' do
expect ( model ) . to receive ( :execute ) . with ( / CREATE EXTENSION IF NOT EXISTS #{ extension } / )
subject
end
context 'without proper permissions' do
before do
2022-07-16 23:28:13 +05:30
allow ( model ) . to receive ( :execute )
. with ( / CREATE EXTENSION IF NOT EXISTS #{ extension } / )
. and_raise ( ActiveRecord :: StatementInvalid , 'InsufficientPrivilege: permission denied' )
2020-10-24 23:57:45 +05:30
end
2022-07-16 23:28:13 +05:30
it 'raises an exception and prints an error message' do
expect { subject }
. to output ( / user is not allowed / ) . to_stderr
. and raise_error ( ActiveRecord :: StatementInvalid , / InsufficientPrivilege / )
2020-10-24 23:57:45 +05:30
end
end
end
describe '#drop_extension' do
subject { model . drop_extension ( extension ) }
let ( :extension ) { 'btree_gist' }
it 'executes CREATE EXTENSION statement' do
expect ( model ) . to receive ( :execute ) . with ( / DROP EXTENSION IF EXISTS #{ extension } / )
subject
end
context 'without proper permissions' do
before do
2022-07-16 23:28:13 +05:30
allow ( model ) . to receive ( :execute )
. with ( / DROP EXTENSION IF EXISTS #{ extension } / )
. and_raise ( ActiveRecord :: StatementInvalid , 'InsufficientPrivilege: permission denied' )
2020-10-24 23:57:45 +05:30
end
2022-07-16 23:28:13 +05:30
it 'raises an exception and prints an error message' do
expect { subject }
. to output ( / user is not allowed / ) . to_stderr
. and raise_error ( ActiveRecord :: StatementInvalid , / InsufficientPrivilege / )
2020-10-24 23:57:45 +05:30
end
end
end
2021-09-30 23:02:18 +05:30
describe '#rename_constraint' do
it " executes the statement to rename constraint " do
expect ( model ) . to receive ( :execute ) . with / ALTER TABLE "test_table" \ nRENAME CONSTRAINT "fk_old_name" TO "fk_new_name" /
model . rename_constraint ( :test_table , :fk_old_name , :fk_new_name )
end
end
2022-07-23 23:45:48 +05:30
describe '#drop_sequence' do
it " executes the statement to drop the sequence " do
expect ( model ) . to receive ( :execute ) . with / ALTER TABLE "test_table" ALTER COLUMN "test_column" DROP DEFAULT; \ nDROP SEQUENCE IF EXISTS "test_table_id_seq" /
model . drop_sequence ( :test_table , :test_column , :test_table_id_seq )
end
end
describe '#add_sequence' do
it " executes the statement to add the sequence " do
expect ( model ) . to receive ( :execute ) . with " CREATE SEQUENCE \" test_table_id_seq \" START 1; \n ALTER TABLE \" test_table \" ALTER COLUMN \" test_column \" SET DEFAULT nextval( \' test_table_id_seq \' ) \n "
model . add_sequence ( :test_table , :test_column , :test_table_id_seq , 1 )
end
end
2016-06-02 11:05:42 +05:30
end