Handler.cs 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  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.GoogleLogin;
  13. internal sealed class Handler(
  14. IAppDbContext db,
  15. IJwtTokenProvider jwtTokenProvider,
  16. IGoogleTokenValidator googleTokenValidator,
  17. IOptions<AppSettings> options,
  18. ILogger<Handler> logger,
  19. ICacheService cache
  20. ) : IRequestHandler<Command, Result<Response>> {
  21. private readonly AppSettings.JwtSection _jwt = options.Value.JWT;
  22. public async Task<Result<Response>> Handle(Command request, CancellationToken ct)
  23. {
  24. // 유효성 검사
  25. if (string.IsNullOrWhiteSpace(request.Credential))
  26. {
  27. return Result.Failure<Response>(Error.Problem("Auth.CredentialRequired", "Google 인증 정보가 필요합니다."));
  28. }
  29. // Config에서 GoogleClientId 조회
  30. var config = await db.Config.AsNoTracking().OrderByDescending(x => x.ID).FirstOrDefaultAsync(ct);
  31. var googleClientId = config?.External.GoogleClientId;
  32. if (string.IsNullOrWhiteSpace(googleClientId))
  33. {
  34. return Result.Failure<Response>(Error.Problem("Auth.GoogleNotConfigured", "Google 로그인이 설정되지 않았습니다."));
  35. }
  36. // Google ID Token 검증
  37. var googleUser = await googleTokenValidator.ValidateAsync(request.Credential, googleClientId, ct);
  38. if (googleUser is null)
  39. {
  40. return Result.Failure<Response>(Error.Unauthorized("Auth.GoogleTokenInvalid", "Google 인증에 실패했습니다."));
  41. }
  42. var email = googleUser.Email.Trim().ToLower();
  43. // Member 조회
  44. var member = await db.Member.FirstOrDefaultAsync(m => m.Email == email, ct);
  45. if (member is null)
  46. {
  47. // 자동 회원가입
  48. var accountConfig = await AccountConfigLoader.GetAccountConfigAsync(cache, db, ct);
  49. if (accountConfig.IsRegisterBlock)
  50. {
  51. return Result.Failure<Response>(Error.Problem("Auth.RegisterBlocked", "현재 회원가입이 차단되어 있습니다."));
  52. }
  53. member = Domain.Entities.Members.Member.Create(email);
  54. await db.Member.AddAsync(member, ct);
  55. await db.SaveChangesAsync(ct);
  56. // 연관 엔티티 생성
  57. await db.MemberApprove.AddAsync(
  58. Domain.Entities.Members.MemberApprove.Create(member.ID), ct
  59. );
  60. await db.MemberStats.AddAsync(
  61. Domain.Entities.Members.MemberStats.Create(member.ID), ct
  62. );
  63. await db.Wallet.AddAsync(
  64. Domain.Entities.Wallets.Wallet.Create(member.ID), ct
  65. );
  66. // 구글 인증이므로 이메일 인증 완료 처리
  67. member.MarkEmailVerified();
  68. await db.SaveChangesAsync(ct);
  69. logger.LogInformation("Google 자동 회원가입: {Email}", email);
  70. }
  71. // 탈퇴 회원 거부
  72. if (member.IsWithdraw)
  73. {
  74. await LogLoginAttempt(member.ID, false, email, "탈퇴 회원 (Google)", request, ct);
  75. return Result.Failure<Response>(Error.Problem("Auth.MemberWithdrawn", "탈퇴한 회원은 이용할 수 없습니다."));
  76. }
  77. // 차단 회원 거부
  78. if (member.IsDenied)
  79. {
  80. await LogLoginAttempt(member.ID, false, email, "차단 회원 (Google)", request, ct);
  81. return Result.Failure<Response>(Error.Problem("Auth.MemberDenied", "차단된 회원이므로 이용할 수 없습니다."));
  82. }
  83. // JWT 토큰 생성
  84. var accessToken = jwtTokenProvider.CreateAccessToken(member.ID, member.Email, member.Name);
  85. var refreshToken = jwtTokenProvider.CreateRefreshToken();
  86. var expiresAt = DateTime.UtcNow.AddMinutes(_jwt.AccessTokenExpiration);
  87. // RefreshToken 저장
  88. var refreshTokenEntity = Domain.Entities.Members.RefreshToken.Create(
  89. member.ID,
  90. refreshToken,
  91. DateTime.UtcNow.AddDays(_jwt.RefreshTokenExpiration)
  92. );
  93. await db.RefreshToken.AddAsync(refreshTokenEntity, ct);
  94. // 로그인 횟수 증가
  95. var memberStats = await db.MemberStats.FirstOrDefaultAsync(x => x.MemberID == member.ID, ct);
  96. if (memberStats is not null)
  97. {
  98. memberStats.LoginCount++;
  99. }
  100. // 성공 로그 기록
  101. await LogLoginAttempt(member.ID, true, email, "Google", request, ct);
  102. await db.SaveChangesAsync(ct);
  103. logger.LogInformation("{Email} Google 로그인", member.Email);
  104. return Result.Success(new Response(accessToken, refreshToken, expiresAt));
  105. }
  106. private async Task LogLoginAttempt(int? memberID, bool success, string account, string? reason, Command request, CancellationToken ct)
  107. {
  108. var log = MemberLoginLog.Create(
  109. memberID,
  110. success,
  111. account,
  112. reason,
  113. ipAddress: request.IpAddress,
  114. userAgent: request.UserAgent
  115. );
  116. await db.MemberLoginLog.AddAsync(log, ct);
  117. await db.SaveChangesAsync(ct);
  118. }
  119. }