Member.cs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. using System.ComponentModel.DataAnnotations;
  2. using System.ComponentModel.DataAnnotations.Schema;
  3. using System.Security.Cryptography;
  4. using Domain.Entities.Members.ValueObject;
  5. using Domain.Entities.Wallets;
  6. using NanoidDotNet;
  7. /// <summary>
  8. /// 사용자단 일반 회원 정보
  9. /// </summary>
  10. namespace Domain.Entities.Members
  11. {
  12. public class Member
  13. {
  14. [Key]
  15. public int ID { get; private set; }
  16. // 회원 등급 정보
  17. [ForeignKey(nameof(MemberGradeID))]
  18. public virtual MemberGrade? MemberGrade { get; private set; }
  19. // 알림/약관 동의 여부
  20. public virtual MemberApprove MemberApprove { get; private set; } = null!;
  21. // 회원 집계 정보
  22. public virtual MemberStats MemberStats { get; private set; } = null!;
  23. // 지갑
  24. public virtual Wallet? Wallet { get; private set; }
  25. // 채널
  26. public virtual Channel? Channel { get; private set; }
  27. public int? MemberGradeID { get; private set; }
  28. public string SID { get; private set; } = Nanoid.Generate(size: 8); // 무작위 문자열
  29. public string Email { get; private set; } = default!;
  30. public string? Name { get; private set; }
  31. public string? FullName { get; private set; }
  32. public string? FirstName { get; private set; }
  33. public string? LastName { get; private set; }
  34. public string? Password { get; private set; } = string.Empty; // 관리자 수정 시 필요
  35. public string? PasswordHash { get; private set; }
  36. public string? Intro { get; private set; }
  37. public string? Summary { get; private set; }
  38. public string? Phone { get; private set; }
  39. public DateOnly? Birthday { get; private set; }
  40. public Gender? Gender { get; private set; }
  41. public string? Thumb { get; private set; }
  42. public string? Icon { get; private set; }
  43. public bool IsEmailVerified { get; private set; } = false;
  44. public bool IsAuthCertified { get; private set; } = false;
  45. public bool IsDenied { get; private set; } = false;
  46. public bool IsAdmin { get; private set; } = false;
  47. public bool IsWithdraw { get; private set; } = false;
  48. public bool IsCreator { get; private set; } = false;
  49. public string? DeviceInfo { get; private set; }
  50. public string? SignupIP { get; private set; }
  51. public string? LastLoginIp { get; private set; }
  52. public string? IpAddress { get; private set; }
  53. public string? UserAgent { get; private set; }
  54. public DateTime? LastLoginAt { get; private set; }
  55. public DateTime? LastEmailChangedAt { get; private set; }
  56. public DateTime? LastNameChangedAt { get; private set; }
  57. public DateTime? LastSummaryChangedAt { get; private set; }
  58. public DateTime? LastIntroChangedAt { get; private set; }
  59. public DateTime? EmailVerifiedAt { get; private set; }
  60. public DateTime? AuthCertifiedAt { get; private set; }
  61. public DateTime PasswordUpdatedAt { get; private set; }
  62. public DateTime CreatedAt { get; private set; } = DateTime.UtcNow;
  63. public DateTime? UpdatedAt { get; private set; }
  64. public DateTime? DeletedAt { get; private set; }
  65. public DateTime? DeniedAt { get; private set; }
  66. private Member() { }
  67. private Member(string email)
  68. {
  69. if (string.IsNullOrWhiteSpace(email))
  70. {
  71. throw new ArgumentException("Email is required.", nameof(email));
  72. }
  73. if (email.Length > 60)
  74. {
  75. throw new ArgumentOutOfRangeException(nameof(email));
  76. }
  77. Email = email;
  78. PasswordUpdatedAt = DateTime.UtcNow;
  79. }
  80. public static Member Create(string email)
  81. {
  82. return new(email);
  83. }
  84. public static Member Create(string email, string password)
  85. {
  86. var member = new Member(email);
  87. member.SetPassword(password);
  88. return member;
  89. }
  90. public void SetPassword(string password)
  91. {
  92. PasswordHash = HashPassword(password);
  93. PasswordUpdatedAt = DateTime.UtcNow;
  94. }
  95. public bool VerifyPassword(string password)
  96. {
  97. if (string.IsNullOrEmpty(PasswordHash))
  98. {
  99. return false;
  100. }
  101. return VerifyHashedPassword(PasswordHash, password);
  102. }
  103. public void SetEmail(string email)
  104. {
  105. Email = email;
  106. IsEmailVerified = false;
  107. LastEmailChangedAt = DateTime.UtcNow;
  108. UpdatedAt = DateTime.UtcNow;
  109. }
  110. public void SetName(string? name)
  111. {
  112. Name = name;
  113. LastNameChangedAt = DateTime.UtcNow;
  114. UpdatedAt = DateTime.UtcNow;
  115. }
  116. public void SetSummary(string? summary)
  117. {
  118. Summary = summary;
  119. LastSummaryChangedAt = DateTime.UtcNow;
  120. UpdatedAt = DateTime.UtcNow;
  121. }
  122. public void SetIntro(string? intro)
  123. {
  124. Intro = intro;
  125. LastIntroChangedAt = DateTime.UtcNow;
  126. UpdatedAt = DateTime.UtcNow;
  127. }
  128. public void SetThumb(string? thumb)
  129. {
  130. Thumb = thumb;
  131. UpdatedAt = DateTime.UtcNow;
  132. }
  133. public void MarkEmailVerified()
  134. {
  135. IsEmailVerified = true;
  136. EmailVerifiedAt = DateTime.UtcNow;
  137. UpdatedAt = DateTime.UtcNow;
  138. }
  139. public void Withdraw()
  140. {
  141. IsWithdraw = true;
  142. DeletedAt = DateTime.UtcNow;
  143. UpdatedAt = DateTime.UtcNow;
  144. }
  145. // PBKDF2 (RFC 2898) - 순수 C# 비밀번호 해싱
  146. private static string HashPassword(string password)
  147. {
  148. var salt = RandomNumberGenerator.GetBytes(16);
  149. var hash = Rfc2898DeriveBytes.Pbkdf2(
  150. password,
  151. salt,
  152. iterations: 100_000,
  153. HashAlgorithmName.SHA256,
  154. outputLength: 32
  155. );
  156. // salt + hash → Base64
  157. var combined = new byte[salt.Length + hash.Length];
  158. Buffer.BlockCopy(salt, 0, combined, 0, salt.Length);
  159. Buffer.BlockCopy(hash, 0, combined, salt.Length, hash.Length);
  160. return Convert.ToBase64String(combined);
  161. }
  162. // 비밀번호 검증
  163. private static bool VerifyHashedPassword(string hashedPassword, string password)
  164. {
  165. var combined = Convert.FromBase64String(hashedPassword);
  166. if (combined.Length != 48) // 16(salt) + 32(hash)
  167. {
  168. return false;
  169. }
  170. var salt = combined[..16];
  171. var storedHash = combined[16..];
  172. var hash = Rfc2898DeriveBytes.Pbkdf2(
  173. password,
  174. salt,
  175. iterations: 100_000,
  176. HashAlgorithmName.SHA256,
  177. outputLength: 32
  178. );
  179. return CryptographicOperations.FixedTimeEquals(storedHash, hash);
  180. }
  181. }
  182. }