Skip to content

Commit

Permalink
Merge pull request rails#44573 from rails/restore-transactions
Browse files Browse the repository at this point in the history
Allow open transactions to be restored upon DB reconnect
  • Loading branch information
matthewd authored Mar 3, 2022
2 parents 39a192f + 2d7bc98 commit 74ba52e
Show file tree
Hide file tree
Showing 7 changed files with 83 additions and 9 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -337,8 +337,24 @@ def transaction_open?
current_transaction.open?
end

def reset_transaction # :nodoc:
def reset_transaction(restore: false) # :nodoc:
# Store the existing transaction state to the side
old_state = @transaction_manager if restore && @transaction_manager&.restorable?

@transaction_manager = ConnectionAdapters::TransactionManager.new(self)

if block_given?
# Reconfigure the connection without any transaction state in the way
result = yield

# Now the connection's fully established, we can swap back
if old_state
@transaction_manager = old_state
@transaction_manager.restore_transactions
end

result
end
end

# Register a record with the current transaction so that its after_commit and after_rollback callbacks
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,10 @@ def materialized?
@materialized
end

def restore!
@materialized = false
end

def rollback_records
return unless records
ite = records.uniq(&:__id__)
Expand Down Expand Up @@ -349,6 +353,20 @@ def dirty_current_transaction
current_transaction.dirty!
end

def restore_transactions
return false unless restorable?

@stack.each(&:restore!)

materialize_transactions unless @lazy_transactions_enabled

true
end

def restorable?
@stack.none?(&:dirty?)
end

def materialize_transactions
return if @materializing_transactions

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,8 @@ def initialize(connection, logger = nil, config = {}) # :nodoc:

@default_timezone = self.class.validate_default_timezone(config[:default_timezone])

@raw_connection_dirty = false

configure_connection
end

Expand Down Expand Up @@ -555,10 +557,11 @@ def active?
# new connection with the database. Implementors should call super
# immediately after establishing the new connection (and while still
# holding @lock).
def reconnect!
clear_cache!(new_connection: true)
reset_transaction
configure_connection
def reconnect!(restore_transactions: false)
reset_transaction(restore: restore_transactions) do
clear_cache!(new_connection: true)
configure_connection
end
end

# Disconnects from the database if already connected. Otherwise, this
Expand Down Expand Up @@ -637,6 +640,7 @@ def verify!
# PostgreSQL's lo_* methods.
def raw_connection
disable_lazy_transactions!
@raw_connection_dirty = true
@raw_connection
end

Expand Down Expand Up @@ -789,6 +793,10 @@ def extract_limit(sql_type)
EXTENDED_TYPE_MAPS = Concurrent::Map.new

private
def reconnect_can_restore_state?
transaction_manager.restorable? && !@raw_connection_dirty
end

def extended_type_map_key
if @default_timezone
{ default_timezone: @default_timezone }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -123,9 +123,9 @@ def active?
@raw_connection.ping
end

def reconnect!
def reconnect!(restore_transactions: false)
@lock.synchronize do
disconnect!
@raw_connection.close
connect
super
end
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -322,7 +322,7 @@ def reload_type_map # :nodoc:
end

# Close then reopen the connection.
def reconnect!
def reconnect!(restore_transactions: false)
@lock.synchronize do
begin
@raw_connection.reset
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -163,7 +163,7 @@ def active?
!@raw_connection.closed?
end

def reconnect!
def reconnect!(restore_transactions: false)
@lock.synchronize do
if active?
@raw_connection.rollback rescue nil
Expand Down
32 changes: 32 additions & 0 deletions activerecord/test/cases/adapter_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -394,6 +394,23 @@ def setup
assert_not raw_transaction_open?(@connection)
end

test "materialized transaction state can be restored after a reconnect" do
@connection.begin_transaction
assert_predicate @connection, :transaction_open?
# +materialize_transactions+ currently automatically dirties the
# connection, which would make it unrestorable
@connection.transaction_manager.stub(:dirty_current_transaction, nil) do
@connection.materialize_transactions
end
assert raw_transaction_open?(@connection)
@connection.reconnect!(restore_transactions: true)
assert_predicate @connection, :transaction_open?
assert_not raw_transaction_open?(@connection)
ensure
@connection.reconnect!
assert_not_predicate @connection, :transaction_open?
end

test "materialized transaction state is reset after a disconnect" do
@connection.begin_transaction
assert_predicate @connection, :transaction_open?
Expand All @@ -417,6 +434,21 @@ def setup
assert_not raw_transaction_open?(@connection)
end

test "unmaterialized transaction state can be restored after a reconnect" do
@connection.begin_transaction
assert_predicate @connection, :transaction_open?
assert_not raw_transaction_open?(@connection)
@connection.reconnect!(restore_transactions: true)
assert_predicate @connection, :transaction_open?
assert_not raw_transaction_open?(@connection)
@connection.materialize_transactions
assert raw_transaction_open?(@connection)
ensure
@connection.reconnect!
assert_not_predicate @connection, :transaction_open?
assert_not raw_transaction_open?(@connection)
end

test "unmaterialized transaction state is reset after a disconnect" do
@connection.begin_transaction
assert_predicate @connection, :transaction_open?
Expand Down

0 comments on commit 74ba52e

Please sign in to comment.