令牌无效。使用 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.Emptynull进行比较,这将总是返回 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 文件中添加&lt;machineKey validationKey="XXX" decryptionKey="XXX" validation="SHA1" decryption="AES"/&gt; 解决了这个问题。您的所有服务器都需要具有相同的机器密钥来验证之前生成的代码。

希望这会有所帮助。

【讨论】:

在 web.config 中添加什么?答案不完整。 嗨@MariuszIgnatowicz,代码标签没有显示配置,我为你修复了。 这适用于 ASP.NET Core 2.0 吗? ASP.NET Core 2.0 中没有 Web.Config 文件。 请在此处查看答案***.com/questions/46668192/…

以上是关于令牌无效。使用 UserManager.ConfirmEmailAsync(user.Id, code) 验证电子邮件验证码时的主要内容,如果未能解决你的问题,请参考以下文章

是否可以使用无状态逻辑(无数据库)使令牌无效?

当会话令牌无效时我应该使用啥状态码?

身份密码重置令牌无效

XSLT 无效令牌导致 XML 文档无效

PushSharp - 多个无效令牌不调用响应委托

AzureAD JWT 令牌受众声明前缀使 JWT 令牌无效