令牌无效。使用 UserManager.ConfirmEmailAsync(user.Id, code) 验证电子邮件验证码时
Posted
技术标签:
【中文标题】令牌无效。使用 UserManager.ConfirmEmailAsync(user.Id, code) 验证电子邮件验证码时【英文标题】:Invalid Token. while verifying email verification code using UserManager.ConfirmEmailAsync(user.Id, code) 【发布时间】:2014-09-26 12:40:41 【问题描述】:我最近将 Asp.net identity 1.0 迁移到 2.0 。我正在尝试使用以下方法验证电子邮件验证码。但我收到“无效令牌”错误消息。
public async Task<HttpResponseMessage> ConfirmEmail(string userName, string code)
ApplicationUser user = UserManager.FindByName(userName);
var result = await UserManager.ConfirmEmailAsync(user.Id, code);
return Request.CreateResponse(HttpStatusCode.OK, result);
使用以下代码生成电子邮件验证令牌(如果我在生成令牌后立即调用 ConfirmEmailAsyc,则工作正常)。但是当我使用不同的方法调用时会出错
public async Task<HttpResponseMessage> GetEmailConfirmationCode(string userName)
ApplicationUser user = UserManager.FindByName(userName);
var code = await UserManager.GenerateEmailConfirmationTokenAsync(user.Id);
//var result = await UserManager.ConfirmEmailAsync(user.Id, code);
return Request.CreateResponse(HttpStatusCode.OK, code);
请帮忙
【问题讨论】:
你找到答案了吗?我整天都在做这件事,而且它似乎确实有效。 嗨,它现在正在工作.. 这是因为我的疏忽。在对验证令牌进行 url 编码和解码后,它就可以工作了。因为我在令牌中有一些加号(+) Kumar,你能分享一下你是怎么解决这个问题的吗? @PitDigger - 一个简单的修复方法是使用 Replace(" ","+"); 在我的情况下(Asp.Net Core 3.0),脚手架页面似乎引入了这个错误。请在此处查看我的answer。 【参考方案1】:我发现您必须在将令牌放入电子邮件之前对其进行编码,但在之后检查时则不需要。所以我发送电子邮件的代码如下:
// Send an email with this link
string code = UserManager.GenerateEmailConfirmationToken(user.Id);
// added html encoding
string codeHtmlVersion = HttpUtility.UrlEncode(code);
// for some weird reason the following commented out line (which should return an absolute URL) returns null instead
// var callbackUrl = Url.Action("ConfirmEmail", "Account", new userId = user.Id, code = code , protocol: Request.Url.Scheme);
string callbackUrl = "(your URL)/Account/ConfirmEmail?userId=" +
user.Id + "&code=" + codeHtmlVersion;
// Send an email with this link using class (not shown here)
var m = new Email();
m.ToAddresses.Add(user.Email);
m.Subject = "Confirm email address for new account";
m.Body =
"Hi " + user.UserName + dcr +
"You have been sent this email because you created an account on our website. " +
"Please click on <a href =\"" + callbackUrl + "\">this link</a> to confirm your email address is correct. ";
确认电子邮件的代码如下:
// user has clicked on link to confirm email
[AllowAnonymous]
public async Task<ActionResult> ConfirmEmail(string userId, string code)
// email confirmation page
// don't HTTP decode
// try to authenticate
if (userId == null || code == null)
// report an error somehow
else
// check if token OK
var result = UserManager.ConfirmEmail(userId, code);
if (result.Succeeded)
// report success
else
// report failure
最终为我工作!
【讨论】:
HttpUtility.UrlEncode 最终解决了这个问题。非常感谢!【参考方案2】:您好,如果我获取 url(full) 并通过 WebClient 调用 api,就会发生这种情况。代码值必须在发送调用之前进行编码。
code = HttpUtility.UrlEncode(code);
【讨论】:
【参考方案3】:希望问题得到解决。否则,下面是运行良好的解决方案的链接。
Asp.NET - Identity 2 - Invalid Token Error
简单使用:
emailConfirmationCode = await
UserManager.GenerateEmailConfirmationTokenAsync(user.Id);
UserManager.ConfirmEmailAsync(userId, code1);
【讨论】:
【参考方案4】:我的问题略有不同。
我创建了自己的 IUserStore,但我做错的一件事是将 SecurityStamp 设置为 null,如果没有值。
安全标记用于生成令牌,但在生成令牌时被空字符串替换,但在验证令牌时不会被替换,因此最终将String.Empty
与null
进行比较,这将总是返回 false。
我通过在从数据库读取时替换 String.Empty
的空值来解决我的问题。
【讨论】:
如果您尝试同时验证 PhoneNumber 和 Email,则此 InvalidToken 可能会由于在调用 ChangePhoneNumberAsync() 以验证 PhoneNumber 时更改 SecurityStamp 而发生。如果这是您的问题,请提供有关此主题的更多见解:***.com/questions/36182915/…【参考方案5】:有同样的问题。 修复是在生成链接时对令牌进行 HTML 编码,并在确认时 - HTML 将其解码回来。
public async Task<IActionResult> ForgotPassword(ForgotPasswordViewModel model)
if (ModelState.IsValid)
var user = await _userManager.FindByEmailAsync(model.Email);
if (user == null )
// Don't reveal that the user does not exist or is not confirmed
return RedirectToAction(nameof(ForgotPasswordConfirmation));
var code = await _userManager.GeneratePasswordResetTokenAsync( user );
var codeHtmlVersion = HttpUtility.UrlEncode( code );
var callbackUrl = Url.ResetPasswordCallbackLink(user.Id, codeHtmlVersion, Request.Scheme);
await _emailSender.SendEmailAsync(
model.Email,
$"You can reset your password by clicking here: <a href='callbackUrl'>link</a>",
_logger );
return RedirectToAction(nameof(ForgotPasswordConfirmation));
// If we got this far, something failed, redisplay form
return View(model);
public async Task<IActionResult> ResetPassword(ResetPasswordViewModel model)
if (!ModelState.IsValid)
return View(model);
var user = await _userManager.FindByEmailAsync(model.Email);
if (user == null)
// Don't reveal that the user does not exist
return RedirectToAction(nameof(ResetPasswordConfirmation));
var codeHtmlDecoded = HttpUtility.UrlDecode( model.Code );
var result = await _userManager.ResetPasswordAsync(user, codeHtmlDecoded, model.Password);
if (result.Succeeded)
return RedirectToAction(nameof(ResetPasswordConfirmation));
AddErrors(result);
return View();
【讨论】:
【参考方案6】:我们遇到了同样的问题,负载平衡导致了这个问题。在 web.config 文件中添加<machineKey validationKey="XXX" decryptionKey="XXX" validation="SHA1" decryption="AES"/>
解决了这个问题。您的所有服务器都需要具有相同的机器密钥来验证之前生成的代码。
希望这会有所帮助。
【讨论】:
在 web.config 中添加什么?答案不完整。 嗨@MariuszIgnatowicz,代码标签没有显示配置,我为你修复了。 这适用于 ASP.NET Core 2.0 吗? ASP.NET Core 2.0 中没有 Web.Config 文件。 请在此处查看答案***.com/questions/46668192/…以上是关于令牌无效。使用 UserManager.ConfirmEmailAsync(user.Id, code) 验证电子邮件验证码时的主要内容,如果未能解决你的问题,请参考以下文章