Allow modules to add their own cache tags to a specific routing cache entry

Created on 18 November 2023, over 1 year ago

Problem/Motivation

Drupal's Routing/RouteProvider caches the association between an incoming path and matching routes (including parameters), for each request. This happens after all other processes are done and after all processInbounds.

The thing is that it only explicitly provides the route_match cache tag for all those cache entries, in getRouteCollectionForRequest(Request $request), which doesn't make it possible to implement more selective cache invalidations in case you want the cache to expire only for a particular path/route.

$this->cache->set($cid, $cache_value, CacheBackendInterface::CACHE_PERMANENT, ['route_match']);

Proposed resolution

Provide a hook to allow modules to add their own cache tags for a specific path/route, based on the $request.

I'm providing a patch that introduces hook_routing_cache_tags(Request $request).

Example:

/**
 * Allow modules to add their own cache tags to the routing cache object.
 *
 * RouteProvider stores the association of paths and routes in the cache.
 * It only adds the generic tag 'route_match'. Modules can implement this hook
 * to add more tags, based on the `$request`, in order to perform more
 * selective cache invalidations.
 */
function hook_routing_cache_tags(Request $request) {
  if ($request->getPathInfo() == 'some_path' || $request->attributes->get('some_attribute')) {
    return [
      'custom_tag:1',
      'custom_tag:2',
      'custom_tag:3',
    ];
  }
}

Then, in your module, you invalidate the tags added to those specific paths:

Cache::invalidateTags(['custom_tag:1']);

✨ Feature request
Status

Active

Version

9.5

Component
RoutingΒ  β†’

Last updated 5 days ago

Created by

πŸ‡¨πŸ‡΄Colombia camilo.escobar

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

Comments & Activities

  • Issue created by @camilo.escobar
  • last update over 1 year ago
    30,307 pass, 2 fail
  • Status changed to Needs review about 1 year ago
  • Status changed to Needs work about 1 year ago
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    This will need test coverage, typically with adding new parameters it has to be backwards compatible.

    Issue summary appears to be missing sections, and this appears to be updating the API so that should be noted.

    Probably will need subsystem maintainer sign off.

  • πŸ‡¨πŸ‡΄Colombia camilo.escobar

    Patch rerolled for 10.3.

  • πŸ‡΅πŸ‡±Poland maksim_matuzka

    Hi guys, I have applied this patch temporary for my business requirements, could you please help me to find the right way to fix this? I was trying to add cacheContexts(), cacheTags() and even cacheMaxAge() but it doesn't help me, routing cache don't work correctly. Thanks!

  • πŸ‡¨πŸ‡΄Colombia camilo.escobar

    @maksim_matuzka, can you clarify your goal more precisely?

    Additionally, the patch you shared is a custom implementation addressing your specific business requirement rather than a general solution applicable to all users. Sharing it as a .patch file might mislead other developers searching for a fix.

    Instead, consider sharing your code as a code block within your comment, making it clear that it's tailored to your use case.

    I recommend deleting the patch file and instead posting a new comment that clearly explains your goal. This will help others better understand your specific requirement and provide more relevant guidance.

  • πŸ‡΅πŸ‡±Poland maksim_matuzka

    Hi @camilo.escobar,

    How to disable cache for the specific route?

  • πŸ‡¨πŸ‡΄Colombia camilo.escobar

    @maksim_matuzka, there are several ways to do this, depending on the origin of the route you want to modify.

    If it’s a custom route

    A) Modify the route definition

    You can add the no_cache attribute to the route definition in yourmodule.routing.yml:

    yourmodule.custom_route:
      path: '/my-custom-route'
      defaults:
        _controller: '\Drupal\yourmodule\Controller\CustomRoute::build'
        _title_callback: '\Drupal\yourmodule\Controller\CustomRoute::getTitle'
      requirements:
        _permission: 'access content'
      options:
        no_cache: 'TRUE'
    

    B) Disable cache in the controller

    You can also call the following function inside the controller handling the route:
    \Drupal::service('page_cache_kill_switch')->trigger();

    You can try A and B together for a more robust solution.

    If the page is generated by a View

    Explore the View cache options in the UI and disable caching.

    If the page is a node

    You can assign the node's render array ($build) a cache max-age<> of <code>0, inside a hook_node_view:

    function mymodule_node_view(array &$build, NodeInterface $node, $display, $view_mode) {
      if ($node->getType() == 'yourtype' && YOUR_OTHER_CONDITIONS) {
        $build['#cache']['max-age'] = 0;
      }
    }
    

    I hope one of these methods works for you. However, patching Drupal Core is not the right approach for this issue. This issue and the patches provided here address a different purpose and should not be used as a solution in your case.

    And again, please delete the patch you provided, as it could mislead other developers searching for a fix.

Production build 0.71.5 2024