InfluencerModel.php 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. <?php
  2. namespace App\Models;
  3. use CodeIgniter\Model;
  4. class InfluencerModel 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. 'NAME',
  16. 'EMAIL',
  17. 'PHONE',
  18. 'MEMBER_TYPE',
  19. 'STATUS',
  20. 'PROFILE_IMAGE',
  21. 'LAST_LOGIN_DATE',
  22. 'IS_ACT',
  23. 'REG_DATE',
  24. 'MOD_DATE',
  25. // 인플루언서 전용 필드들
  26. 'INFLUENCER_TYPE',
  27. 'PRIMARY_CATEGORY',
  28. 'SECONDARY_CATEGORY',
  29. 'FOLLOWER_COUNT',
  30. 'ENGAGEMENT_RATE',
  31. 'AVERAGE_VIEWS',
  32. 'SNS_CHANNELS',
  33. 'REGION',
  34. 'DESCRIPTION',
  35. 'PORTFOLIO_URL',
  36. 'BANK_NAME',
  37. 'ACCOUNT_NUMBER',
  38. 'ACCOUNT_HOLDER',
  39. 'TAX_INFO',
  40. 'RATING',
  41. 'VERIFICATION_STATUS',
  42. 'PREFERRED_CATEGORIES',
  43. 'MIN_COMMISSION_RATE',
  44. 'AVAILABLE_REGIONS'
  45. ];
  46. protected $useTimestamps = true;
  47. protected $createdField = 'REG_DATE';
  48. protected $updatedField = 'MOD_DATE';
  49. protected $dateFormat = 'datetime';
  50. protected $validationRules = [
  51. 'ID' => 'required|min_length[4]|max_length[50]|is_unique[USER_LIST.ID,SEQ,{SEQ}]',
  52. 'PASSWORD' => 'required|min_length[8]',
  53. 'NICK_NAME' => 'required|min_length[2]|max_length[50]',
  54. 'EMAIL' => 'required|valid_email|is_unique[USER_LIST.EMAIL,SEQ,{SEQ}]',
  55. 'PHONE' => 'permit_empty|min_length[10]|max_length[15]',
  56. 'MEMBER_TYPE' => 'required|in_list[INFLUENCER]',
  57. 'STATUS' => 'required|in_list[ACTIVE,INACTIVE,SUSPENDED,PENDING]',
  58. 'INFLUENCER_TYPE' => 'permit_empty|in_list[MICRO,MACRO,MEGA,NANO]',
  59. 'FOLLOWER_COUNT' => 'permit_empty|integer|greater_than_equal_to[0]',
  60. 'ENGAGEMENT_RATE' => 'permit_empty|decimal|greater_than_equal_to[0]|less_than_equal_to[100]',
  61. 'IS_ACT' => 'required|in_list[Y,N]'
  62. ];
  63. protected $validationMessages = [
  64. 'ID' => [
  65. 'required' => '아이디는 필수입니다.',
  66. 'min_length' => '아이디는 최소 4자 이상이어야 합니다.',
  67. 'is_unique' => '이미 사용 중인 아이디입니다.'
  68. ],
  69. 'EMAIL' => [
  70. 'required' => '이메일은 필수입니다.',
  71. 'valid_email' => '올바른 이메일 형식이 아닙니다.',
  72. 'is_unique' => '이미 사용 중인 이메일입니다.'
  73. ],
  74. 'NICK_NAME' => [
  75. 'required' => '닉네임은 필수입니다.',
  76. 'min_length' => '닉네임은 최소 2자 이상이어야 합니다.'
  77. ]
  78. ];
  79. protected $beforeInsert = ['hashPassword', 'setInfluencerDefaults'];
  80. protected $beforeUpdate = ['hashPassword'];
  81. /**
  82. * 패스워드 해시화
  83. */
  84. protected function hashPassword(array $data)
  85. {
  86. if (isset($data['data']['PASSWORD'])) {
  87. $data['data']['PASSWORD'] = password_hash($data['data']['PASSWORD'], PASSWORD_DEFAULT);
  88. }
  89. return $data;
  90. }
  91. /**
  92. * 인플루언서 기본값 설정
  93. */
  94. protected function setInfluencerDefaults(array $data)
  95. {
  96. if (!isset($data['data']['MEMBER_TYPE'])) {
  97. $data['data']['MEMBER_TYPE'] = 'INFLUENCER';
  98. }
  99. if (!isset($data['data']['STATUS'])) {
  100. $data['data']['STATUS'] = 'PENDING';
  101. }
  102. if (!isset($data['data']['IS_ACT'])) {
  103. $data['data']['IS_ACT'] = 'Y';
  104. }
  105. if (!isset($data['data']['VERIFICATION_STATUS'])) {
  106. $data['data']['VERIFICATION_STATUS'] = 'PENDING';
  107. }
  108. return $data;
  109. }
  110. /**
  111. * 인플루언서 목록 조회 (필터링)
  112. */
  113. public function getInfluencers($filters = [])
  114. {
  115. $builder = $this->builder();
  116. $builder->where('MEMBER_TYPE', 'INFLUENCER');
  117. $builder->where('IS_ACT', 'Y');
  118. // 상태 필터
  119. if (isset($filters['status'])) {
  120. $builder->where('STATUS', $filters['status']);
  121. }
  122. // 카테고리 필터
  123. if (isset($filters['category'])) {
  124. $builder->where('PRIMARY_CATEGORY', $filters['category']);
  125. }
  126. // 지역 필터
  127. if (isset($filters['region'])) {
  128. $builder->where('REGION', $filters['region']);
  129. }
  130. // 팔로워 수 범위
  131. if (isset($filters['min_followers'])) {
  132. $builder->where('FOLLOWER_COUNT >=', $filters['min_followers']);
  133. }
  134. if (isset($filters['max_followers'])) {
  135. $builder->where('FOLLOWER_COUNT <=', $filters['max_followers']);
  136. }
  137. // 인플루언서 타입
  138. if (isset($filters['influencer_type'])) {
  139. $builder->where('INFLUENCER_TYPE', $filters['influencer_type']);
  140. }
  141. // 검증 상태
  142. if (isset($filters['verification_status'])) {
  143. $builder->where('VERIFICATION_STATUS', $filters['verification_status']);
  144. }
  145. // 키워드 검색
  146. if (isset($filters['keyword'])) {
  147. $builder->groupStart()
  148. ->like('NICK_NAME', $filters['keyword'])
  149. ->orLike('NAME', $filters['keyword'])
  150. ->orLike('DESCRIPTION', $filters['keyword'])
  151. ->groupEnd();
  152. }
  153. // 정렬
  154. $sortBy = $filters['sort_by'] ?? 'REG_DATE';
  155. $sortOrder = $filters['sort_order'] ?? 'DESC';
  156. $builder->orderBy($sortBy, $sortOrder);
  157. return $builder;
  158. }
  159. /**
  160. * 인플루언서 프로필 조회
  161. */
  162. public function getProfile($influencerSeq)
  163. {
  164. return $this->select('
  165. SEQ, ID, NICK_NAME, NAME, EMAIL, PHONE, PROFILE_IMAGE,
  166. INFLUENCER_TYPE, PRIMARY_CATEGORY, SECONDARY_CATEGORY,
  167. FOLLOWER_COUNT, ENGAGEMENT_RATE, AVERAGE_VIEWS,
  168. SNS_CHANNELS, REGION, DESCRIPTION, PORTFOLIO_URL,
  169. RATING, VERIFICATION_STATUS, PREFERRED_CATEGORIES,
  170. MIN_COMMISSION_RATE, AVAILABLE_REGIONS,
  171. REG_DATE, MOD_DATE, LAST_LOGIN_DATE
  172. ')
  173. ->where('SEQ', $influencerSeq)
  174. ->where('MEMBER_TYPE', 'INFLUENCER')
  175. ->where('IS_ACT', 'Y')
  176. ->first();
  177. }
  178. /**
  179. * 인플루언서 통계 조회
  180. */
  181. public function getStats($influencerSeq)
  182. {
  183. // 파트너십 통계는 별도 모델에서 처리
  184. $partnershipModel = new \App\Models\InfluencerPartnershipModel();
  185. return $partnershipModel->getInfluencerStats($influencerSeq);
  186. }
  187. /**
  188. * 인플루언서 검증 상태 업데이트
  189. */
  190. public function updateVerificationStatus($influencerSeq, $status, $reason = '')
  191. {
  192. $data = [
  193. 'VERIFICATION_STATUS' => $status,
  194. 'MOD_DATE' => date('Y-m-d H:i:s')
  195. ];
  196. if (!empty($reason)) {
  197. $data['VERIFICATION_REASON'] = $reason;
  198. }
  199. return $this->update($influencerSeq, $data);
  200. }
  201. /**
  202. * 카테고리별 인플루언서 수 조회
  203. */
  204. public function getCountByCategory()
  205. {
  206. return $this->select('PRIMARY_CATEGORY, COUNT(*) as count')
  207. ->where('MEMBER_TYPE', 'INFLUENCER')
  208. ->where('IS_ACT', 'Y')
  209. ->where('STATUS', 'ACTIVE')
  210. ->groupBy('PRIMARY_CATEGORY')
  211. ->findAll();
  212. }
  213. /**
  214. * 인플루언서 타입별 통계
  215. */
  216. public function getCountByType()
  217. {
  218. return $this->select('INFLUENCER_TYPE, COUNT(*) as count')
  219. ->where('MEMBER_TYPE', 'INFLUENCER')
  220. ->where('IS_ACT', 'Y')
  221. ->where('STATUS', 'ACTIVE')
  222. ->groupBy('INFLUENCER_TYPE')
  223. ->findAll();
  224. }
  225. /**
  226. * 로그인 검증
  227. */
  228. public function verifyLogin($id, $password)
  229. {
  230. $user = $this->where('ID', $id)
  231. ->where('MEMBER_TYPE', 'INFLUENCER')
  232. ->where('IS_ACT', 'Y')
  233. ->first();
  234. if ($user && password_verify($password, $user['PASSWORD'])) {
  235. // 마지막 로그인 시간 업데이트
  236. $this->update($user['SEQ'], [
  237. 'LAST_LOGIN_DATE' => date('Y-m-d H:i:s')
  238. ]);
  239. unset($user['PASSWORD']); // 패스워드 제거 후 반환
  240. return $user;
  241. }
  242. return false;
  243. }
  244. /**
  245. * 인플루언서 랭킹 조회
  246. */
  247. public function getTopInfluencers($limit = 10, $category = null)
  248. {
  249. $builder = $this->select('
  250. SEQ, NICK_NAME, PROFILE_IMAGE, FOLLOWER_COUNT,
  251. ENGAGEMENT_RATE, RATING, PRIMARY_CATEGORY
  252. ');
  253. $builder->where('MEMBER_TYPE', 'INFLUENCER');
  254. $builder->where('IS_ACT', 'Y');
  255. $builder->where('STATUS', 'ACTIVE');
  256. $builder->where('VERIFICATION_STATUS', 'VERIFIED');
  257. if ($category) {
  258. $builder->where('PRIMARY_CATEGORY', $category);
  259. }
  260. $builder->orderBy('RATING', 'DESC');
  261. $builder->orderBy('FOLLOWER_COUNT', 'DESC');
  262. $builder->limit($limit);
  263. return $builder->get()->getResultArray();
  264. }
  265. }