Caching Web Pages
In this lesson, we will learn how to control response caching.
Responses returned by Web servers can be cached in various places during their path to their target client applications. They can be cached in various Internet intermediary nodes, and in the browser cache itself.
HTTP standard headers control the way a response must be cached and when it can be cached. ASP.NET Core supports Web response caching through the ResponseCacheAttribute
action filter that manipulates the HTTP headers that affect response caching. It also allows response caching on the Web server itself.
In the sections below, we will describe the HTTP caching headers, how to use the ResponseCacheAttribute
action filter, and how to cache responses on the server.
HTTP cache headers
The main HTTP header that affects response caching is Cache-Control
which can assume several commas separated values. The table below describes all allowed values:
Cache-Control value | Effect | Example |
---|---|---|
public | The response is not specific for a single user so, it can be stored in the browser and on all intermediary nodes. | public |
private | The response is specific for a single user so, it can be stored in the browser and in intermediary nodes that serve that specific user (usually just in the browser). | private |
max-age | The number of seconds the cache is considered valid. When this time expires the cached response is not accepted by the client and validation is asked to the server, which, in turn, can respond that the cached content is still valid or can provide a fresher response. | public, max-age=80 |
no-cache | The response can be cached but must be validated by the original server before being used. Therefore, a request to the server is issued anyway, to ask it if the cache is expired. | no-cache |
no-store | Response must not be cached. | no-store |
It is worth mentioning when max-age
expires with no-cache
, cache records continue to be stored. However, the server must confirm they are still valid. So the request must be transmitted to the server. Usually, the server answers with a new response that overrides all cached results.
Usually, servers also issue the Pragma: no-cache
header together with Cache-Control: no-cache
for retro compatibility with old specifications. However, the Pragma
header is now obsolete.
It is also possible to specify an absolute date+time after which the cache is no more valid with the Expires
header, as shown below:
Expires: Wed, 10 Mar 2021 07:28:00 GMT
The Vary
header is also important. It contains a comma-separated list of HTTP headers. A different cache entry is created for each combination of the listed header values, as shown below:
Vary: accept-encoding, accept-language
It is worth pointing out that GET
ASP.NET Core requests with different routes or query string parameters are cached in different entries since their URLs are different. Therefore, the cache must be refreshed only when a response to the same parameters changes.
Moreover, by default, ASP.NET Core only allows caching for GET
requests, as it assumes that the response to other verbs depends on the body content. Thus we should only worry about GET
requests caching.
The ResponseCacheAttribute action filter
The ResponseCacheAttribute
action filter can be used to configure all cache-related headers of an action method response. It can be applied both to the controller and to each action method. The action method configuration overrides the controller configuration.
It has a Location
property where ResponseCacheLocation.Any
, and ResponseCacheLocation.Client
values map respectively to the public
and private
Cache-Control
values.
The Duration
property maps to the max-age
Cache-Control
value.
The Vary
property maps directly to the Vary
header value.
Finally, when the NoStore
property is set, the no-store
value is added to the Cache-Control
header. While, when the Location
property is set to ResponseCacheLocation.None
the no-cache
value is added to the Cache-Control
header.
The example below allows caching of 3 minutes on both the browser and any intermediate node:
Get hands-on with 1300+ tech skills courses.