ttempleton / craft-nocache
A Twig extension to escape caching inside cache blocks
Installs: 45 647
Dependents: 1
Suggesters: 0
Security: 0
Stars: 46
Watchers: 3
Forks: 6
Open Issues: 5
Type:craft-plugin
Requires
- php: ^8.0.2
- craftcms/cms: ^4.0.0|^5.0.0-alpha
Requires (Dev)
- craftcms/ecs: dev-main
- craftcms/rector: dev-main
- 3.x-dev
- 3.0.4
- 3.0.3
- 3.0.2
- 3.0.1
- 3.0.0
- 3.0.0-beta.1
- 2.x-dev
- 2.0.8
- 2.0.7
- 2.0.6
- 2.0.5
- 2.0.4
- 2.0.3
- 2.0.2
- 2.0.1
- 2.0.0
- dev-dependabot/composer/craftcms/cms-5.5.2
- dev-dependabot/composer/symfony/http-client-6.4.15
- dev-dependabot/composer/craftcms/cms-5.4.8
- dev-dependabot/composer/twig/twig-3.14.2
- dev-dependabot/composer/symfony/process-7.1.7
This package is auto-updated.
Last update: 2024-12-18 19:49:07 UTC
README
A Craft CMS Twig extension that escapes caching inside cache blocks
{% cache %} This will be cached {% nocache %} This won't be {% endnocache %} {% endcache %}
It also works when disabling the cache from included files:
{% cache %} This will be cached {% include 'template' %} {% endcache %}
template.twig:
{% nocache %} This won't be {% endnocache %}
If you need to reference variables outside of the nocache
tag, you will need to pass a context – much like how the include
tag works when passing variables. Note that you do not have to pass global variables, such as craft
or currentUser
, but you will have to import your macros again.
{% set variable = 5 %} {% nocache with {x: variable} %} The following value should be 5: {{ x }} {% endnocache %}
Requirements
No-Cache requires Craft CMS 4.0.0 or later.
Installation
No-Cache can be installed from the Craft Plugin Store or with Composer.
Craft Plugin Store
Open your project's control panel, navigate to the Plugin Store, search for No-Cache and click Install.
Composer
Open your terminal, navigate to your project's root directory and run the following command:
composer require ttempleton/craft-nocache
Then open your project's control panel, navigate to Settings → Plugins, find No-Cache and click Install.
Example: User information
Say you have a list of products you want to show on your page. Under each product, you want an "add to cart" button. However, you only want to show this button if a user is logged in. Not only that, but you also want to disable the button if the user already has it in their cart. Unfortunately you're outputting 20 products a page with images, so caching the list seems like the responsible thing to do.
{% cache %} {% for product in craft.entries().section('products').limit(20).all() %} <article> <figure>{{ product.image.one().img }}</figure> <h1>{{ product.title }}</h1> {% if currentUser %} <button{{ currentUser.cart.id(product.id).count() > 0 ? ' disabled' }}>Add to cart</button> {% endif %} </article> {% endfor %} {% endcache %}
Now we have a problem. The cache around the list of products will cause the currentUser
logic to essentially not work, since they'll be cached along with the products. You can't isolate the user logic by separating things into multiple cache blocks, since you're in a loop, and the whole point was to cache the database call that grabs the product entries. So you either have to apply your user checking in Javascript (far from ideal), or disregard caching altogether.
With nocache
tags you can fix this very easily:
{% cache %} {% for product in craft.entries().section('products').limit(20).all() %} <article> <figure>{{ product.image.one().img }}</figure> <h1>{{ product.title }}</h1> {% nocache with {productId: product.id} %} {% if currentUser %} <button{{ currentUser.cart.id(productId).count() > 0 ? ' disabled' }}>Add to cart</button> {% endif %} {% endnocache %} </article> {% endfor %} {% endcache %}
The nocache
block will allow you to cache the entire product list, but still perform your user logic outside of the cache. It also allows passing of context, so you can still refer to products and entries inside the nocache
block and access their properties, without any additional database calls.
Example: CSRF tokens
A fantastic security feature, but one that basically renders caching impossible to use. Often you might find yourself outputting a form to the frontend, but it's nested deep within a stack of template includes and macros. At the top of this stack you've conveniently wrapped a cache tag around it.
Well, now your CSRF tokens are going to be cached and there's basically nothing you can do about it. Using nocache
tags, this is no longer a problem:
<form> {% nocache %}{{ csrfInput() }}{% endnocache %} ... </form>
Now you can include this form anywhere in your templates and not have to worry about your CSRF tokens being cached. As a side note, yes nocache
tags will work even when they are not inside a cache block (though try and avoid doing this as nocache
tags do add some overhead).
Caveat
Content inside nocache
blocks will render slightly different than normal. Variables declared outside of the nocache
block will actually have their values cached for the duration of the cache block.
This causes an issue in situations like the following:
{% set article = craft.entries().section('news').one() %} {% cache %} ... {% nocache with {article: article} %} {{ article.title }} {% endnocache %} {% endcache %}
You would expect that if you were to change the title of the article, it will update inside the nocache
block. This is not the case, as the article itself would be cached due to the cache block.
There's a few ways around this. You could move the {% set articles %}
statement within the cache block, so updating the article would cause the cache to bust. In situations where you are using the article inside the cache (but outside the nocache
block) this is the preferred method, as you won't spend any database calls grabbing the article inside the nocache
block.
{% cache %} {% set article = craft.entries().section('news').one() %} ... {% nocache with {article: article} %} {{ article.title }} {% endnocache %} {% endcache %}
The other option is to query for the article inside the nocache
block. This can be better than the above solution as updating the article title will not bust the cache. The difference in this situation is now the contents of the nocache
block will cause a database call.
{% cache %} ... {% nocache %} {% set article = craft.entries().section('news').one() %} {{ article.title }} {% endnocache %} {% endcache %}
Every situation will be different, so use your best judgement.
Big thanks to Ben Fleming for creating No-Cache and for letting me take it over.