InfluencerController.php 16 KB

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