- 🇮🇳India Vivek Panicker Kolkata
I can replicate this issue in D10.3 also in a view block.
- Merge request !10Issue #3115736: Filtered out tokens before populating cache tags. → (Open) created by Vivek Panicker
- 🇮🇳India Vivek Panicker Kolkata
Updated the MR to make it compatible with the latest code version.
- Status changed to Needs review
about 1 month ago 12:59am 24 February 2025 - 🇨🇦Canada mdolnik
@Berdir
I don't understand why they would get duplicated though. Sounds to me like you might have another view that doesn't have the same argument or not value for it?
I am having the same issue as everyone else when embedding a view on a page.
How to reproduce:
- Add a view with a contextual filter.
- Add custom cache tag for:
my-custom-tag:{{ arguments.field_name }}
- Add a custom page controller, and add an embedded view via:
$render_array = Views::getView($view_name)->buildRenderable($display_name, [$arg]);
- Visit the page
- Inspect the results of
$render_array['#cache']['tags']
and notice that the tokens remain un-resolved
After investigation, the following is occurring:
-
buildRenderable()
will:- Call
DisplayPluginBase::buildRenderable()
- Which then calls
DisplayPluginBase::applyDisplayCacheabilityMetadata()
- Which then calls
ViewExecutable::getCacheTags()
- Which ultimately calls
CustomTag::getCacheTags()
- Call
-
The issue here is that the arguments are not provided to
DisplayPluginBase::applyDisplayCacheabilityMetadata()
at this stage so there are no values to process the tokens with.- It is at this point where the raw un-resolved cache tags are added to the render array and ultimately stored in the DB
-
Then, when the view gets rendered, it first caches the results data:
View::preRenderViewElement()
is called during render.- Which then calls
ViewExecutable::executeDisplay()
- Which then calls
Block::execute()
- Which then calls
ViewExecutable::render()
- Which, if storing new cache data calls
CachePluginBase::cacheSet()
- Which then calls
CustomTag::getCacheTags()
- NOW, since the view is built properly, the arguments are present and the tokens get processed correctly.
-
Further down the render pipeline,
ViewExecutable::render()
will then:- Call
DisplayPluginBase::render()
. - Which then calls
DisplayPluginBase::applyDisplayCacheabilityMetadata()
- Which then calls
ViewExecutable::getCacheTags()
- Which ultimately calls
CustomTag::getCacheTags()
- Call
- BUT, the issue here is
applyDisplayCacheabilityMetadata()
is applying the cache tags to an existing element, and this element already contains the cache tags with the un-resolved tokens from the first call toCustomTag::getCacheTags()
, so the element ends up getting both resolved and unresolved tags.
As for blocks, the process is the same, where
ViewsBlock::build()
will callDisplayPluginBase::buildRenderable()
.I was playing around with modifying
CustomTag::getCacheTags()
in an attempt to extract the arguments from the element/render array before callingtokenizeValue()
:if (empty($this->view->build_info['substitutions'])) { if (!empty($this->view->element['#arguments'])) { $this->view->setArguments($this->view->element['#arguments']); // This seems to be the only public way to call ViewExecutable::_buildArguments() without executing the view. $this->view->buildTitle(); } }
...Which seemed to work for embedded views where the arguments are passed into
buildRenderable()
, but I did not have luck with embedded blocks as it doesn't seem to have the arguments by the time it callsbuildRenderable()
.With my specific issues, the other proposed code in this issue queue seems to resolve the invalid cache tags, while retaining proper caching.
Since the issue is only in the initial render array building and the views render and views results caching end up providing the proper cache tags, changing
CustomTag::getCacheTags()
to omit the initial broken cache tags seems to have no ill effects.