【发布时间】:2021-06-21 18:06:59
【问题描述】:
我在我的网络应用程序中使用 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;dr:
我想从我自己的数据库中加载 Google OAuth 服务的 ClientId 和 ClientSecret 设置,并且我希望能够在服务器运行时动态更改它们。
【问题讨论】:
标签: c# entity-framework asp.net-core dependency-injection