InfluencerControllerV2.php 15 KB

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