InfluencerController.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. <?php
  2. namespace App\Controllers;
  3. use CodeIgniter\RESTful\ResourceController;
  4. use App\Models\VendorInfluencerMappingModel;
  5. use App\Models\VendorInfluencerStatusHistoryModel;
  6. use App\Models\InfluencerPartnershipModel;
  7. use App\Models\VendorModel;
  8. use App\Models\InfluencerModel;
  9. class InfluencerController extends ResourceController
  10. {
  11. protected $modelName = 'App\Models\VendorInfluencerMappingModel';
  12. protected $format = 'json';
  13. protected $vendorInfluencerModel;
  14. protected $influencerPartnershipModel;
  15. protected $statusHistoryModel;
  16. protected $vendorModel;
  17. protected $influencerModel;
  18. public function __construct()
  19. {
  20. $this->vendorInfluencerModel = new VendorInfluencerMappingModel();
  21. $this->influencerPartnershipModel = new InfluencerPartnershipModel();
  22. $this->statusHistoryModel = new VendorInfluencerStatusHistoryModel();
  23. $this->vendorModel = new VendorModel();
  24. $this->influencerModel = new InfluencerModel();
  25. }
  26. /**
  27. * 벤더사 검색 (상태 정보 포함)
  28. */
  29. public function searchVendors()
  30. {
  31. try {
  32. $request = $this->request->getJSON();
  33. $influencerSeq = $request->influencerSeq ?? null;
  34. $keyword = $request->keyword ?? '';
  35. $category = $request->category ?? '';
  36. $region = $request->region ?? '';
  37. $sortBy = $request->sortBy ?? 'latest';
  38. $page = (int)($request->page ?? 1);
  39. $size = (int)($request->size ?? 12);
  40. if (!$influencerSeq) {
  41. return $this->response->setStatusCode(400)->setJSON([
  42. 'success' => false,
  43. 'message' => '인플루언서 SEQ는 필수입니다.'
  44. ]);
  45. }
  46. // 필터 배열 구성 (VendorModel에 맞는 형식)
  47. $filters = [
  48. 'keyword' => $keyword,
  49. 'category' => $category,
  50. 'region' => $region,
  51. 'sortBy' => $sortBy
  52. ];
  53. // 벤더사 목록 조회
  54. $vendors = $this->vendorModel->searchVendors($filters, $page, $size);
  55. $totalCount = $this->vendorModel->countSearchResults($filters);
  56. // 각 벤더사와의 파트너십 상태 확인
  57. foreach ($vendors as &$vendor) {
  58. $partnership = $this->vendorInfluencerModel
  59. ->select('VENDOR_INFLUENCER_MAPPING.SEQ, VENDOR_INFLUENCER_MAPPING.REQUEST_TYPE,
  60. VENDOR_INFLUENCER_STATUS_HISTORY.STATUS as CURRENT_STATUS,
  61. VENDOR_INFLUENCER_STATUS_HISTORY.STATUS_MESSAGE,
  62. VENDOR_INFLUENCER_STATUS_HISTORY.CHANGED_DATE')
  63. ->join('VENDOR_INFLUENCER_STATUS_HISTORY',
  64. 'VENDOR_INFLUENCER_STATUS_HISTORY.MAPPING_SEQ = VENDOR_INFLUENCER_MAPPING.SEQ AND VENDOR_INFLUENCER_STATUS_HISTORY.IS_CURRENT = "Y"', 'left')
  65. ->where('VENDOR_SEQ', $vendor['SEQ'])
  66. ->where('INFLUENCER_SEQ', $influencerSeq)
  67. ->where('VENDOR_INFLUENCER_MAPPING.IS_ACT', 'Y')
  68. ->orderBy('VENDOR_INFLUENCER_MAPPING.REG_DATE', 'DESC')
  69. ->first();
  70. if ($partnership) {
  71. $vendor['PARTNERSHIP_STATUS'] = $partnership['CURRENT_STATUS'];
  72. $vendor['PARTNERSHIP_SEQ'] = $partnership['SEQ'];
  73. $vendor['REQUEST_TYPE'] = $partnership['REQUEST_TYPE'];
  74. $vendor['STATUS_MESSAGE'] = $partnership['STATUS_MESSAGE'];
  75. $vendor['STATUS_DATE'] = $partnership['CHANGED_DATE'];
  76. } else {
  77. $vendor['PARTNERSHIP_STATUS'] = null;
  78. $vendor['PARTNERSHIP_SEQ'] = null;
  79. $vendor['REQUEST_TYPE'] = null;
  80. $vendor['STATUS_MESSAGE'] = null;
  81. $vendor['STATUS_DATE'] = null;
  82. }
  83. }
  84. // 페이지네이션 정보 계산
  85. $totalPages = ceil($totalCount / $size);
  86. return $this->response->setJSON([
  87. 'success' => true,
  88. 'data' => [
  89. 'items' => $vendors,
  90. 'pagination' => [
  91. 'currentPage' => $page,
  92. 'totalPages' => $totalPages,
  93. 'totalCount' => $totalCount,
  94. 'pageSize' => $size
  95. ]
  96. ]
  97. ]);
  98. } catch (\Exception $e) {
  99. log_message('error', '벤더사 검색 오류: ' . $e->getMessage());
  100. log_message('error', '스택 트레이스: ' . $e->getTraceAsString());
  101. return $this->response->setStatusCode(500)->setJSON([
  102. 'success' => false,
  103. 'message' => '벤더사 검색 중 오류가 발생했습니다.',
  104. 'error' => $e->getMessage()
  105. ]);
  106. }
  107. }
  108. /**
  109. * 승인 요청 생성
  110. */
  111. public function createApprovalRequest()
  112. {
  113. try {
  114. $request = $this->request->getJSON();
  115. $vendorSeq = $request->vendorSeq ?? null;
  116. $influencerSeq = $request->influencerSeq ?? null;
  117. $requestMessage = $request->requestMessage ?? '';
  118. $requestedBy = $request->requestedBy ?? null;
  119. $commissionRate = $request->commissionRate ?? null;
  120. $specialConditions = $request->specialConditions ?? '';
  121. if (!$vendorSeq || !$influencerSeq || !$requestedBy) {
  122. return $this->response->setStatusCode(400)->setJSON([
  123. 'success' => false,
  124. 'message' => '필수 파라미터가 누락되었습니다.'
  125. ]);
  126. }
  127. // 데이터 구성
  128. $data = [
  129. 'VENDOR_SEQ' => $vendorSeq,
  130. 'INFLUENCER_SEQ' => $influencerSeq,
  131. 'REQUEST_MESSAGE' => $requestMessage,
  132. 'REQUESTED_BY' => $requestedBy,
  133. 'COMMISSION_RATE' => $commissionRate,
  134. 'SPECIAL_CONDITIONS' => $specialConditions
  135. ];
  136. // InfluencerPartnershipModel을 통해 요청 생성
  137. $mappingSeq = $this->influencerPartnershipModel->createApprovalRequest($data);
  138. return $this->response->setStatusCode(201)->setJSON([
  139. 'success' => true,
  140. 'message' => '승인 요청이 성공적으로 생성되었습니다.',
  141. 'data' => [
  142. 'mappingSeq' => $mappingSeq,
  143. 'status' => 'PENDING'
  144. ]
  145. ]);
  146. } catch (\Exception $e) {
  147. log_message('error', '승인 요청 생성 오류: ' . $e->getMessage());
  148. log_message('error', '스택 트레이스: ' . $e->getTraceAsString());
  149. return $this->response->setStatusCode(500)->setJSON([
  150. 'success' => false,
  151. 'message' => '승인 요청 생성에 실패했습니다.',
  152. 'error' => $e->getMessage()
  153. ]);
  154. }
  155. }
  156. /**
  157. * 재승인 요청 생성 (히스토리 테이블 기반)
  158. */
  159. public function createReapplyRequest()
  160. {
  161. try {
  162. $request = $this->request->getJSON();
  163. $vendorSeq = $request->vendorSeq ?? null;
  164. $influencerSeq = $request->influencerSeq ?? null;
  165. $requestMessage = $request->requestMessage ?? '';
  166. $requestedBy = $request->requestedBy ?? null;
  167. $commissionRate = $request->commissionRate ?? null;
  168. $specialConditions = $request->specialConditions ?? '';
  169. log_message('debug', '재승인 요청 파라미터: ' . json_encode([
  170. 'vendorSeq' => $vendorSeq,
  171. 'influencerSeq' => $influencerSeq,
  172. 'requestedBy' => $requestedBy
  173. ]));
  174. if (!$vendorSeq || !$influencerSeq || !$requestedBy) {
  175. return $this->response->setStatusCode(400)->setJSON([
  176. 'success' => false,
  177. 'message' => '필수 파라미터가 누락되었습니다.'
  178. ]);
  179. }
  180. // 재승인 가능한 파트너십 확인 (TERMINATED 또는 REJECTED 상태)
  181. $eligiblePartnership = $this->vendorInfluencerModel->checkReapplyEligiblePartnership($vendorSeq, $influencerSeq);
  182. if (!$eligiblePartnership) {
  183. return $this->response->setStatusCode(400)->setJSON([
  184. 'success' => false,
  185. 'message' => '재승인을 요청할 수 있는 이전 파트너십이 없습니다.'
  186. ]);
  187. }
  188. // 이미 재승인 요청 중인지 확인
  189. $existingReapply = $this->vendorInfluencerModel->checkExistingPendingRequest($vendorSeq, $influencerSeq);
  190. if ($existingReapply) {
  191. return $this->response->setStatusCode(409)->setJSON([
  192. 'success' => false,
  193. 'message' => '이미 재승인 요청이 진행 중입니다.'
  194. ]);
  195. }
  196. // 재승인 요청 생성
  197. $data = [
  198. 'VENDOR_SEQ' => $vendorSeq,
  199. 'INFLUENCER_SEQ' => $influencerSeq,
  200. 'REQUEST_TYPE' => 'INFLUENCER_REAPPLY',
  201. 'REQUEST_MESSAGE' => $requestMessage,
  202. 'REQUESTED_BY' => $requestedBy,
  203. 'COMMISSION_RATE' => $commissionRate ?: $eligiblePartnership['COMMISSION_RATE'],
  204. 'SPECIAL_CONDITIONS' => $specialConditions ?: $eligiblePartnership['SPECIAL_CONDITIONS'],
  205. 'ADD_INFO1' => 'REAPPLY',
  206. 'ADD_INFO2' => $eligiblePartnership['SEQ'], // 이전 파트너십 SEQ
  207. 'ADD_INFO3' => date('Y-m-d H:i:s') // 재신청 일시
  208. ];
  209. $mappingSeq = $this->vendorInfluencerModel->insert($data);
  210. // afterInsert 콜백에서 자동으로 PENDING 상태 히스토리 생성됨
  211. if ($mappingSeq) {
  212. log_message('debug', "재승인 요청 성공 - 새 매핑 SEQ: " . $mappingSeq);
  213. return $this->response->setStatusCode(201)->setJSON([
  214. 'success' => true,
  215. 'message' => '재승인 요청이 성공적으로 생성되었습니다.',
  216. 'data' => [
  217. 'mappingSeq' => $mappingSeq,
  218. 'status' => 'PENDING',
  219. 'isReapply' => true,
  220. 'previousPartnership' => $eligiblePartnership['SEQ']
  221. ]
  222. ]);
  223. } else {
  224. log_message('error', '재승인 요청 삽입 실패');
  225. return $this->response->setStatusCode(500)->setJSON([
  226. 'success' => false,
  227. 'message' => '재승인 요청 데이터 삽입에 실패했습니다.'
  228. ]);
  229. }
  230. } catch (\Exception $e) {
  231. log_message('error', '재승인 요청 처리 중 예외 발생: ' . $e->getMessage());
  232. log_message('error', '재승인 요청 스택 트레이스: ' . $e->getTraceAsString());
  233. return $this->response->setStatusCode(500)->setJSON([
  234. 'success' => false,
  235. 'message' => '재승인 요청 생성 중 오류가 발생했습니다.',
  236. 'error' => $e->getMessage()
  237. ]);
  238. }
  239. }
  240. /**
  241. * 내 파트너십 목록 조회 (상태 히스토리 포함)
  242. */
  243. public function getMyPartnerships()
  244. {
  245. try {
  246. $request = $this->request->getJSON();
  247. $influencerSeq = $request->influencerSeq ?? null;
  248. $status = $request->status ?? null;
  249. $page = $request->page ?? 1;
  250. $size = $request->size ?? 20;
  251. if (!$influencerSeq) {
  252. return $this->response->setStatusCode(400)->setJSON([
  253. 'success' => false,
  254. 'message' => '인플루언서 SEQ는 필수입니다.'
  255. ]);
  256. }
  257. $result = $this->influencerPartnershipModel->getInfluencerPartnerships($influencerSeq, $page, $size, $status);
  258. return $this->response->setJSON([
  259. 'success' => true,
  260. 'data' => $result['data'],
  261. 'pagination' => $result['pagination']
  262. ]);
  263. } catch (\Exception $e) {
  264. log_message('error', '파트너십 목록 조회 오류: ' . $e->getMessage());
  265. return $this->response->setStatusCode(500)->setJSON([
  266. 'success' => false,
  267. 'message' => '파트너십 목록 조회 중 오류가 발생했습니다.',
  268. 'error' => $e->getMessage()
  269. ]);
  270. }
  271. }
  272. /**
  273. * 파트너십 해지 (히스토리 테이블 기반)
  274. */
  275. public function terminatePartnership()
  276. {
  277. try {
  278. $request = $this->request->getJSON();
  279. $mappingSeq = $request->mappingSeq ?? null;
  280. $reason = $request->reason ?? '';
  281. $terminatedBy = $request->terminatedBy ?? null;
  282. if (!$mappingSeq || !$terminatedBy) {
  283. return $this->response->setStatusCode(400)->setJSON([
  284. 'success' => false,
  285. 'message' => '필수 파라미터가 누락되었습니다.'
  286. ]);
  287. }
  288. // 현재 상태 확인
  289. $mapping = $this->vendorInfluencerModel->getWithCurrentStatus($mappingSeq);
  290. if (!$mapping) {
  291. return $this->response->setStatusCode(404)->setJSON([
  292. 'success' => false,
  293. 'message' => '해당 파트너십을 찾을 수 없습니다.'
  294. ]);
  295. }
  296. if ($mapping['CURRENT_STATUS'] !== 'APPROVED') {
  297. return $this->response->setStatusCode(400)->setJSON([
  298. 'success' => false,
  299. 'message' => '승인된 파트너십만 해지할 수 있습니다.'
  300. ]);
  301. }
  302. // 상태를 TERMINATED로 변경
  303. $this->statusHistoryModel->changeStatus($mappingSeq, 'TERMINATED', '파트너십 해지: ' . $reason, $terminatedBy);
  304. // 해지 날짜 업데이트
  305. $this->vendorInfluencerModel->update($mappingSeq, [
  306. 'PARTNERSHIP_END_DATE' => date('Y-m-d H:i:s')
  307. ]);
  308. return $this->response->setJSON([
  309. 'success' => true,
  310. 'message' => '파트너십이 해지되었습니다.',
  311. 'data' => [
  312. 'mappingSeq' => $mappingSeq,
  313. 'status' => 'TERMINATED'
  314. ]
  315. ]);
  316. } catch (\Exception $e) {
  317. log_message('error', '파트너십 해지 오류: ' . $e->getMessage());
  318. return $this->response->setStatusCode(500)->setJSON([
  319. 'success' => false,
  320. 'message' => '파트너십 해지 중 오류가 발생했습니다.',
  321. 'error' => $e->getMessage()
  322. ]);
  323. }
  324. }
  325. /**
  326. * 인플루언서 프로필 조회
  327. */
  328. public function getProfile()
  329. {
  330. try {
  331. $request = $this->request->getJSON();
  332. $influencerSeq = $request->influencerSeq ?? null;
  333. if (!$influencerSeq) {
  334. return $this->response->setStatusCode(400)->setJSON([
  335. 'success' => false,
  336. 'message' => '인플루언서 SEQ는 필수입니다.'
  337. ]);
  338. }
  339. $profile = $this->influencerModel
  340. ->where('SEQ', $influencerSeq)
  341. ->where('IS_ACT', 'Y')
  342. ->first();
  343. if (!$profile) {
  344. return $this->response->setStatusCode(404)->setJSON([
  345. 'success' => false,
  346. 'message' => '인플루언서를 찾을 수 없습니다.'
  347. ]);
  348. }
  349. return $this->response->setJSON([
  350. 'success' => true,
  351. 'data' => $profile
  352. ]);
  353. } catch (\Exception $e) {
  354. log_message('error', '인플루언서 프로필 조회 오류: ' . $e->getMessage());
  355. return $this->response->setStatusCode(500)->setJSON([
  356. 'success' => false,
  357. 'message' => '프로필 조회 중 오류가 발생했습니다.',
  358. 'error' => $e->getMessage()
  359. ]);
  360. }
  361. }
  362. }