AuthController.cs 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. using Microsoft.AspNetCore.Mvc;
  2. using Microsoft.EntityFrameworkCore;
  3. using bitforum.DTOs.Request;
  4. using bitforum.DTOs.Response;
  5. using bitforum.Helpers;
  6. using bitforum.Services;
  7. using bitforum.Constants;
  8. using bitforum.Models.Log;
  9. using System.Web;
  10. namespace bitforum.Controllers.API
  11. {
  12. [ApiController]
  13. [Route("api/[controller]")]
  14. public class AuthController : ControllerBase
  15. {
  16. private readonly ILogger<AuthController> _logger;
  17. private readonly DefaultDbContext _db;
  18. private readonly IJwtAuthService _jwtAuthService;
  19. private readonly IMailService _mailService;
  20. private ResultDto _result = new ResultDto();
  21. public AuthController(ILogger<AuthController> logger, DefaultDbContext db, IJwtAuthService jwtAuthService, IMailService mailService)
  22. {
  23. _logger = logger;
  24. _db = db;
  25. _jwtAuthService = jwtAuthService;
  26. _mailService = mailService;
  27. }
  28. // 인증번호 검증
  29. [HttpPost("verification")]
  30. public async Task<ActionResult<ResultDto>> Verification([FromBody] VerificationNumberDto request)
  31. {
  32. try
  33. {
  34. // 유효성 검사
  35. if (!ModelState.IsValid)
  36. {
  37. _result.Errors = ModelState.GetErrors();
  38. throw new Exception("인증 거부");
  39. }
  40. // 회원 확인
  41. var member = await _db.Member.FirstOrDefaultAsync(c => c.Email == request.Email);
  42. if (member is null)
  43. {
  44. throw new Exception("인증 불가");
  45. }
  46. // 인증번호 확인
  47. var isValid = await _mailService.VerifyCodeAsync(request.Email, request.Code, request.Type);
  48. if (!isValid)
  49. {
  50. throw new Exception("인증 실패");
  51. }
  52. var cookie = new CookieOptions
  53. {
  54. HttpOnly = true,
  55. Secure = true,
  56. SameSite = SameSiteMode.None
  57. };
  58. switch (request.Type)
  59. {
  60. case VerificationType.Registration:
  61. cookie.Expires = DateTime.UtcNow.AddMinutes(5);
  62. break;
  63. case VerificationType.ForgotPassword:
  64. cookie.Expires = DateTime.UtcNow.AddMinutes(10);
  65. break;
  66. }
  67. Response.Cookies.Append($"isVerified-{request.Type}", "true", cookie);
  68. Console.WriteLine("Set-Cookie Header:", Response.Headers["Set-Cookie"]);
  69. _result.Message = "인증 성공";
  70. }
  71. catch (Exception e)
  72. {
  73. _logger.LogError(e, e.Message);
  74. _result.Ok = false;
  75. _result.Status = StatusCodes.Status400BadRequest;
  76. _result.Message = e.Message;
  77. }
  78. return _result;
  79. }
  80. // 인증번호 다시 보내기
  81. [HttpPost("resend-verify")]
  82. public async Task<ActionResult<ResultDto>> ResendVerify([FromBody] ResendVerifyNumberDto request)
  83. {
  84. try
  85. {
  86. // 유효성 검사
  87. if (!ModelState.IsValid)
  88. {
  89. _result.Errors = ModelState.GetErrors();
  90. throw new Exception("잘못된 접근입니다.");
  91. }
  92. // 회원 확인
  93. var member = await _db.Member.FirstOrDefaultAsync(c => c.Email == request.Email);
  94. if (member is null)
  95. {
  96. throw new Exception("회원을 조회할 수 없습니다.");
  97. }
  98. switch (request.Type)
  99. {
  100. // 회원가입 이메일 인증 메일 발송
  101. case VerificationType.Registration:
  102. await _mailService.SendRegisterEmailAsync(member);
  103. break;
  104. // 비밀번호 재설정 이메일 발송
  105. case VerificationType.ForgotPassword:
  106. await _mailService.SendForgotPasswordEmailAsync(member);
  107. break;
  108. }
  109. _result.Message = "인증 재전송 완료";
  110. }
  111. catch (Exception e)
  112. {
  113. _logger.LogError(e, e.Message);
  114. _result.Ok = false;
  115. _result.Status = StatusCodes.Status400BadRequest;
  116. _result.Message = e.Message;
  117. }
  118. return _result;
  119. }
  120. [HttpGet("verify-token")]
  121. public ActionResult<ResultDto> VerifyAccessToken()
  122. {
  123. try
  124. {
  125. if (!Request.Cookies.TryGetValue("accessToken", out var accessToken))
  126. {
  127. throw new Exception("잘못된 접근입니다.");
  128. }
  129. var isValid = _jwtAuthService.ValidateAccessToken(accessToken);
  130. if (!isValid)
  131. {
  132. throw new Exception("`AccessToken`이 유효하지 않습니다.");
  133. }
  134. var member = _jwtAuthService.GetMemberFromAccessToken(accessToken);
  135. if (member is null)
  136. {
  137. throw new Exception("회원 정보를 찾을 수 없습니다.");
  138. }
  139. _logger.LogInformation("AccessToken 유효!");
  140. _result.Message = "OK";
  141. }
  142. catch (Exception e)
  143. {
  144. _logger.LogError(e, e.Message);
  145. _result.Ok = false;
  146. _result.Status = StatusCodes.Status400BadRequest;
  147. _result.Message = e.Message;
  148. }
  149. return _result;
  150. }
  151. // 재로그인 토큰 재발급
  152. [HttpGet("refresh-token")]
  153. public async Task<ActionResult<ResultDto>> RefreshToken()
  154. {
  155. try
  156. {
  157. if (!Request.Cookies.TryGetValue("refreshToken", out var refreshToken))
  158. {
  159. throw new Exception("잘못된 접근입니다.");
  160. }
  161. var isValid = await _jwtAuthService.ValidateRefreshToken(refreshToken);
  162. if (!isValid)
  163. {
  164. throw new Exception("확인할 수 없는 요청입니다.");
  165. }
  166. var member = await _jwtAuthService.GetMemberFromRefreshToken(refreshToken);
  167. if (member is null)
  168. {
  169. throw new Exception("회원 정보를 찾을 수 없습니다.");
  170. }
  171. // JWT 토큰 발급
  172. await _jwtAuthService.SetRefreshTokenAndCookieAsync(member);
  173. _logger.LogInformation("RefreshToken 유효!");
  174. _result.Message = "OK";
  175. }
  176. catch (Exception e)
  177. {
  178. _logger.LogError(e, e.Message);
  179. _result.Ok = false;
  180. _result.Status = StatusCodes.Status400BadRequest;
  181. _result.Message = e.Message;
  182. }
  183. return _result;
  184. }
  185. [HttpGet("verify-email/{token}")]
  186. public async Task<ActionResult<ResultDto>> VerifyEmail([FromRoute] string token)
  187. {
  188. try
  189. {
  190. if (string.IsNullOrEmpty(token))
  191. {
  192. _result.Errors = ModelState.GetErrors();
  193. throw new Exception("유효성 검사에 실패했습니다.");
  194. }
  195. token = HttpUtility.UrlDecode(token);
  196. // 인증주소 확인
  197. var tokenData = await _mailService.VerifyTokenAsync(token, VerificationType.ChangedEmail);
  198. if (tokenData is null)
  199. {
  200. throw new Exception("인증 실패");
  201. }
  202. if (string.IsNullOrEmpty(tokenData?.AdditionalData?.Email))
  203. {
  204. throw new Exception("최소 필수 정보가 누락되었습니다.");
  205. }
  206. // 회원 조회
  207. var member = await _db.Member.FirstOrDefaultAsync(c => c.Email == tokenData.AdditionalData.Email);
  208. if (member is null)
  209. {
  210. throw new Exception("회원 정보를 찾을 수 없습니다.");
  211. }
  212. await _db.EmailChangeLog.AddAsync(new EmailChangeLog
  213. {
  214. MemberID = member.ID,
  215. BeforeEmail = member.Email,
  216. AfterEmail = tokenData.Email,
  217. CreatedAt = DateTime.UtcNow
  218. });
  219. member.Email = tokenData.Email;
  220. member.LastEmailChangedAt = DateTime.UtcNow;
  221. await _db.SaveChangesAsync();
  222. await _mailService.SendChangedEmailAsync(member); // 이메일 변경 완료 메일 발송
  223. _result.Message = "이메일이 변경되었습니다.";
  224. }
  225. catch (Exception e)
  226. {
  227. _logger.LogError(e, e.Message);
  228. _result.Ok = false;
  229. _result.Status = StatusCodes.Status400BadRequest;
  230. _result.Message = e.Message;
  231. }
  232. return _result;
  233. }
  234. }
  235. }