Redirect response should get cache tags from overridden config

Created on 24 February 2025, about 1 month ago

Problem/Motivation

The redirect response that is created in CasRedirector depends on the config from cas.settings.
The module adds the 'config:cas.settings' cache tag which is hard-coded in CasRedirectData.

Modules that override config using the config override mechanism may add additional cache tags.
One example is cas_mock_server, which has CasMockServerConfigOverrider.
These additional cache tags do currently _not_ end up in the response.

As a consequence, we get a cached value even after one of the additional cache tags was invalidated.

Steps to reproduce

For us this occured in a test with cas_mock_server.
A simplified version will look somehow like this:

  public function testCasMockPageCache() {
    // Go to front page.
    $this->drupalGet('');
    // This path is rewritten as /user/login which is the default front page path.
    // This is then redirected to the cas server based on cas settings.
    $this->assertStringStartsWith('https://real-cas-server.example.com/cas/login?', $this->getUrl());
    \Drupal::service('cas_mock_server.server_manager')->start();
    $this->drupalGet('');
    $this->assertStringStartsWith('https://real-cas-server.example.com/cas/login?', $this->getUrl());
    \Drupal::service('cache_tags.invalidator')->invalidateTags(['config:cas.settings']);
    $this->drupalGet('');
    $this->assertStringStartsWith('http://localhost/build/cas-mock-server/login?', $this->getUrl());
  }

The ->start(); call sets a state value, but we still get the old value from cache.
Only after invalidating the config cache tags, we get the new value.

Proposed resolution

Add all the cache tags / metadata from the config object to the response.

Remaining tasks

User interface changes

API changes

Data model changes

🐛 Bug report
Status

Active

Version

3.0

Component

CAS

Created by

🇩🇪Germany donquixote

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

Merge Requests

Comments & Activities

  • Issue created by @donquixote
  • Pipeline finished with Failed
    about 1 month ago
    Total: 256s
    #432887
  • 🇩🇪Germany donquixote

    Let's see if anything wrong with this direction, then I can add tests.

  • Pipeline finished with Failed
    about 1 month ago
    Total: 264s
    #433895
  • Merge request !47Draft: Probe / kick off test run → (Closed) created by donquixote
  • Pipeline finished with Failed
    about 1 month ago
    Total: 241s
    #434150
  • Pipeline finished with Failed
    about 1 month ago
    Total: 243s
    #441295
  • Pipeline finished with Failed
    about 1 month ago
    Total: 237s
    #441321
  • Pipeline finished with Success
    about 1 month ago
    Total: 224s
    #441333
  • 🇮🇹Italy sardara

    The fix and tests look good.

  • 🇷🇴Romania claudiu.cristea Arad 🇷🇴

    I have a question, see the MR

  • 🇷🇴Romania claudiu.cristea Arad 🇷🇴

    Either we implement a full deprecation path, or we keep it as it is and remove the @todo

    A full deprecation path would look like:

    1. Remove the initial property value assignation.
    2. CasRedirectData to implement RefinableCacheableDependencyInterface and use RefinableCacheableDependencyTrait
    3. Remove all properties and methods already inherited from trait
    4. Add config object as 3rd param in CasRedirectData::__construct() which will default to NULL
    5. In CasRedirectData::__construct() check if config object has been passed. If not set it to \Drupal::configFactory()->get('cas.settings') and trigger deprecation (see https://www.drupal.org/about/core/policies/core-change-policies/how-to-d... )
    6. In CasRedirectData::__construct(), add config as cache dependency of CasRedirectData
    7. In redirector add $data (CasRedirectData) as cache dependency

    But I'm OK also to keep the property as it is, but then remove the TODO

  • Pipeline finished with Success
    16 days ago
    Total: 448s
    #453451
  • 🇩🇪Germany donquixote

    I updated the comment, remove the todo.

    But to recap my point from our conversation:
    We should add cache metadata for dependencies when we actually use them.

    In some cases this means adding the same dependency twice in two different methods, so that if we remove one of them, the other is still there.

    On the other hand, in general we should not add cache metadata for something that we think will be used elsewhere.
    This does not cause much harm for a config object, but it breaks the pattern and can lead people astray, which now rely on that cache metadata being added elsewhere.
    Here we are already in that scenario, where we now have to support those hypothetical people.

    For this reason it seems wrong to add config cache metadata to CasRedirectData, if nothing in that object really depends on that config.
    E.g. there is nothing you could update in the config that would lead to different outward behavior of CasRedirectData.
    Of course this can change if an event subscriber adds some data to CasRedirectData that is dependent on that config. Also because the config is made available in the event object. Normally that event subscriber would then have to add the config cache metadata to the CasRedirectData.

    CasRedirectData to implement RefinableCacheableDependencyInterface and use RefinableCacheableDependencyTrait

    This actually seems like a good idea.
    However, perhaps it is better to rethink the architecture for cas 4.x, maybe a lot of it becomes obsolete.
    But if this can be done BC-friendly for 3.x, sure.

  • 🇷🇴Romania claudiu.cristea Arad 🇷🇴

    Fair enough. Thank you for catching & fixing.

  • Pipeline finished with Skipped
    16 days ago
    #453466
  • 🇷🇴Romania claudiu.cristea Arad 🇷🇴

    Thank you all

  • 🇷🇴Romania claudiu.cristea Arad 🇷🇴

    Fixing credits

  • Automatically closed - issue fixed for 2 weeks with no activity.

Production build 0.71.5 2024