首页 > 解决方案 > 如何在 ASP.NET Core 5 运行时动态重新配置服务?

问题描述

我在我的网络应用程序中使用 Google 身份验证,并且 OAuth 密钥当前在 ConfigureServices 中硬编码:

services.AddAuthentication()
    .AddGoogle(options =>
    {
        options.ClientId = "my-client-id";
        options.ClientSecret = "my-client-secret";
    });

但是,我想让站点管理员有机会从 Web 应用程序的设置页面更改 ClientId 和 ClientSecret,最好不必重新启动服务器。

为此,当用户在设置页面上点击“保存”时,我必须以某种方式触发 Google 服务和 GoogleOptions 对象的重新配置。这就是我遇到的麻烦。另外,我想将这些设置存储在 EF Core DbContext 中,而不是物理配置文件中。

到目前为止,我已经尝试将设置移动到实现 IPostConfigureOptions 的单独类中。这应该允许我注入我的数据库上下文,因为根据文档, PostConfigure 应该在所有其他配置发生后运行。设置已从这个新类正确加载,但数据库上下文的注入失败,并出现以下异常:

System.InvalidOperationException: Cannot consume scoped service 'AppDatabase' from singleton 'IOptionsMonitor`1[GoogleOptions]'

这很奇怪,因为 ConfigureGoogleOptions 注册为 Scoped,而不是 Singleton。

这是我的选项类:

public class ConfigureGoogleOptions : IPostConfigureOptions<GoogleOptions>
{
    private readonly AppDatabase database;

    public ConfigureGoogleOptions(AppDatabase database)
    {
        this.database = database;
    }
    
    public void PostConfigure(string name, GoogleOptions options)
    {
        options.ClientId = "my-client-id.apps.googleusercontent.com";
        options.ClientSecret = "my-client-secret";
    }
}

并在 ConfigureServices 中注册:

services.AddScoped<IPostConfigureOptions<GoogleOptions>, ConfigureGoogleOptions>();

即使数据库注入有效,还有第二个问题。我的类中的PostConfigure函数只在应用程序启动后被调用一次,并且再也不会被调用。我假设它将设置缓存在某处,我不知道如何使此缓存无效或禁用,以便我可以动态提供值。

简短摘要/ tl;博士:

我想从我自己的数据库中加载 Google OAuth 服务的 ClientId 和 ClientSecret 设置,并且我希望能够在服务器运行时动态更改它们。

标签: c#entity-frameworkasp.net-coredependency-injection

解决方案


在内部,Google处理程序将使用IOptionsMonitor<GoogleOptions>一次GoogleOptions 直到重新加载一次(例如,何时从配置文件绑定选项并保存文件将触发重新加载)。内部IOptionsMonitor将使用IOptionsMonitorCache并且此缓存注册为单例。因此,您从中获得的选项实例与用于处理程序内部各种操作的选项实例IOptionsMonitor<GoogleOptions>相同(参考) 。AuthenticationHandler<GoogleOptions>.Options如果使用该选项,即使其他代码也应该正确地从IOptionsMonitor<GoogleOptions>.

所以要在运行时更改选项,就像这样简单:

//inject the IOptionsMonitor<GoogleOptions> into _googleOptionsMonitor;
var runtimeOptions = _googleOptionsMonitor.Get(GoogleDefaults.AuthenticationScheme);
//you change properties of runtimeOptions here
//...

这里重要的一点是我们需要使用GoogleDefaults.AuthenticationScheme作为获取正确选项实例的键。将IOptionsMonitor.CurrentValue使用默认键Options.DefaultName(它是一个空字符串)。


推荐阅读