Skip to content

Caching

Mark Croxton edited this page Mar 16, 2015 · 30 revisions

Do cache

Global fragments: rendered partials, such as navigational elements, footers, sidebars and other reusable, globally visible page elements (scope="site").

Addressable pages: the unique output of individual pages in your site, where the URI alone is sufficient to generate all possible interaction states ("URI addressability").

Don't cache

Non-global fragments: Partials that generate output that is specific to the URI or user.

Non-addressable pages: where the URI of a page alone is not sufficient to determine it's interaction state. E.g. forms that POST data, shopping carts, 404 pages.

Dynamic content

  • Search results
  • User-specific content**
  • Forms

Cache planning

Build for URI addressability: as far as possible plan your site so that all interaction states are represented by a unique URI.

Stay DRY: separate out commonly used global page fragments into snippets, Low Variables or Stash embeds. Cache structured data, not marked-up data, so you can re-use the data with different markups.

Group like items: use context and/or bundles to create groups of related cached items, so that they can be targetted for cache-breaking.

Use layers: use fragment-caching inside page-caching to allow for progressive cache rebuilds and avoid 'stampedes'.

Common pitfalls, and how to avoid them

Domain aliases - Cached domain-specific absolute URLs can lead to serious complications like broken AJAX requests due to cross-domain restrictions. Ideally you should only allow access to your website via a single known domain. If you can't be sure of this, try to eliminate absolute URLs entirely from your templates.

Non-strict URLs - It's very important to ensure that your templates return a 404 if additional segments are added to the URL or an entry doesn't exist. URL variants can bloat your cache.

Fragment caching

  {!-- save a list of 5 latest jobs for an hour --}
  {exp:stash:set_list 	
     name="jobs"	
     parse_tags="yes"	
     save="yes"	
     scope="site"	
     refresh="60"	
     replace="no"}	
     {exp:channel:entries channel="jobs" limit="5"}			
        {stash:job_title}{title}{/stash:pg_title}			
        {stash:job_url}{url_title}{/stash:job_url}	
     {/exp:channel:entries}
  {/exp:stash:set_list}

  {!-- Make the above a snippet or low variable and re-use --}
  {snippet:stash_set_jobs_list}

Page caching

With {exp:stash:cache}

  {!-- templates/default_site/blog/index.html --}
  {exp:stash:cache bundle="blog"}

     {stash:embed:layouts:standard}

     {!-- capture channel data for a single entry --}  
     {exp:channel:entries channel="blog" limit="1"}       
        {exp:stash:set}        
           {stash:pg_title}{title}{/stash:pg_title}       
           {stash:pg_intro}{cf_blog_intro}{/stash:pg_intro}        
           {stash:pg_body}{cf_blog_body}{/stash:pg_body}     
        {/exp:stash:set} 
     {/exp:channel:entries}
     
     {!-- this will be escaped --}   
     {stash:nocache}     
        Hi, {screen_name}   
     {/stash:nocache}
  {/exp:stash:cache}

Or with {stash:embed}

  {!-- templates/default_site/blog/index.html --}
  {stash:embed  
     name="@URI:base"  
     file_name="viewmodels:base_vm"  
     parse_stage="both"
  }

  {!-- stash_templates/viewmodels/base_vm.html --}
  {stash:embed:layouts:standard}

  {!-- capture channel data for a single entry --}  
  {exp:channel:entries channel="blog" limit="1"}       
     {exp:stash:set}        
        {stash:pg_title}{title}{/stash:pg_title}       
        {stash:pg_intro}{cf_blog_intro}{/stash:pg_intro}        
        {stash:pg_body}{cf_blog_body}{/stash:pg_body}     
     {/exp:stash:set} 
  {/exp:channel:entries}
  
  {!-- this will be escaped --}   
  {stash:nocache}     
     Hi, {screen_name}   
  {/stash:nocache}
Clone this wiki locally