-
Notifications
You must be signed in to change notification settings - Fork 21.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Ensure transaction tracking finishes when reconnecting #49391
Ensure transaction tracking finishes when reconnecting #49391
Conversation
a88ff07
to
9e46520
Compare
@@ -364,6 +364,8 @@ def transaction_open? | |||
end | |||
|
|||
def reset_transaction(restore: false) # :nodoc: | |||
@transaction_manager.suspend_transactions if defined?(@transaction_manager) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why we need to use defined?
here and not in the line below?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good question - since this method is called in initialize
without the restore
option, the @transaction_manager
was always initialized before being accessed. Maybe it would be better to do something like this here:
if restore
@transaction_manager.suspend_transactions
old_state = @transaction_manager if @transaction_manager.restorable?
end
but I'm going to rework this based on the additional feedback, so I'll remove this line entirely.
Agree naming is hard here... but I don't like "suspend". "detach" is probably the closest active verb I can come up with ("abandon" feels too final), but perhaps "lost" is a better fit: it's something that happened to us, not something we want to do. Or maybe "break"? In the current state of things1, I'd much prefer that the It's certainly true that Transaction is already rather sensitive to its caller knowing what methods are permissible to call at any given time, but I think most existing possible confusions will produce an exception at some point. It seems like Footnotes
|
activerecord/lib/active_record/connection_adapters/abstract/transaction.rb
Outdated
Show resolved
Hide resolved
activerecord/lib/active_record/connection_adapters/abstract/transaction.rb
Outdated
Show resolved
Hide resolved
0b014b0
to
a1bab92
Compare
We noticed a couple of spots where transaction tracking events were potentially incorrect. When the connection reconnects with `restore_transactions: true`, we were keeping the original start time for the transaction. In this case it seems more accurate to treat these as separte transactions where the first one finishes as incomplete. Instead of forcing the adapter to tell the transaction manager to suspend (or `detach` or `lost`) the transactions, we can keep that logic encapsulated inside of the Transaction class for when to broadcast a finish event. This means that we won't capture a finish event if you manually call `reconnect!` without `restore_transactions: true`, but that might be worthy of the tradeoff since this is a rare use-case anyway. We also start raising here if the TransactionInstrumenter receives `#start` when already started, or `#finish` when not started. This ensures that we don't quietly end up in any weird states. When marking that a transaction is incomplete, we also want to check that the transaction is materialized to avoid finishing our instrumentation if it hasn't already started. Also added a test to simulate losing a connection without ever materializing a transaction. Co-authored-by: Daniel Colson <[email protected]>
a1bab92
to
306ef99
Compare
Ensure transaction tracking finishes when reconnecting
Motivation / Background
We noticed a couple of spots where transaction tracking events were potentially incorrect.
restore_transactions: true
, we were keeping the original start time for the transaction. In this case, it makes more sense to treat these as two separate transactions for tracking purposes.reconnect!
within a transaction opened viabeing_transaction
rather thantransaction do
, we were never marking the event as finished and thus never broadcasting the event. This isn't public API and likely isn't very common, but we did run into one scenario in a test file that was doing this in a setup block.Detail
We added a line to the
reset_transactions
method to mark any open transactions as finished. We went back and forth on the naming for this new method, but eventually settled onsuspend_transactions
since the transactions may potentially be restarted.Additional information
This is a followup to #49192 and #49262
Checklist
Before submitting the PR make sure the following are checked:
[Fix #issue-number]