Handler.cs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. using Application.Abstractions.Authentication;
  2. using Application.Abstractions.Cache;
  3. using Application.Abstractions.Data;
  4. using Application.Helpers;
  5. using Domain.Entities.Members.Logs;
  6. using MediatR;
  7. using SharedKernel;
  8. using SharedKernel.Results;
  9. using Microsoft.EntityFrameworkCore;
  10. using Microsoft.Extensions.Logging;
  11. using Microsoft.Extensions.Options;
  12. namespace Application.Features.Api.Auth.Login;
  13. internal sealed class Handler(
  14. IAppDbContext db,
  15. IJwtTokenProvider jwtTokenProvider,
  16. IOptions<AppSettings> options,
  17. ILogger<Handler> logger,
  18. ICacheService cache
  19. ) : IRequestHandler<Command, Result<Response>> {
  20. private readonly AppSettings.JwtSection _jwt = options.Value.JWT;
  21. public async Task<Result<Response>> Handle(Command request, CancellationToken ct)
  22. {
  23. // 유효성 검사
  24. if (string.IsNullOrWhiteSpace(request.Email))
  25. {
  26. return Result.Failure<Response>(Error.Problem("Auth.EmailRequired", "이메일은 필수입니다."));
  27. }
  28. if (string.IsNullOrWhiteSpace(request.Password))
  29. {
  30. return Result.Failure<Response>(Error.Problem("Auth.PasswordRequired", "비밀번호는 필수입니다."));
  31. }
  32. // Config 로드
  33. var accountConfig = await AccountConfigLoader.GetAccountConfigAsync(cache, db, ct);
  34. // Member 조회 (비밀번호 검증을 위해 Tracking 모드)
  35. var email = request.Email.Trim().ToLower();
  36. var member = await db.Member.FirstOrDefaultAsync(m => m.Email == email, ct);
  37. if (member is null)
  38. {
  39. // 실패 로그 기록 (회원 없음)
  40. await LogLoginAttempt(null, false, email, "회원 정보 없음", request, ct);
  41. return Result.Failure<Response>(Error.Unauthorized("Auth.InvalidCredentials", "이메일 또는 비밀번호가 올바르지 않습니다."));
  42. }
  43. // 탈퇴 회원 거부
  44. if (member.IsWithdraw)
  45. {
  46. await LogLoginAttempt(member.ID, false, email, "탈퇴 회원", request, ct);
  47. return Result.Failure<Response>(Error.Problem("Auth.MemberWithdrawn", "탈퇴한 회원은 이용할 수 없습니다."));
  48. }
  49. // 차단 회원 거부
  50. if (member.IsDenied)
  51. {
  52. await LogLoginAttempt(member.ID, false, email, "차단 회원", request, ct);
  53. return Result.Failure<Response>(Error.Problem("Auth.MemberDenied", "차단된 회원이므로 이용할 수 없습니다."));
  54. }
  55. // 로그인 시도 횟수 제한 확인
  56. if (accountConfig.MaxLoginTryCount is > 0 && accountConfig.MaxLoginTryLimitSecond is > 0)
  57. {
  58. var limitTime = DateTime.UtcNow.AddSeconds(-accountConfig.MaxLoginTryLimitSecond.Value);
  59. var failedCount = await db.MemberLoginLog.CountAsync(l => l.MemberID == member.ID && !l.Success && l.CreatedAt >= limitTime, ct);
  60. if (failedCount >= accountConfig.MaxLoginTryCount.Value)
  61. {
  62. await LogLoginAttempt(member.ID, false, email, "로그인 시도 횟수 초과", request, ct);
  63. return Result.Failure<Response>(Error.Problem("Auth.LoginTryExceeded", $"로그인 시도 횟수를 초과하였습니다. {accountConfig.MaxLoginTryLimitSecond}초 후에 다시 시도해주세요."));
  64. }
  65. }
  66. // 비밀번호 검증
  67. if (!member.VerifyPassword(request.Password))
  68. {
  69. await LogLoginAttempt(member.ID, false, email, "비밀번호 불일치", request, ct);
  70. return Result.Failure<Response>(Error.Unauthorized("Auth.InvalidCredentials", "이메일 또는 비밀번호가 올바르지 않습니다."));
  71. }
  72. // 로그인 시 이메일 인증자 여부 확인
  73. if (accountConfig.IsLoginEmailVerifiedOnly && !member.IsEmailVerified)
  74. {
  75. await LogLoginAttempt(member.ID, false, email, "이메일 미인증", request, ct);
  76. return Result.Failure<Response>(Error.Unauthorized("Auth.EmailNotVerified", "이메일 인증이 완료되지 않은 사용자는 로그인할 수 없습니다."));
  77. }
  78. // JWT 토큰 생성
  79. var accessToken = jwtTokenProvider.CreateAccessToken(member.ID, member.Email, member.Name);
  80. var refreshToken = jwtTokenProvider.CreateRefreshToken();
  81. var expiresAt = DateTime.UtcNow.AddMinutes(_jwt.AccessTokenExpiration);
  82. // RefreshToken 저장
  83. var refreshTokenEntity = Domain.Entities.Members.RefreshToken.Create(
  84. member.ID,
  85. refreshToken,
  86. DateTime.UtcNow.AddDays(_jwt.RefreshTokenExpiration)
  87. );
  88. await db.RefreshToken.AddAsync(refreshTokenEntity, ct);
  89. // 로그인 횟수 증가
  90. var memberStats = await db.MemberStats.FirstOrDefaultAsync(x => x.MemberID == member.ID, ct);
  91. if (memberStats is not null)
  92. {
  93. memberStats.LoginCount++;
  94. }
  95. // 성공 로그 기록
  96. await LogLoginAttempt(member.ID, true, email, null, request, ct);
  97. await db.SaveChangesAsync(ct);
  98. logger.LogInformation("{0} 로그인", member.Email);
  99. return Result.Success(new Response(accessToken, refreshToken, expiresAt));
  100. }
  101. private async Task LogLoginAttempt(int? memberID, bool success, string account, string? reason, Command request, CancellationToken ct)
  102. {
  103. var log = MemberLoginLog.Create(
  104. memberID,
  105. success,
  106. account,
  107. reason,
  108. ipAddress: request.IpAddress,
  109. userAgent: request.UserAgent
  110. );
  111. await db.MemberLoginLog.AddAsync(log, ct);
  112. await db.SaveChangesAsync(ct);
  113. }
  114. }