AspNetCore.Docs/aspnetcore/performance/caching/output.md

12 KiB

title author description monikerRange ms.author ms.custom ms.date uid
Output caching middleware in ASP.NET Core tdykstra Learn how to configure and use output caching middleware in ASP.NET Core. >= aspnetcore-7.0 riande mvc 12/06/2022 performance/caching/output

Output caching middleware in ASP.NET Core

By Tom Dykstra

:::moniker range=">= aspnetcore-7.0"

This article explains how to configure output caching middleware in an ASP.NET Core app. For an introduction to output caching, see Output caching.

The output caching middleware can be used in all types of ASP.NET Core apps: Minimal API, Web API with controllers, MVC, and Razor Pages. The sample app is a Minimal API, but every caching feature it illustrates is also supported in the other app types.

Add the middleware to the app

Add the output caching middleware to the service collection by calling xref:Microsoft.Extensions.DependencyInjection.OutputCacheServiceCollectionExtensions.AddOutputCache%2A.

Add the middleware to the request processing pipeline by calling xref:Microsoft.AspNetCore.Builder.OutputCacheApplicationBuilderExtensions.UseOutputCache%2A.

[!NOTE]

Configure one endpoint or page

For minimal API apps, configure an endpoint to do caching by calling CacheOutput, or by applying the [OutputCache] attribute, as shown in the following examples:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="oneendpoint":::

For apps with controllers, apply the [OutputCache] attribute to the action method. For Razor Pages apps, apply the attribute to the Razor page class.

Configure multiple endpoints or pages

Create policies when calling AddOutputCache to specify caching configuration that applies to multiple endpoints. A policy can be selected for specific endpoints, while a base policy provides default caching configuration for a collection of endpoints.

The following highlighted code configures caching for all of the app's endpoints, with expiration time of 10 seconds. If an expiration time isn't specified, it defaults to one minute.

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies1" highlight="3-4":::

The following highlighted code creates two policies, each specifying a different expiration time. Selected endpoints can use the 20 second expiration, and others can use the 30 second expiration.

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies1" highlight="5-8":::

You can select a policy for an endpoint when calling the CacheOutput method or using the [OutputCache] attribute:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="selectpolicy":::

For apps with controllers, apply the [OutputCache] attribute to the action method. For Razor Pages apps, apply the attribute to the Razor page class.

Default output caching policy

By default, output caching follows these rules:

  • Only HTTP 200 responses are cached.
  • Only HTTP GET or HEAD requests are cached.
  • Responses that set cookies aren't cached.
  • Responses to authenticated requests aren't cached.

The following code applies all of the default caching rules to all of an app's endpoints:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies3":::

Override the default policy

The following code shows how to override the default rules. The highlighted lines in the following custom policy code enable caching for HTTP POST methods and HTTP 301 responses:

:::code language="csharp" source="output/samples/7.x/MyCustomPolicy.cs" highlight="50,68":::

To use this custom policy, create a named policy:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies3b":::

And select the named policy for an endpoint:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="post":::

Alternative default policy override

Alternatively, use Dependency Injection (DI) to initialize an instance, with the following changes to the custom policy class:

  • A public constructor instead of a private constructor.
  • Eliminate the Instance property in the custom policy class.

For example:

:::code language="csharp" source="output/samples/7.x/MyCustomPolicy2.cs" id="fordi":::

The remainder of the class is the same as shown previously. Add the custom policy as shown in the following example:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies3c":::

The preceding code uses DI to create the instance of the custom policy class. Any public arguments in the constructor are resolved.

When using a custom policy as a base policy, don't call OutputCache() (with no arguments) on any endpoint that the base policy should apply to. Calling OutputCache() adds the default policy to the endpoint.

Specify the cache key

By default, every part of the URL is included as the key to a cache entry, that is, the scheme, host, port, path, and query string. However, you might want to explicitly control the cache key. For example, suppose you have an endpoint that returns a unique response only for each unique value of the culture query string. Variation in other parts of the URL, such as other query strings, shouldn't result in different cache entries. You can specify such rules in a policy, as shown in the following highlighted code:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies2" highlight="7":::

You can then select the VaryByQuery policy for an endpoint:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="selectquery":::

Here are some of the options for controlling the cache key:

Use xref:Microsoft.AspNetCore.OutputCaching.OutputCacheOptions.UseCaseSensitivePaths?displayProperty=nameWithType to specify that the path part of the key is case sensitive. The default is case insensitive.

For more options, see the xref:Microsoft.AspNetCore.OutputCaching.OutputCachePolicyBuilder class.

Cache revalidation

Cache revalidation means the server can return a 304 Not Modified HTTP status code instead of the full response body. This status code informs the client that the response to the request is unchanged from what the client previously received.

The following code illustrates the use of an Etag header to enable cache revalidation. If the client sends an If-None-Match header with the etag value of an earlier response, and the cache entry is fresh, the server returns 304 Not Modified instead of the full response:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="etag":::

Another way to do cache revalidation is to check the date of the cache entry creation compared to the date requested by the client. When the request header If-Modified-Since is provided, output caching returns 304 if the cached entry is older and isn't expired.

Cache revalidation is automatic in response to these headers sent from the client. No special configuration is required on the server to enable this behavior, aside from enabling output caching.

Use tags to evict cache entries

You can use tags to identify a group of endpoints and evict all cache entries for the group. For example, the following code creates a pair of endpoints whose URLs begin with "blog", and tags them "tag-blog":

:::code language="csharp" source="output/samples/7.x/Program.cs" id="tagendpoint":::

An alternative way to assign tags for the same pair of endpoints is to define a base policy that applies to endpoints that begin with blog:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies2" highlight="3-5":::

Another alternative is to call MapGroup:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="taggroup":::

In the preceding tag assignment examples, both endpoints are identified by the tag-blog tag. You can then evict the cache entries for those endpoints with a single statement that references that tag:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="evictbytag":::

With this code, an HTTP POST request sent to https://localhost:<port>/purge/tag-blog will evict cache entries for these endpoints.

You might want a way to evict all cache entries for all endpoints. To do that, create a base policy for all endpoints as the following code does:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies2" highlight="6":::

This base policy enables you to use the "tag-all" tag to evict everything in cache.

Disable resource locking

By default, resource locking is enabled to mitigate the risk of cache stampede and thundering herd. For more information, see Output Caching.

To disable resource locking, call SetLocking(false) while creating a policy, as shown in the following example:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="policies2" highlight="9":::

The following example selects the no-locking policy for an endpoint:

:::code language="csharp" source="output/samples/7.x/Program.cs" id="selectnolock":::

Limits

The following properties of xref:Microsoft.AspNetCore.OutputCaching.OutputCacheOptions let you configure limits that apply to all endpoints:

Cache storage

xref:Microsoft.AspNetCore.OutputCaching.IOutputCacheStore is used for storage. By default it's used with xref:System.Runtime.Caching.MemoryCache. We don't recommend xref:Microsoft.Extensions.Caching.Distributed.IDistributedCache for use with output caching. IDistributedCache doesn't have atomic features, which are required for tagging. We recommend that you create custom xref:Microsoft.AspNetCore.OutputCaching.IOutputCacheStore implementations by using direct dependencies on the underlying storage mechanism, such as Redis.

See also

:::moniker-end