UserModel.php 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. <?php
  2. namespace App\Models;
  3. use CodeIgniter\Model;
  4. class UserModel extends Model
  5. {
  6. protected $table = 'USER_LIST';
  7. protected $primaryKey = 'SEQ';
  8. protected $useAutoIncrement = true;
  9. protected $returnType = 'array';
  10. protected $useSoftDeletes = false;
  11. protected $allowedFields = [
  12. 'ID',
  13. 'PASSWORD',
  14. 'NICK_NAME',
  15. 'EMAIL',
  16. 'PHONE',
  17. 'MEMBER_TYPE',
  18. 'STATUS',
  19. 'LAST_LOGIN_DATE',
  20. 'IS_ACT',
  21. 'REG_DATE',
  22. 'MOD_DATE',
  23. // 인플루언서 관련 필드들
  24. 'INFLUENCER_TYPE',
  25. 'PRIMARY_CATEGORY',
  26. 'FOLLOWER_COUNT',
  27. 'AVG_VIEWS',
  28. 'PROFILE_IMAGE',
  29. 'BIO',
  30. 'INSTAGRAM_URL',
  31. 'YOUTUBE_URL',
  32. 'TIKTOK_URL',
  33. 'BLOG_URL',
  34. 'PREFERRED_REGION',
  35. 'MIN_COMMISSION_RATE',
  36. 'VERIFICATION_STATUS',
  37. 'VERIFIED_DATE'
  38. ];
  39. protected $useTimestamps = true;
  40. protected $createdField = 'REG_DATE';
  41. protected $updatedField = 'MOD_DATE';
  42. protected $validationRules = [
  43. 'ID' => 'required|max_length[50]|is_unique[USER_LIST.ID,SEQ,{SEQ}]',
  44. 'PASSWORD' => 'required|min_length[8]',
  45. 'NICK_NAME' => 'required|max_length[100]',
  46. 'EMAIL' => 'required|valid_email|is_unique[USER_LIST.EMAIL,SEQ,{SEQ}]',
  47. 'PHONE' => 'permit_empty|max_length[20]',
  48. 'MEMBER_TYPE' => 'required|in_list[ADMIN,INFLUENCER,VENDOR]',
  49. 'STATUS' => 'required|in_list[ACTIVE,INACTIVE,SUSPENDED,PENDING]',
  50. 'IS_ACT' => 'required|in_list[Y,N]',
  51. 'INFLUENCER_TYPE' => 'permit_empty|in_list[MACRO,MICRO,NANO,MEGA]',
  52. 'PRIMARY_CATEGORY' => 'permit_empty|in_list[FASHION_BEAUTY,FOOD_HEALTH,LIFESTYLE,TECH_ELECTRONICS,SPORTS_LEISURE,CULTURE_ENTERTAINMENT]',
  53. 'FOLLOWER_COUNT' => 'permit_empty|integer|greater_than_equal_to[0]',
  54. 'AVG_VIEWS' => 'permit_empty|integer|greater_than_equal_to[0]',
  55. 'PREFERRED_REGION' => 'permit_empty|in_list[SEOUL,GYEONGGI,INCHEON,BUSAN,DAEGU,DAEJEON,GWANGJU,ULSAN,OTHER]',
  56. 'MIN_COMMISSION_RATE' => 'permit_empty|decimal|greater_than_equal_to[0]|less_than_equal_to[100]',
  57. 'VERIFICATION_STATUS' => 'permit_empty|in_list[UNVERIFIED,PENDING,VERIFIED,REJECTED]'
  58. ];
  59. protected $validationMessages = [
  60. 'ID' => [
  61. 'required' => '아이디는 필수입니다.',
  62. 'max_length' => '아이디는 50자를 초과할 수 없습니다.',
  63. 'is_unique' => '이미 사용 중인 아이디입니다.'
  64. ],
  65. 'PASSWORD' => [
  66. 'required' => '비밀번호는 필수입니다.',
  67. 'min_length' => '비밀번호는 최소 8자 이상이어야 합니다.'
  68. ],
  69. 'NICK_NAME' => [
  70. 'required' => '닉네임은 필수입니다.',
  71. 'max_length' => '닉네임은 100자를 초과할 수 없습니다.'
  72. ],
  73. 'EMAIL' => [
  74. 'required' => '이메일은 필수입니다.',
  75. 'valid_email' => '유효한 이메일 형식이 아닙니다.',
  76. 'is_unique' => '이미 사용 중인 이메일입니다.'
  77. ],
  78. 'MEMBER_TYPE' => [
  79. 'required' => '회원 유형은 필수입니다.',
  80. 'in_list' => '유효하지 않은 회원 유형입니다.'
  81. ],
  82. 'STATUS' => [
  83. 'required' => '상태는 필수입니다.',
  84. 'in_list' => '유효하지 않은 상태입니다.'
  85. ],
  86. 'IS_ACT' => [
  87. 'required' => '활성 상태는 필수입니다.',
  88. 'in_list' => '활성 상태는 Y 또는 N이어야 합니다.'
  89. ]
  90. ];
  91. protected $skipValidation = false;
  92. protected $cleanValidationRules = true;
  93. /**
  94. * 인플루언서 목록 조회
  95. */
  96. public function getInfluencers($filters = [], $page = 1, $perPage = 12)
  97. {
  98. $builder = $this->where('MEMBER_TYPE', 'INFLUENCER')
  99. ->where('IS_ACT', 'Y')
  100. ->where('STATUS', 'ACTIVE');
  101. // 키워드 검색
  102. if (!empty($filters['keyword'])) {
  103. $builder->groupStart()
  104. ->like('NICK_NAME', $filters['keyword'])
  105. ->orLike('ID', $filters['keyword'])
  106. ->groupEnd();
  107. }
  108. // 카테고리 필터
  109. if (!empty($filters['category'])) {
  110. $builder->where('PRIMARY_CATEGORY', $filters['category']);
  111. }
  112. // 인플루언서 타입 필터
  113. if (!empty($filters['influencer_type'])) {
  114. $builder->where('INFLUENCER_TYPE', $filters['influencer_type']);
  115. }
  116. // 팔로워 수 범위
  117. if (!empty($filters['follower_min'])) {
  118. $builder->where('FOLLOWER_COUNT >=', $filters['follower_min']);
  119. }
  120. if (!empty($filters['follower_max'])) {
  121. $builder->where('FOLLOWER_COUNT <=', $filters['follower_max']);
  122. }
  123. // 페이징
  124. $offset = ($page - 1) * $perPage;
  125. return $builder->limit($perPage, $offset)->findAll();
  126. }
  127. /**
  128. * 인플루언서 검색 결과 총 개수
  129. */
  130. public function countInfluencers($filters = [])
  131. {
  132. $builder = $this->where('MEMBER_TYPE', 'INFLUENCER')
  133. ->where('IS_ACT', 'Y')
  134. ->where('STATUS', 'ACTIVE');
  135. // 키워드 검색
  136. if (!empty($filters['keyword'])) {
  137. $builder->groupStart()
  138. ->like('NICK_NAME', $filters['keyword'])
  139. ->orLike('ID', $filters['keyword'])
  140. ->groupEnd();
  141. }
  142. // 카테고리 필터
  143. if (!empty($filters['category'])) {
  144. $builder->where('PRIMARY_CATEGORY', $filters['category']);
  145. }
  146. // 인플루언서 타입 필터
  147. if (!empty($filters['influencer_type'])) {
  148. $builder->where('INFLUENCER_TYPE', $filters['influencer_type']);
  149. }
  150. // 팔로워 수 범위
  151. if (!empty($filters['follower_min'])) {
  152. $builder->where('FOLLOWER_COUNT >=', $filters['follower_min']);
  153. }
  154. if (!empty($filters['follower_max'])) {
  155. $builder->where('FOLLOWER_COUNT <=', $filters['follower_max']);
  156. }
  157. return $builder->countAllResults();
  158. }
  159. /**
  160. * 인플루언서 타입별 통계
  161. */
  162. public function getInfluencerTypeStats()
  163. {
  164. return $this->select('INFLUENCER_TYPE, COUNT(*) as count')
  165. ->where('MEMBER_TYPE', 'INFLUENCER')
  166. ->where('IS_ACT', 'Y')
  167. ->where('STATUS', 'ACTIVE')
  168. ->groupBy('INFLUENCER_TYPE')
  169. ->findAll();
  170. }
  171. /**
  172. * 카테고리별 인플루언서 통계
  173. */
  174. public function getInfluencerCategoryStats()
  175. {
  176. return $this->select('PRIMARY_CATEGORY, COUNT(*) as count')
  177. ->where('MEMBER_TYPE', 'INFLUENCER')
  178. ->where('IS_ACT', 'Y')
  179. ->where('STATUS', 'ACTIVE')
  180. ->groupBy('PRIMARY_CATEGORY')
  181. ->findAll();
  182. }
  183. /**
  184. * 사용자 로그인
  185. */
  186. public function authenticate($id, $password)
  187. {
  188. $user = $this->where('ID', $id)
  189. ->where('IS_ACT', 'Y')
  190. ->first();
  191. if ($user && password_verify($password, $user['PASSWORD'])) {
  192. // 로그인 날짜 업데이트
  193. $this->update($user['SEQ'], ['LAST_LOGIN_DATE' => date('Y-m-d H:i:s')]);
  194. return $user;
  195. }
  196. return false;
  197. }
  198. }