PasswordController.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. using Microsoft.AspNetCore.Mvc;
  2. using bitforum.DTOs.Request;
  3. using bitforum.DTOs.Response;
  4. using bitforum.Constants;
  5. using bitforum.Helpers;
  6. using bitforum.Services;
  7. using bitforum.Repository;
  8. namespace bitforum.Controllers.API.Auth
  9. {
  10. [ApiController]
  11. [Route("api/auth")]
  12. public class PasswordController : ControllerBase
  13. {
  14. private readonly ILogger<PasswordController> _logger;
  15. private readonly DefaultDbContext _db;
  16. private readonly IConfigService _configService;
  17. private readonly IMailService _mailService;
  18. private readonly IMemberRepository _memberRepository;
  19. private ResultDto _result = new ResultDto();
  20. public PasswordController(ILogger<PasswordController> logger, DefaultDbContext db, IConfigService configService, IMailService mailService, IEmailVerifyNumberRepository emailVerifyNumberRepository, IMemberRepository memberRepository)
  21. {
  22. _logger = logger;
  23. _db = db;
  24. _configService = configService;
  25. _mailService = mailService;
  26. _memberRepository = memberRepository;
  27. }
  28. // 비밀번호 재설정 요청
  29. [HttpGet("forgot-password/{email}")]
  30. public async Task<ActionResult<ResultDto>> ForgotPassword([FromRoute] string email)
  31. {
  32. try
  33. {
  34. if (string.IsNullOrEmpty(email))
  35. {
  36. throw new Exception("잘못된 접근입니다.");
  37. }
  38. // 이메일 거부 목록 확인
  39. if (_configService.IsDeniedEmail(email))
  40. {
  41. throw new Exception($"`{email}`은 조회할 수 없습니다.");
  42. }
  43. // 회원 확인
  44. var member = await _memberRepository.FindMemberByEmail(email);
  45. if (member is null)
  46. {
  47. throw new Exception("회원 정보를 찾을 수 없습니다.");
  48. }
  49. // 차단 회원은 거부
  50. if (member.IsDenied)
  51. {
  52. throw new Exception("권한이 없습니다.");
  53. }
  54. await _mailService.SendForgotPasswordEmailAsync(member);
  55. _result.Message = "이메일 인증 확인 후 비밀번호를 변경할 수 있습니다.";
  56. }
  57. catch (Exception e)
  58. {
  59. _logger.LogError(e, e.Message);
  60. _result.Ok = false;
  61. _result.Status = StatusCodes.Status400BadRequest;
  62. _result.Message = e.Message;
  63. }
  64. return _result;
  65. }
  66. // 비밀번호 변경 처리
  67. [HttpPost("reset-password")]
  68. public async Task<ActionResult<ResultDto>> ResetPassword([FromBody] ResetPasswordDto request)
  69. {
  70. try
  71. {
  72. var cookieName = $"isVerified-{VerificationType.ForgotPassword}";
  73. if (!Request.Cookies.ContainsKey(cookieName) || Request.Cookies[cookieName] != "true")
  74. {
  75. throw new Exception("사전 인증을 먼저 수행하세요.");
  76. }
  77. // 유효성 검사
  78. if (!ModelState.IsValid)
  79. {
  80. _result.Errors = ModelState.GetErrors();
  81. throw new Exception("유효성 검사에 실패하였습니다.");
  82. }
  83. // 회원 확인
  84. var member = await _memberRepository.FindMemberByEmail(request.Email);
  85. if (member is null)
  86. {
  87. throw new Exception("회원 정보를 찾을 수 없습니다.");
  88. }
  89. var newPassword = BCrypt.Net.BCrypt.HashPassword(request.Password);
  90. if (member.Password == newPassword)
  91. {
  92. throw new Exception("기존 비밀번호와 동일합니다.");
  93. }
  94. var isValid = _configService.IsPasswordPolicyValid(request.Password);
  95. if (!isValid)
  96. {
  97. string message = "";
  98. int minLengthConfig = Config.Register.PasswordMinLength ?? 6;
  99. int uppercaseLengthConfig = Config.Register.PasswordUppercaseLength ?? 0;
  100. int numbersLengthConfig = Config.Register.PasswordNumbersLength ?? 0;
  101. int specialCharsLengthConfig = Config.Register.PasswordSpecialcharsLength ?? 0;
  102. // 위에 데이터로 안내 문구 만들어줘
  103. if (minLengthConfig > 0)
  104. {
  105. message += $"최소 {minLengthConfig}자 이상, ";
  106. }
  107. if (uppercaseLengthConfig > 0)
  108. {
  109. message += $"대문자 {uppercaseLengthConfig}자 이상, ";
  110. }
  111. if (numbersLengthConfig > 0)
  112. {
  113. message += $"숫자 {numbersLengthConfig}자 이상, ";
  114. }
  115. if (specialCharsLengthConfig > 0)
  116. {
  117. message += $"특수문자 {specialCharsLengthConfig}자 이상, ";
  118. }
  119. message = message.TrimEnd(',', ' ');
  120. throw new Exception($"입력하신 비밀번호는 사용하실 수 없습니다. 다음 조건을 충족해주세요.\n\n{message}");
  121. }
  122. member.Password = newPassword;
  123. member.PasswordUpdatedAt = DateTime.UtcNow;
  124. await _db.SaveChangesAsync();
  125. await _mailService.SendChangedPasswordEmailAsync(member);
  126. // 쿠키 삭제
  127. Response.Cookies.Delete(cookieName);
  128. _result.Message = "비밀번호가 변경되었습니다.";
  129. }
  130. catch (Exception e)
  131. {
  132. _logger.LogError(e, e.Message);
  133. _result.Ok = false;
  134. _result.Status = StatusCodes.Status400BadRequest;
  135. _result.Message = e.Message;
  136. }
  137. return _result;
  138. }
  139. }
  140. }