Commit c5e8aef3 authored by Adam Hegyi's avatar Adam Hegyi

Remove PG warning with with_lock_retries

This change prevents disabling idle transaction timeout when no outer
transaction is present. This prevents the following error message:

"WARNING:  SET LOCAL can only be used in transaction blocks"
parent 16fdd83b
...@@ -95,7 +95,7 @@ module Gitlab ...@@ -95,7 +95,7 @@ module Gitlab
run_block_with_transaction run_block_with_transaction
rescue ActiveRecord::LockWaitTimeout rescue ActiveRecord::LockWaitTimeout
if retry_with_lock_timeout? if retry_with_lock_timeout?
disable_idle_in_transaction_timeout disable_idle_in_transaction_timeout if ActiveRecord::Base.connection.transaction_open?
wait_until_next_retry wait_until_next_retry
reset_db_settings reset_db_settings
...@@ -149,7 +149,7 @@ module Gitlab ...@@ -149,7 +149,7 @@ module Gitlab
log(message: "Couldn't acquire lock to perform the migration", current_iteration: current_iteration) log(message: "Couldn't acquire lock to perform the migration", current_iteration: current_iteration)
log(message: "Executing the migration without lock timeout", current_iteration: current_iteration) log(message: "Executing the migration without lock timeout", current_iteration: current_iteration)
execute("SET LOCAL lock_timeout TO '0'") disable_lock_timeout if ActiveRecord::Base.connection.transaction_open?
run_block run_block
...@@ -184,6 +184,10 @@ module Gitlab ...@@ -184,6 +184,10 @@ module Gitlab
execute("SET LOCAL idle_in_transaction_session_timeout TO '0'") execute("SET LOCAL idle_in_transaction_session_timeout TO '0'")
end end
def disable_lock_timeout
execute("SET LOCAL lock_timeout TO '0'")
end
def reset_db_settings def reset_db_settings
execute('RESET idle_in_transaction_session_timeout; RESET lock_timeout') execute('RESET idle_in_transaction_session_timeout; RESET lock_timeout')
end end
......
...@@ -104,9 +104,69 @@ RSpec.describe Gitlab::Database::WithLockRetries do ...@@ -104,9 +104,69 @@ RSpec.describe Gitlab::Database::WithLockRetries do
end end
context 'after 3 iterations' do context 'after 3 iterations' do
let(:retry_count) { 4 } it_behaves_like 'retriable exclusive lock on `projects`' do
let(:retry_count) { 4 }
end
context 'setting the idle transaction timeout' do
context 'when there is no outer transaction: disable_ddl_transaction! is set in the migration' do
it 'does not disable the idle transaction timeout' do
allow(ActiveRecord::Base.connection).to receive(:transaction_open?).and_return(false)
allow(subject).to receive(:run_block_with_transaction).once.and_raise(ActiveRecord::LockWaitTimeout)
allow(subject).to receive(:run_block_with_transaction).once
expect(subject).not_to receive(:disable_idle_in_transaction_timeout)
subject.run {}
end
end
it_behaves_like 'retriable exclusive lock on `projects`' context 'when there is outer transaction: disable_ddl_transaction! is not set in the migration' do
it 'disables the idle transaction timeout so the code can sleep and retry' do
allow(ActiveRecord::Base.connection).to receive(:transaction_open?).and_return(true)
n = 0
allow(subject).to receive(:run_block_with_transaction).twice do
n += 1
raise(ActiveRecord::LockWaitTimeout) if n == 1
end
expect(subject).to receive(:disable_idle_in_transaction_timeout).once
subject.run {}
end
end
end
end
context 'after the retries are exhausted' do
let(:timing_configuration) do
[
[1.second, 1.second]
]
end
context 'when there is no outer transaction: disable_ddl_transaction! is set in the migration' do
it 'does not disable the lock_timeout' do
allow(ActiveRecord::Base.connection).to receive(:transaction_open?).and_return(false)
allow(subject).to receive(:run_block_with_transaction).once.and_raise(ActiveRecord::LockWaitTimeout)
expect(subject).not_to receive(:disable_lock_timeout)
subject.run {}
end
end
context 'when there is outer transaction: disable_ddl_transaction! is not set in the migration' do
it 'disables the lock_timeout' do
allow(ActiveRecord::Base.connection).to receive(:transaction_open?).and_return(true)
allow(subject).to receive(:run_block_with_transaction).once.and_raise(ActiveRecord::LockWaitTimeout)
expect(subject).to receive(:disable_lock_timeout)
subject.run {}
end
end
end end
context 'after the retries, without setting lock_timeout' do context 'after the retries, without setting lock_timeout' do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment