Sessions are invalidated on subsequent page request when written while transaction is open

Created on 31 May 2024, 8 months ago

Problem/Motivation

I am testing social auth locally (Sign in with Facebook) and noticed that often - but not always - a newly-created user had their session cookie immediately invalidated/deleted on the first page request following the session's initial set-cookie header. This was mysterious to say the least.

I believe I've tracked this down to transactions. If there is a transaction open, the merge statement in SessionHandler::write() is added as a savepoint to it and it's not committed until the transaction is closed, perhaps even after the response is sent (due to post-response destructable services, for instance.) This means that the session ID has already been sent off to the client, but the session may still be writing/waiting to be written. In that case, it's not valid or may not contain a uid (if it's being regenerated.)

Steps to reproduce

Write a new session for a user (or migrate a session to associate with a user) while a transaction is open. Almost immediately (e.g., redirect after login) make a request with that session. It will not load the session, because of a race condition on reading a write that isn't done.

Proposed resolution

Commit any pending transactions prior to writing a session.

Remaining tasks

Review. I'm not really sure how to write a test for this since it would need to be timed just right or mock a slow write... I think recent policy changes mean we may not need a test?

User interface changes

None.

API changes

None. There's an outside chance that committing an open transaction here would be unexpected, but session writes are usually only done in very explicit cases or during shutdown. I'm also not sure how else to achieve this.

Data model changes

None.

Release notes snippet

Not sure if this needs one?

🐛 Bug report
Status

Closed: duplicate

Version

11.0 🔥

Component
Base 

Last updated about 11 hours ago

Created by

🇺🇸United States bradjones1 Digital Nomad Life

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @bradjones1
  • Merge request !8239Commit transaction prior to session write. → (Open) created by bradjones1
  • 🇺🇸United States bradjones1 Digital Nomad Life
  • Pipeline finished with Canceled
    8 months ago
    Total: 24s
    #186764
  • Pipeline finished with Success
    8 months ago
    Total: 778s
    #186763
  • Status changed to Needs review 8 months ago
  • 🇺🇸United States bradjones1 Digital Nomad Life

    No regressions in the existing test suite. I think the correct answer to this is 📌 [PP-1] Convert all transactions to explicit COMMIT Postponed but that change won't come until at least Drupal 12.

    Some discussion on Slack as well.

  • 🇺🇸United States bradjones1 Digital Nomad Life
  • 🇮🇹Italy mondrake 🇮🇹

    OMG. This feels wrong. commitAll() was introduced in 🐛 Transaction autocommit during shutdown relies on unreliable object destruction order Active to fix during shutdown an edge case of xdebug keeping alive Transaction objects that should have been naturally destructed. Using it in normal operations means you cannot rely on transaction state after it being called.

    I think the question here is not even related to implicit vs explicit commit, but rather why should a transaction be active at this point.

  • 🇺🇸United States bradjones1 Digital Nomad Life

    Thanks for chiming in.

    I think the question here is not even related to implicit vs explicit commit, but rather why should a transaction be active at this point.

    Perhaps the question isn't directly implicit vs. explicit, but since Drupal currently supports implicit transaction commits/completion, then why wouldn't there be a transaction active?

    I dropped a breakpoint on TransactionManagerBase::push() and found that the open transaction was started in SqlContentEntityStorage::save(), where it is not explicitly destroyed. So core is opening the transaction. What's more, the object isn't returned to the caller so there's no way to explicitly commit. If you follow the instructions on the transaction docs, then yes, you could start a transaction, then ::save() would add a savepoint, and then the transaction could be explicitly committed. But that would mean a pretty significant shift for developers to always have to wrap their entity saves in transactions in order to avoid the condition that prompted this issue.

    That also explains why I only experience this bug after a user is created, because of the entity save operation.

  • 🇺🇸United States mfb San Francisco

    SqlContentEntityStorage::save() should be automatically committing the transaction when the transaction goes out of scope.. so if that's not happening, I guess something else already started a transaction?

  • 🇺🇸United States bradjones1 Digital Nomad Life

    Re: #8 - At first I was like, duh, yes it would be destroyed when going out of scope, I was barking up the wrong tree.

    Except then, I tried to go in and reconcile each transaction "push" with a destruction. Each push was from ::save(), however I had two that were not cleaned up. I even added an explicit unset($transaction) before the return and same result - there were two objects that were only destroyed later during shutdown.

    I think this is at least in part due to the fact I am doing an entity save during an entity insert hook - e.g., I create a profile entity for the newly-created user. I suspect this is related somehow. There's no other reference to the Transaction object returned from ::push(), so I'm kinda mystified. PHP must not be allowing the reference to be destroyed, even when I explicitly unset() it in ::save(). Does this ring a bell to anyone?

  • 🇺🇸United States bradjones1 Digital Nomad Life

    I am curious if this is an edge case with TransactionManagerBase. In my case, two transactions that are not destroyed seem to fall under this docblock in ::unpile():

        // If the $id does not correspond to the one in the stack for that $name,
        // we are facing an orphaned Transaction object (for example in case of a
        // DDL statement breaking an active transaction). That should be listed in
        // $voidedItems, so we can remove it from there.
        if (!isset($this->stack()[$id]) || $this->stack()[$id]->name !== $name) {
          unset($this->voidedItems[$id]);
          return;
        }
    

    That got me looking at $this->voidedItems, which is empty. So these were never "voided." While these transactions were never added to the voided item list, the docblock in ::voidStackItem() says:

        // The item should be removed from $stack and added to $voidedItems for
        // later processing.
    

    There isn't a read accessor of $voidedItems that I can find - did I just discover a bug in the transaction manager, or am I just not familiar enough with this to know what I'm looking at?

  • Status changed to Closed: duplicate 8 months ago
  • 🇺🇸United States bradjones1 Digital Nomad Life

    Well... it turns out this is basically a dupe of 🐛 Transaction autocommit during shutdown relies on unreliable object destruction order Active .

    Discussion in Slack is that this is indeed insidious and would be avoided in general by using a separate, non-transactional database connection a la 🐛 Race conditions with lock/cache using non-database storage - add a non-transactional database connection Needs work and deprecating/removing non-explicitly-committed database transactions at 📌 [PP-1] Convert all transactions to explicit COMMIT Postponed

    Closing Won't Fix (Fix Elsewhere.)

Production build 0.71.5 2024