Handler.cs 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  1. using Application.Abstractions.Data;
  2. using Application.Abstractions.Cache;
  3. using Application.Abstractions.Messaging;
  4. using Application.Helpers;
  5. using SharedKernel.Results;
  6. using Microsoft.EntityFrameworkCore;
  7. namespace Application.Features.Api.MyPage.ChangePassword;
  8. internal sealed class Handler(
  9. IAppDbContext db,
  10. ICacheService cache
  11. ) : ICommandHandler<Command, Result>
  12. {
  13. public async Task<Result> Handle(Command request, CancellationToken ct)
  14. {
  15. if (string.IsNullOrWhiteSpace(request.CurrentPassword))
  16. {
  17. return Result.Failure(Error.Problem("MyPage.CurrentPasswordRequired", "현재 비밀번호는 필수입니다."));
  18. }
  19. if (string.IsNullOrWhiteSpace(request.NewPassword))
  20. {
  21. return Result.Failure(Error.Problem("MyPage.NewPasswordRequired", "새 비밀번호는 필수입니다."));
  22. }
  23. if (request.NewPassword != request.ConfirmPassword)
  24. {
  25. return Result.Failure(Error.Problem("MyPage.PasswordMismatch", "새 비밀번호가 일치하지 않습니다."));
  26. }
  27. // Config 로드 + 비밀번호 복잡도 검증
  28. var accountConfig = await AccountConfigLoader.GetAccountConfigAsync(cache, db, ct);
  29. var passwordResult = PasswordPolicyValidator.Validate(request.NewPassword, accountConfig);
  30. if (!passwordResult.IsSuccess)
  31. {
  32. return passwordResult;
  33. }
  34. var member = await db.Member.FirstOrDefaultAsync(m => m.ID == request.MemberID, ct);
  35. if (member is null)
  36. {
  37. return Result.Failure(Error.NotFound("MyPage.MemberNotFound", "회원 정보를 찾을 수 없습니다."));
  38. }
  39. // 비밀번호 변경 주기 확인
  40. if (accountConfig.ChangePasswordDay is > 0)
  41. {
  42. var nextChangeDate = member.PasswordUpdatedAt.AddDays(accountConfig.ChangePasswordDay.Value);
  43. if (DateTime.UtcNow < nextChangeDate)
  44. {
  45. return Result.Failure(Error.Problem("MyPage.PasswordChangeTooSoon",
  46. $"비밀번호는 {accountConfig.ChangePasswordDay}일마다 변경 가능합니다."));
  47. }
  48. }
  49. if (!member.VerifyPassword(request.CurrentPassword))
  50. {
  51. return Result.Failure(Error.Problem("MyPage.InvalidPassword", "현재 비밀번호가 올바르지 않습니다."));
  52. }
  53. member.SetPassword(request.NewPassword);
  54. await db.SaveChangesAsync(ct);
  55. return Result.Success();
  56. }
  57. }