c# - ASP.NET MVC5 中的密码恢复

标签 c# asp.net-identity-2 asp.net-mvc-5.1

我正在使用 ASP.NET MVC 5 应用程序。用户可以毫无问题地注册和登录。然而,当一个用户忘记他/她的密码时,忘记密码过程(已经存在)不会做任何事情!不会向用户发送带有单击此处重置密码链接的电子邮件。

目前我的 ForgotPassword 操作方法如下所示:

[HttpPost]
[AllowAnonymous]
[ValidateAntiForgeryToken]
public async Task<ActionResult> ForgotPassword(ForgotPasswordViewModel model)
{
    if (ModelState.IsValid)
    {
        var user = await UserManager.FindByNameAsync(model.Email);
        if (user == null || !(await UserManager.IsEmailConfirmedAsync(user.Id)))
        {
            // Don't reveal that the user does not exist or is not confirmed
            return View("ForgotPasswordConfirmation");
        }
    }

    // If we got this far, something failed, redisplay form
    return View(model);
}

我猜这留给开发人员来实现。我四处搜索,没有发现任何直接的东西。

允许这样做的最简单方法是什么?

最佳答案

忘记密码操作以生成重置 token :

[HttpPost]
[AllowAnonymous]
[ValidateAntiForgeryToken]
public async Task<ActionResult> ForgotPassword(ForgotPasswordViewModel model)
{
    if (ModelState.IsValid)
    {
        var user = await UserManager.FindByNameAsync(model.Email);
        if (user == null || !(await UserManager.IsEmailConfirmedAsync(user.Id)))
        {
            // Don't reveal that the user does not exist or is not confirmed
            return View("ForgotPasswordConfirmation");
        }

        // For more information on how to enable account confirmation and password reset please visit http://go.microsoft.com/fwlink/?LinkID=320771
        // Send an email with this link
        string code = await UserManager.GeneratePasswordResetTokenAsync(user.Id);
        var callbackUrl = Url.Action("ResetPassword", "Account", new { userId = user.Id, code = code }, protocol: Request.Url.Scheme);      
        await UserManager.SendEmailAsync(user.Id, "Reset Password", "Please reset your password by clicking <a href=\"" + callbackUrl + "\">here</a>");
        return RedirectToAction("ForgotPasswordConfirmation", "Account");
    }

    // If we got this far, something failed, redisplay form
    return View(model);
}

重置密码操作以根据生成的 token 重置密码:

[AllowAnonymous]
public ActionResult ResetPassword(string code)
{
    return code == null ? View("Error") : View();
}

[HttpPost]
[AllowAnonymous]
[ValidateAntiForgeryToken]
public async Task<ActionResult> ResetPassword(ResetPasswordViewModel model)
{
    if (!ModelState.IsValid)
    {
        return View(model);
    }
    var user = await UserManager.FindByNameAsync(model.Email);
    if (user == null)
    {
        // Don't reveal that the user does not exist
        return RedirectToAction("ResetPasswordConfirmation", "Account");
    }
    var result = await UserManager.ResetPasswordAsync(user.Id, model.Code, model.Password);
    if (result.Succeeded)
    {
        return RedirectToAction("ResetPasswordConfirmation", "Account");
    }
    AddErrors(result);
    return View();
}

相关 View 模型:

public class ResetPasswordViewModel
{
    public string Email { get; set; }
    public string Password { get; set; }
    public string ConfirmPassword { get; set; }
    public string Code { get; set; }
}

public class ForgotPasswordViewModel
{
    public string Email { get; set; }
}

但是在发送邮件之前需要先配置Email服务。

public class EmailService : IIdentityMessageService
{
    public Task SendAsync(IdentityMessage message)
    {
        return configSendGridasync(message);
    }

    private Task configSendGridasync(IdentityMessage message)
    {
        var myMessage = new SendGridMessage();
        myMessage.AddTo(message.Destination);
        myMessage.From = new System.Net.Mail.MailAddress(
                      "you@somewhere.com", "My name");
        myMessage.Subject = message.Subject;
        myMessage.Text = message.Body;
        myMessage.Html = message.Body;

        var credentials = new NetworkCredential("userName","Password");

        // Create a Web transport for sending email.
        var transportWeb = new Web(credentials);

       // Send the email.
       if (transportWeb != null)
       {
           return transportWeb.DeliverAsync(myMessage);
       }
       else
       {
           return Task.FromResult(0);
       }
   }
}

最后,您需要在用户管理器配置器中注册此类身份并添加以下行:

public static ApplicationUserManager Create(IdentityFactoryOptions<ApplicationUserManager> options, IOwinContext context)
{
    var manager = new ApplicationUserManager(new UserStore<ApplicationUser>(context.Get<ApplicationDbContext>()));

    // some code here

    manager.EmailService = new EmailService();
}

参见 Account Confirmation and Password Recovery with ASP.NET Identity (C#)作为分步教程。

关于c# - ASP.NET MVC5 中的密码恢复,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32025709/

相关文章:

c# - 何时调用 SaveChanges

HttpListener 的 C# 问题

c# - MaskedTextBox 多行asciionly

c# - 如何使用 Identity v2 从 Startup.Auth.cs 公开/ token 端点

twitter-bootstrap-3 - MVC5和Bootstrap3:Html.Editor对于错误的类?如何改变?

c# - 如何使用 Try 和 Catch 来设置值

asp.net-identity-2 - 没有 Entity Framework 的 ASP.NET Identity 2.0 实现

asp.net-mvc - 只有当用户的用户名与数据库中的电子邮件完全相同时,我才能登录

c# - 将 HttpWebResponse 转换为 HttpResponseMessage

oauth-2.0 - 使用 WebAPI 的 VS.2015 MVC 中没有帐户 Controller View ?