AspNetCore.Docs/aspnetcore/fundamentals/change-tokens.md

198 lines
14 KiB
Markdown
Raw Normal View History

---
2017-11-18 03:22:43 +08:00
title: Detect changes with change tokens in ASP.NET Core
author: guardrex
2017-11-18 03:22:43 +08:00
description: Learn how to use change tokens to track changes.
ms.author: riande
2017-11-11 05:08:08 +08:00
ms.date: 11/10/2017
uid: fundamentals/change-tokens
---
2017-11-18 03:22:43 +08:00
# Detect changes with change tokens in ASP.NET Core
By [Luke Latham](https://github.com/guardrex)
2017-11-18 03:22:43 +08:00
A *change token* is a general-purpose, low-level building block used to track changes.
[View or download sample code](https://github.com/aspnet/AspNetCore.Docs/tree/master/aspnetcore/fundamentals/change-tokens/sample/) ([how to download](xref:index#how-to-download-a-sample))
2017-11-11 05:08:08 +08:00
## IChangeToken interface
2018-06-03 08:04:59 +08:00
[IChangeToken](/dotnet/api/microsoft.extensions.primitives.ichangetoken) propagates notifications that a change has occurred. `IChangeToken` resides in the [Microsoft.Extensions.Primitives](/dotnet/api/microsoft.extensions.primitives) namespace. For apps that don't use the [Microsoft.AspNetCore.App metapackage](xref:fundamentals/metapackage-app) (ASP.NET Core 2.1 or later), reference the [Microsoft.Extensions.Primitives](https://www.nuget.org/packages/Microsoft.Extensions.Primitives/) NuGet package in the project file.
2017-11-11 05:08:08 +08:00
2017-11-29 05:30:49 +08:00
`IChangeToken` has two properties:
2017-11-11 05:08:08 +08:00
2017-11-29 05:41:05 +08:00
* [ActiveChangedCallbacks](/dotnet/api/microsoft.extensions.primitives.ichangetoken.activechangecallbacks) indicate if the token proactively raises callbacks. If `ActiveChangedCallbacks` is set to `false`, a callback is never called, and the app must poll `HasChanged` for changes. It's also possible for a token to never be cancelled if no changes occur or the underlying change listener is disposed or disabled.
2017-11-11 05:08:08 +08:00
* [HasChanged](/dotnet/api/microsoft.extensions.primitives.ichangetoken.haschanged) gets a value that indicates if a change has occurred.
2017-11-29 05:41:05 +08:00
The interface has one method, [RegisterChangeCallback(Action<Object>, Object)](/dotnet/api/microsoft.extensions.primitives.ichangetoken.registerchangecallback), which registers a callback that's invoked when the token has changed. `HasChanged` must be set before the callback is invoked.
2017-11-11 05:08:08 +08:00
## ChangeToken class
2018-06-03 08:04:59 +08:00
`ChangeToken` is a static class used to propagate notifications that a change has occurred. `ChangeToken` resides in the [Microsoft.Extensions.Primitives](/dotnet/api/microsoft.extensions.primitives) namespace. For apps that don't use the [Microsoft.AspNetCore.App metapackage](xref:fundamentals/metapackage-app), reference the [Microsoft.Extensions.Primitives](https://www.nuget.org/packages/Microsoft.Extensions.Primitives/) NuGet package in the project file.
2017-11-29 05:30:49 +08:00
The `ChangeToken` [OnChange(Func<IChangeToken>, Action)](/dotnet/api/microsoft.extensions.primitives.changetoken.onchange?view=aspnetcore-2.0#Microsoft_Extensions_Primitives_ChangeToken_OnChange_System_Func_Microsoft_Extensions_Primitives_IChangeToken__System_Action_) method registers an `Action` to call whenever the token changes:
2018-06-01 09:03:31 +08:00
* `Func<IChangeToken>` produces the token.
* `Action` is called when the token changes.
2017-11-29 05:41:05 +08:00
`ChangeToken` has an [OnChange&lt;TState&gt;(Func&lt;IChangeToken&gt;, Action&lt;TState&gt;, TState)](/dotnet/api/microsoft.extensions.primitives.changetoken.onchange?view=aspnetcore-2.0#Microsoft_Extensions_Primitives_ChangeToken_OnChange__1_System_Func_Microsoft_Extensions_Primitives_IChangeToken__System_Action___0____0_) overload that takes an additional `TState` parameter that's passed into the token consumer `Action`.
2017-11-11 05:08:08 +08:00
`OnChange` returns an [IDisposable](/dotnet/api/system.idisposable). Calling [Dispose](/dotnet/api/system.idisposable.dispose) stops the token from listening for further changes and releases the token's resources.
2017-11-18 03:22:43 +08:00
## Example uses of change tokens in ASP.NET Core
2017-11-29 05:30:49 +08:00
Change tokens are used in prominent areas of ASP.NET Core monitoring changes to objects:
2017-11-11 05:08:08 +08:00
* For monitoring changes to files, [IFileProvider](/dotnet/api/microsoft.extensions.fileproviders.ifileprovider)'s [Watch](/dotnet/api/microsoft.extensions.fileproviders.ifileprovider.watch) method creates an `IChangeToken` for the specified files or folder to watch.
* `IChangeToken` tokens can be added to cache entries to trigger cache evictions on change.
2017-11-11 05:08:08 +08:00
* For `TOptions` changes, the default [OptionsMonitor](/dotnet/api/microsoft.extensions.options.optionsmonitor-1) implementation of [IOptionsMonitor](/dotnet/api/microsoft.extensions.options.ioptionsmonitor-1) has an overload that accepts one or more [IOptionsChangeTokenSource](/dotnet/api/microsoft.extensions.options.ioptionschangetokensource-1) instances. Each instance returns an `IChangeToken` to register a change notification callback for tracking options changes.
## Monitoring for configuration changes
2018-08-25 03:10:45 +08:00
By default, ASP.NET Core templates use [JSON configuration files](xref:fundamentals/configuration/index#json-configuration-provider) (*appsettings.json*, *appsettings.Development.json*, and *appsettings.Production.json*) to load app configuration settings.
2017-11-11 05:08:08 +08:00
2018-08-25 03:10:45 +08:00
These files are configured using the [AddJsonFile(IConfigurationBuilder, String, Boolean, Boolean)](/dotnet/api/microsoft.extensions.configuration.jsonconfigurationextensions.addjsonfile#Microsoft_Extensions_Configuration_JsonConfigurationExtensions_AddJsonFile_Microsoft_Extensions_Configuration_IConfigurationBuilder_System_String_System_Boolean_System_Boolean_) extension method on [ConfigurationBuilder](/dotnet/api/microsoft.extensions.configuration.configurationbuilder) that accepts a `reloadOnChange` parameter (ASP.NET Core 1.1 and later). `reloadOnChange` indicates if configuration should be reloaded on file changes. See this setting in the [WebHost](/dotnet/api/microsoft.aspnetcore.webhost) convenience method [CreateDefaultBuilder](/dotnet/api/microsoft.aspnetcore.webhost.createdefaultbuilder):
```csharp
config.AddJsonFile("appsettings.json", optional: true, reloadOnChange: true)
.AddJsonFile($"appsettings.{env.EnvironmentName}.json", optional: true, reloadOnChange: true);
```
File-based configuration is represented by [FileConfigurationSource](/dotnet/api/microsoft.extensions.configuration.fileconfigurationsource). `FileConfigurationSource` uses [IFileProvider](/dotnet/api/microsoft.extensions.fileproviders.ifileprovider) to monitor files.
By default, the `IFileMonitor` is provided by a [PhysicalFileProvider](/dotnet/api/microsoft.extensions.fileproviders.physicalfileprovider), which uses [FileSystemWatcher](/dotnet/api/system.io.filesystemwatcher) to monitor for configuration file changes.
2017-11-29 05:30:49 +08:00
The sample app demonstrates two implementations for monitoring configuration changes. If either the *appsettings.json* file changes or the Environment version of the file changes, each implementation executes custom code. The sample app writes a message to the console.
2017-11-11 05:08:08 +08:00
A configuration file's `FileSystemWatcher` can trigger multiple token callbacks for a single configuration file change. The sample's implementation guards against this problem by checking file hashes on the configuration files. Checking file hashes ensures that at least one of the configuration files has changed before running the custom code. The sample uses SHA1 file hashing (*Utilities/Utilities.cs*):
[!code-csharp[](change-tokens/sample/Utilities/Utilities.cs?name=snippet1)]
2017-11-11 05:08:08 +08:00
A retry is implemented with an exponential back-off. The re-try is present because file locking may occur that temporarily prevents computing a new hash on one of the files.
2017-11-18 03:22:43 +08:00
### Simple startup change token
2017-11-11 05:08:08 +08:00
Register a token consumer `Action` callback for change notifications to the configuration reload token (*Startup.cs*):
[!code-csharp[](change-tokens/sample/Startup.cs?name=snippet2)]
2017-11-11 05:08:08 +08:00
`config.GetReloadToken()` provides the token. The callback is the `InvokeChanged` method:
[!code-csharp[](change-tokens/sample/Startup.cs?name=snippet3)]
2017-11-11 05:08:08 +08:00
2017-11-29 05:41:05 +08:00
The `state` of the callback is used to pass in the `IHostingEnvironment`. This is useful to determine the correct *appsettings* configuration JSON file to monitor, *appsettings.&lt;Environment&gt;.json*. File hashes are used to prevent the `WriteConsole` statement from running multiple times due to multiple token callbacks when the configuration file has only changed once.
2017-11-11 05:08:08 +08:00
This system runs as long as the app is running and can't be disabled by the user.
### Monitoring configuration changes as a service
2017-11-29 05:30:49 +08:00
The sample implements:
* Basic startup token monitoring.
* Monitoring as a service.
* A mechanism to enable and disable monitoring.
The sample establishes an `IConfigurationMonitor` interface (*Extensions/ConfigurationMonitor.cs*):
2017-11-11 05:08:08 +08:00
[!code-csharp[](change-tokens/sample/Extensions/ConfigurationMonitor.cs?name=snippet1)]
2017-11-11 05:08:08 +08:00
The constructor of the implemented class, `ConfigurationMonitor`, registers a callback for change notifications:
[!code-csharp[](change-tokens/sample/Extensions/ConfigurationMonitor.cs?name=snippet2)]
`config.GetReloadToken()` supplies the token. `InvokeChanged` is the callback method. The `state` in this instance is a reference to the `IConfigurationMonitor` instance that is used to access the monitoring state. Two properties are used:
2017-11-11 05:08:08 +08:00
* `MonitoringEnabled` indicates if the callback should run its custom code.
* `CurrentState` describes the current monitoring state for use in the UI.
The `InvokeChanged` method is similar to the earlier approach, except that it:
* Doesn't run its code unless `MonitoringEnabled` is `true`.
* Notes the current `state` in its `WriteConsole` output.
[!code-csharp[](change-tokens/sample/Extensions/ConfigurationMonitor.cs?name=snippet3)]
2017-11-11 05:08:08 +08:00
An instance `ConfigurationMonitor` is registered as a service in `ConfigureServices` of *Startup.cs*:
[!code-csharp[](change-tokens/sample/Startup.cs?name=snippet1)]
2017-11-11 05:08:08 +08:00
The Index page offers the user control over configuration monitoring. The instance of `IConfigurationMonitor` is injected into the `IndexModel`:
[!code-csharp[](change-tokens/sample/Pages/Index.cshtml.cs?name=snippet1)]
2017-11-11 05:08:08 +08:00
2017-11-29 05:30:49 +08:00
A button enables and disables monitoring:
2017-11-11 05:08:08 +08:00
[!code-cshtml[](change-tokens/sample/Pages/Index.cshtml?range=35)]
2017-11-11 05:08:08 +08:00
[!code-csharp[](change-tokens/sample/Pages/Index.cshtml.cs?name=snippet2)]
2017-11-11 05:08:08 +08:00
When `OnPostStartMonitoring` is triggered, monitoring is enabled, and the current state is cleared. When `OnPostStopMonitoring` is triggered, monitoring is disabled, and the state is set to reflect that monitoring isn't occurring.
## Monitoring cached file changes
File content can be cached in-memory using [IMemoryCache](/dotnet/api/microsoft.extensions.caching.memory.imemorycache). In-memory caching is described in the [Cache in-memory](xref:performance/caching/memory) topic. Without taking additional steps, such as the implementation described below, *stale* (outdated) data is returned from a cache if the source data changes.
2017-11-29 05:30:49 +08:00
Not taking into account the status of a cached source file when renewing a [sliding expiration](/dotnet/api/microsoft.extensions.caching.memory.memorycacheentryoptions.slidingexpiration) period leads to stale cache data. Each request for the data renews the sliding expiration period, but the file is never reloaded into the cache. Any app features that use the file's cached content are subject to possibly receiving stale content.
Using change tokens in a file caching scenario prevents stale file content in the cache. The sample app demonstrates an implementation of the approach.
2017-11-29 05:30:49 +08:00
The sample uses `GetFileContent` to:
2017-11-29 05:30:49 +08:00
* Return file content.
* Implement a retry algorithm with exponential back-off to cover cases where a file lock is temporarily preventing a file from being read.
2017-11-29 05:30:49 +08:00
*Utilities/Utilities.cs*:
[!code-csharp[](change-tokens/sample/Utilities/Utilities.cs?name=snippet2)]
2017-11-18 03:22:43 +08:00
A `FileService` is created to handle cached file lookups. The `GetFileContent` method call of the service attempts to obtain file content from the in-memory cache and return it to the caller (*Services/FileService.cs*).
2017-11-29 05:30:49 +08:00
If cached content isn't found using the cache key, the following actions are taken:
1. The file content is obtained using `GetFileContent`.
2017-11-29 05:30:49 +08:00
1. A change token is obtained from the file provider with [IFileProviders.Watch](/dotnet/api/microsoft.extensions.fileproviders.ifileprovider.watch). The token's callback is triggered when the file is modified.
2017-11-18 03:22:43 +08:00
1. The file content is cached with a [sliding expiration](/dotnet/api/microsoft.extensions.caching.memory.memorycacheentryoptions.slidingexpiration) period. The change token is attached with [MemoryCacheEntryExtensions.AddExpirationToken](/dotnet/api/microsoft.extensions.caching.memory.memorycacheentryextensions.addexpirationtoken) to evict the cache entry if the file changes while it's cached.
[!code-csharp[](change-tokens/sample/Services/FileService.cs?name=snippet1)]
2017-11-18 03:22:43 +08:00
The `FileService` is registered in the service container along with the memory caching service (*Startup.cs*):
[!code-csharp[](change-tokens/sample/Startup.cs?name=snippet4)]
The page model loads the file's content using the service (*Pages/Index.cshtml.cs*):
[!code-csharp[](change-tokens/sample/Pages/Index.cshtml.cs?name=snippet3)]
## CompositeChangeToken class
For representing one or more `IChangeToken` instances in a single object, use the [CompositeChangeToken](/dotnet/api/microsoft.extensions.primitives.compositechangetoken) class.
```csharp
var firstCancellationTokenSource = new CancellationTokenSource();
var secondCancellationTokenSource = new CancellationTokenSource();
var firstCancellationToken = firstCancellationTokenSource.Token;
var secondCancellationToken = secondCancellationTokenSource.Token;
var firstCancellationChangeToken = new CancellationChangeToken(firstCancellationToken);
var secondCancellationChangeToken = new CancellationChangeToken(secondCancellationToken);
var compositeChangeToken =
new CompositeChangeToken(
new List<IChangeToken>
{
firstCancellationChangeToken,
secondCancellationChangeToken
});
```
2017-11-11 05:08:08 +08:00
`HasChanged` on the composite token reports `true` if any represented token `HasChanged` is `true`. `ActiveChangeCallbacks` on the composite token reports `true` if any represented token `ActiveChangeCallbacks` is `true`. If multiple concurrent change events occur, the composite change callback is invoked exactly one time.
2017-11-18 03:22:43 +08:00
2018-06-01 09:03:31 +08:00
## Additional resources
2017-11-18 03:22:43 +08:00
* <xref:performance/caching/memory>
* <xref:performance/caching/distributed>
* <xref:performance/caching/response>
* <xref:performance/caching/middleware>
* <xref:mvc/views/tag-helpers/builtin-th/cache-tag-helper>
* <xref:mvc/views/tag-helpers/builtin-th/distributed-cache-tag-helper>