我正在从事一个asp.net核心mvc项目(.net 5),但在与身份相关的问题上苦苦挣扎。
我有一个类HedgehogUserAccount
,它继承自IdentityUser
,还有另外两个从HedgehogUserAccount
继承的类:CustomerAccount
和UserAccount
(我知道名称的选择很糟糕--当一切正常时,我会改变它)。在做了大量工作之后,我成功地让迁移工作起来,程序编译并运行,但是当我尝试注册一个User
时,我得到了以下错误:
NotSupportedException: No IUserTwoFactorTokenProvider<TUser> named 'Default' is registered.
错误来自我注册代码中的第90行(请参见下面粘贴的完整代码--它几乎与脚手架注册页面相同):
var code = await _userManager.GenerateEmailConfirmationTokenAsync(user);
我的Startup.cs的相关部分(我认为)如下所示:
services.AddIdentity<HedgehogUserAccount, IdentityRole>(options => options.SignIn.RequireConfirmedAccount = true)
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders()
.AddDefaultUI();
services.AddIdentityCore<UserAccount>(
options => options.SignIn.RequireConfirmedAccount = true)
.AddRoles<IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders()
.AddDefaultUI();
services.AddIdentityCore<CustomerAccount>(options => options.SignIn.RequireConfirmedAccount = true)
.AddRoles<IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders()
.AddDefaultUI();
当我试图注册我的UserManager
类时,我得到了这段代码。
我尝试了一些不同的方法(取代AddDefaultTokenProviders
):
.AddTokenProvider(TokenOptions.DefaultEmailProvider, typeof(EmailTokenProvider<UserAccount>))
.AddTokenProvider<EmailTokenProvider<UserAccount>>(TokenOptions.DefaultEmailProvider)
但在我的代码中,它们在相同的地方产生相同的错误。但是当我尝试的时候,我得到了一个有趣的信息:
services.AddIdentityCore<UserAccount>(
options => {
options.Tokens.ProviderMap.Add("Default", new
TokenProviderDescriptor(typeof(IUserTwoFactorTokenProvider<UserAccount>)));
})
.AddRoles<IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddDefaultTokenProviders()
.AddDefaultUI();
然后我得到的错误消息是:System.ArgumentException: 'An item with the same key has already been added. Key: Default'
,它感觉有点奇怪,因为上一条消息说不存在Default
。
我想我必须做的是以某种方式提供我自己的EmailTokenProvider
?或者还有别的办法吗?我非常感谢你的帮助!
更新
我尝试用以下行替换生成确认令牌的行:
var code = "testtoken";//await _userManager.GenerateEmailConfirmationTokenAsync(user);
当我这样做,程序和注册运行完美无缺!我猜这意味着默认的令牌提供程序在某种程度上没有被正确注册,但是我不知道如何继续。
Register.cshtml.cs
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Text.Encodings.Web;
using System.Threading.Tasks;
using Hedgehog.Core.Application;
using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Identity.UI.Services;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.RazorPages;
using Microsoft.AspNetCore.WebUtilities;
using Microsoft.Extensions.Logging;
namespace Hedgehog.UI.Areas.Identity.Pages.Account
{
[AllowAnonymous]
public class RegisterModel : PageModel
{
private readonly SignInManager<UserAccount> _signInManager;
private readonly UserManager<UserAccount> _userManager;
private readonly RoleManager<IdentityRole> _roleManager;
private readonly ILogger<RegisterModel> _logger;
private readonly IEmailSender _emailSender;
public RegisterModel(
UserManager<UserAccount> userManager,
RoleManager<IdentityRole> roleManager,
SignInManager<UserAccount> signInManager,
ILogger<RegisterModel> logger,
IEmailSender emailSender)
{
_userManager = userManager;
_roleManager = roleManager;
_signInManager = signInManager;
_logger = logger;
_emailSender = emailSender;
}
[BindProperty]
public InputModel Input { get; set; }
public string ReturnUrl { get; set; }
public IList<AuthenticationScheme> ExternalLogins { get; set; }
public class InputModel
{
[Required]
[EmailAddress]
[Display(Name = "Email")]
public string Email { get; set; }
[Required]
[StringLength(100, ErrorMessage = "The {0} must be at least {2} and at max {1} characters long.", MinimumLength = 6)]
[DataType(DataType.Password)]
[Display(Name = "Password")]
public string Password { get; set; }
[DataType(DataType.Password)]
[Display(Name = "Confirm password")]
[Compare("Password", ErrorMessage = "The password and confirmation password do not match.")]
public string ConfirmPassword { get; set; }
}
public async Task OnGetAsync(string returnUrl = null)
{
ReturnUrl = returnUrl;
ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList();
}
public async Task<IActionResult> OnPostAsync(string returnUrl = null)
{
returnUrl ??= Url.Content("~/");
ExternalLogins = (await _signInManager.GetExternalAuthenticationSchemesAsync()).ToList();
if (ModelState.IsValid)
{
var user = new UserAccount { UserName = Input.Email, Email = Input.Email };
var result = await _userManager.CreateAsync(user, Input.Password);
if (result.Succeeded)
{
_logger.LogInformation("User created a new account with password.");
_userManager.AddToRoleAsync(user, "User").Wait();
var code = await _userManager.GenerateEmailConfirmationTokenAsync(user);
code = WebEncoders.Base64UrlEncode(Encoding.UTF8.GetBytes(code));
var callbackUrl = Url.Page(
"/Account/ConfirmEmail",
pageHandler: null,
values: new { area = "Identity", userId = user.Id, code = code, returnUrl = returnUrl },
protocol: Request.Scheme);
await _emailSender.SendEmailAsync(Input.Email, "Confirm your email",
$"Please confirm your account by <a href='{HtmlEncoder.Default.Encode(callbackUrl)}'>clicking here</a>.");
if (_userManager.Options.SignIn.RequireConfirmedAccount)
{
return RedirectToPage("RegisterConfirmation", new { email = Input.Email, returnUrl = returnUrl });
}
else
{
await _signInManager.SignInAsync(user, isPersistent: false);
return LocalRedirect(returnUrl);
}
}
foreach (var error in result.Errors)
{
ModelState.AddModelError(string.Empty, error.Description);
}
}
// If we got this far, something failed, redisplay form
return Page();
}
}
}
发布于 2021-05-04 10:31:48
在经历了许多头痛之后,我现在对我的问题得出了一个结论。我通过实现自己的令牌提供程序并将其注册到系统来解决这个问题。如果其他人和我一样撞上了墙,我就把我的解决方案留在这里。
步骤1:实现IUserTwoFactorTokenProvider
public class HedgehogEmailTwoFactorAuthentication<TUser> : IUserTwoFactorTokenProvider<TUser> where TUser : HedgehogUserAccount
{
public Task<bool> CanGenerateTwoFactorTokenAsync(UserManager<TUser> manager, TUser user)
{
if (manager != null && user != null)
{
return Task.FromResult(true);
}
else
{
return Task.FromResult(false);
}
}
// Genereates a simple token based on the user id, email and another string.
private string GenerateToken(HedgehogUserAccount user, string purpose)
{
string secretString = "coffeIsGood";
return secretString + user.Email + purpose + user.Id;
}
public Task<string> GenerateAsync(string purpose, UserManager<TUser> manager, TUser user)
{
return Task.FromResult(GenerateToken(user, purpose));
}
public Task<bool> ValidateAsync(string purpose, string token, UserManager<TUser> manager, TUser user)
{
return Task.FromResult(token == GenerateToken(user, purpose));
}
}
步骤2:在服务中注册新的和闪亮的令牌提供程序
services.AddIdentityCore<CustomerAccount>(options => options.SignIn.RequireConfirmedAccount = true)
.AddRoles<IdentityRole>()
.AddEntityFrameworkStores<ApplicationDbContext>()
.AddTokenProvider("Default", typeof(HedgehogEmailTwoFactorAuthentication<CustomerAccount>))
.AddDefaultUI();
步骤3:使用DI容器注册
在ConfigureContainer中(我使用Autofac):
builder.RegisterType<HedgehogEmailTwoFactorAuthentication<CustomerAccount>>()
.As<IUserTwoFactorTokenProvider<CustomerAccount>>()
.InstancePerLifetimeScope();
如果不使用Autofac,则需要调用AddScoped
而不是InstancePerLifetimeScope
(假设这是令牌生成器的正确生存期)。
另外,您可能希望使用更好的令牌生成器来实现:)
参考资料:Microsoft文档
https://stackoverflow.com/questions/67361087
复制相似问题