AspNetCore.Docs/aspnetcore/fundamentals/configuration/options.md

16 KiB

title author description monikerRange ms.author ms.custom ms.date uid
Options pattern in ASP.NET Core rick-anderson Discover how to use the options pattern to represent groups of related settings in ASP.NET Core apps. >= aspnetcore-3.1 riande mvc 01/13/2022 fundamentals/configuration/options

Options pattern in ASP.NET Core

[!INCLUDE]

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

By Rick Anderson.

The options pattern uses classes to provide strongly typed access to groups of related settings. When configuration settings are isolated by scenario into separate classes, the app adheres to two important software engineering principles:

  • Encapsulation:
    • Classes that depend on configuration settings depend only on the configuration settings that they use.
  • Separation of Concerns:
    • Settings for different parts of the app aren't dependent or coupled to one another.

Options also provide a mechanism to validate configuration data. For more information, see the Options validation section.

This article provides information on the options pattern in ASP.NET Core. For information on using the options pattern in console apps, see Options pattern in .NET.

Bind hierarchical configuration

[!INCLUDE]

Options interfaces

xref:Microsoft.Extensions.Options.IOptions%601:

xref:Microsoft.Extensions.Options.IOptionsSnapshot%601:

xref:Microsoft.Extensions.Options.IOptionsMonitor%601:

Post-configuration scenarios enable setting or changing options after all xref:Microsoft.Extensions.Options.IConfigureOptions%601 configuration occurs.

xref:Microsoft.Extensions.Options.IOptionsFactory%601 is responsible for creating new options instances. It has a single xref:Microsoft.Extensions.Options.IOptionsFactory%601.Create%2A method. The default implementation takes all registered xref:Microsoft.Extensions.Options.IConfigureOptions%601 and xref:Microsoft.Extensions.Options.IPostConfigureOptions%601 and runs all the configurations first, followed by the post-configuration. It distinguishes between xref:Microsoft.Extensions.Options.IConfigureNamedOptions%601 and xref:Microsoft.Extensions.Options.IConfigureOptions%601 and only calls the appropriate interface.

xref:Microsoft.Extensions.Options.IOptionsMonitorCache%601 is used by xref:Microsoft.Extensions.Options.IOptionsMonitor%601 to cache TOptions instances. The xref:Microsoft.Extensions.Options.IOptionsMonitorCache%601 invalidates options instances in the monitor so that the value is recomputed (xref:Microsoft.Extensions.Options.IOptionsMonitorCache%601.TryRemove%2A). Values can be manually introduced with xref:Microsoft.Extensions.Options.IOptionsMonitorCache%601.TryAdd%2A. The xref:Microsoft.Extensions.Options.IOptionsMonitorCache%601.Clear%2A method is used when all named instances should be recreated on demand.

Use IOptionsSnapshot to read updated data

Using xref:Microsoft.Extensions.Options.IOptionsSnapshot%601:

  • Options are computed once per request when accessed and cached for the lifetime of the request.
  • May incur a significant performance penalty because it's a Scoped service and is recomputed per request. For more information, see this GitHub issue and Improve the performance of configuration binding.
  • Changes to the configuration are read after the app starts when using configuration providers that support reading updated configuration values.

The difference between IOptionsMonitor and IOptionsSnapshot is that:

  • IOptionsMonitor is a Singleton service that retrieves current option values at any time, which is especially useful in singleton dependencies.
  • IOptionsSnapshot is a Scoped service and provides a snapshot of the options at the time the IOptionsSnapshot<T> object is constructed. Options snapshots are designed for use with transient and scoped dependencies.

The following code uses xref:Microsoft.Extensions.Options.IOptionsSnapshot%601.

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/Pages/TestSnap.cshtml.cs" id="snippet":::

The following code registers a configuration instance which MyOptions binds against:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/program.cs" id="snippet":::

In the preceding code, changes to the JSON configuration file after the app has started are read.

IOptionsMonitor

The following code registers a configuration instance which MyOptions binds against.

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/program.cs" id="snippet":::

The following example uses xref:Microsoft.Extensions.Options.IOptionsMonitor%601:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/Pages/TestMonitor.cshtml.cs" id="snippet":::

In the preceding code, by default, changes to the JSON configuration file after the app has started are read.

Named options support using IConfigureNamedOptions

Named options:

  • Are useful when multiple configuration sections bind to the same properties.
  • Are case sensitive.

Consider the following appsettings.json file:

:::code language="json" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/appsettings.NO.json":::

Rather than creating two classes to bind TopItem:Month and TopItem:Year, the following class is used for each section:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/Models/TopItemSettings.cs":::

The following code configures the named options:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/program.cs" id="snippet_om":::

The following code displays the named options:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/Pages/TestNO.cshtml.cs" id="snippet":::

All options are named instances. xref:Microsoft.Extensions.Options.IConfigureOptions%601 instances are treated as targeting the Options.DefaultName instance, which is string.Empty. xref:Microsoft.Extensions.Options.IConfigureNamedOptions%601 also implements xref:Microsoft.Extensions.Options.IConfigureOptions%601. The default implementation of the xref:Microsoft.Extensions.Options.IOptionsFactory%601 has logic to use each appropriately. The null named option is used to target all of the named instances instead of a specific named instance. xref:Microsoft.Extensions.DependencyInjection.OptionsServiceCollectionExtensions.ConfigureAll%2A and xref:Microsoft.Extensions.DependencyInjection.OptionsServiceCollectionExtensions.PostConfigureAll%2A use this convention.

OptionsBuilder API

xref:Microsoft.Extensions.Options.OptionsBuilder%601 is used to configure TOptions instances. OptionsBuilder streamlines creating named options as it's only a single parameter to the initial AddOptions<TOptions>(string optionsName) call instead of appearing in all of the subsequent calls. Options validation and the ConfigureOptions overloads that accept service dependencies are only available via OptionsBuilder.

OptionsBuilder is used in the Options validation section.

See Use AddOptions to configure custom repository for information adding a custom repository.

Use DI services to configure options

Services can be accessed from dependency injection while configuring options in two ways:

We recommend passing a configuration delegate to xref:Microsoft.Extensions.Options.OptionsBuilder%601.Configure%2A, since creating a service is more complex. Creating a type is equivalent to what the framework does when calling xref:Microsoft.Extensions.Options.OptionsBuilder%601.Configure%2A. Calling xref:Microsoft.Extensions.Options.OptionsBuilder%601.Configure%2A registers a transient generic xref:Microsoft.Extensions.Options.IConfigureNamedOptions%601, which has a constructor that accepts the generic service types specified.

Options validation

Options validation enables option values to be validated.

Consider the following appsettings.json file:

:::code language="json" source="~/fundamentals/configuration/options/samples/3.x/OptionsValidationSample/appsettings.Dev2.json":::

The following class is used to bind to the "MyConfig" configuration section and applies a couple of DataAnnotations rules:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/3.x/OptionsValidationSample/Configuration/MyConfigOptions.cs" id="snippet":::

The following code:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Program.cs" id="snippet":::

The ValidateDataAnnotations extension method is defined in the Microsoft.Extensions.Options.DataAnnotations NuGet package. For web apps that use the Microsoft.NET.Sdk.Web SDK, this package is referenced implicitly from the shared framework.

The following code displays the configuration values or the validation errors:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Controllers/HomeController.cs" id="snippet":::

The following code applies a more complex validation rule using a delegate:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Program.cs" id="snippet_mc":::

IValidateOptions<TOptions> and IValidatableObject

The following class implements xref:Microsoft.Extensions.Options.IValidateOptions%601:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Configuration/MyConfigValidation.cs" id="snippet":::

IValidateOptions enables moving the validation code out of Program.cs and into a class.

Using the preceding code, validation is enabled in Program.cs with the following code:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Program.cs" id="snippet_xm":::

Options validation also supports xref:System.ComponentModel.DataAnnotations.IValidatableObject. To perform class-level validation of a class within the class itself:

ValidateOnStart

Options validation runs the first time an xref:Microsoft.Extensions.Options.IOptions%601, xref:Microsoft.Extensions.Options.IOptionsSnapshot%601, or xref:Microsoft.Extensions.Options.IOptionsMonitor%601 implementation is created. To run options validation eagerly, when the app starts, call xref:Microsoft.Extensions.DependencyInjection.OptionsBuilderExtensions.ValidateOnStart%2A in Program.cs:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Snippets/Program.cs" id="snippet_ValidateOnStart" highlight="4":::

Options post-configuration

Set post-configuration with xref:Microsoft.Extensions.Options.IPostConfigureOptions%601. Post-configuration runs after all xref:Microsoft.Extensions.Options.IConfigureOptions%601 configuration occurs:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Program.cs" id="snippet_p1" highlight="10-99":::

xref:Microsoft.Extensions.Options.IPostConfigureOptions%601.PostConfigure%2A is available to post-configure named options:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/program.cs" id="snippet_nmo" highlight="10-14":::

Use xref:Microsoft.Extensions.DependencyInjection.OptionsServiceCollectionExtensions.PostConfigureAll%2A to post-configure all configuration instances:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsValidationSample/Program.cs" id="snippet_p3" highlight="10-99":::

Access options in Program.cs

To access xref:Microsoft.Extensions.Options.IOptions%601 or xref:Microsoft.Extensions.Options.IOptionsMonitor%601 in Program.cs, call xref:Microsoft.Extensions.DependencyInjection.ServiceProviderServiceExtensions.GetRequiredService%2A on xref:Microsoft.AspNetCore.Builder.WebApplication.Services%2A?displayProperty=nameWithType:

:::code language="csharp" source="~/fundamentals/configuration/options/samples/6.x/OptionsSample/program.cs" id="snippet_grs":::

Additional resources

:::moniker-end

[!INCLUDE]