VendorInfluencerController.php 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622
  1. <?php
  2. namespace App\Controllers;
  3. use App\Controllers\BaseController;
  4. use App\Models\VendorModel;
  5. use App\Models\UserModel;
  6. use App\Models\VendorInfluencerMappingModel;
  7. use CodeIgniter\HTTP\ResponseInterface;
  8. class VendorInfluencerController extends BaseController
  9. {
  10. protected $vendorModel;
  11. protected $userModel;
  12. protected $vendorInfluencerModel;
  13. public function __construct()
  14. {
  15. $this->vendorModel = new VendorModel();
  16. $this->userModel = new UserModel();
  17. $this->vendorInfluencerModel = new VendorInfluencerMappingModel();
  18. }
  19. /**
  20. * 벤더사 검색
  21. */
  22. public function searchVendors()
  23. {
  24. try {
  25. $request = $this->request->getJSON();
  26. $keyword = $request->keyword ?? '';
  27. $category = $request->category ?? '';
  28. $region = $request->region ?? '';
  29. $sortBy = $request->sortBy ?? 'latest';
  30. $page = intval($request->page ?? 1);
  31. $size = intval($request->size ?? 12);
  32. $influencerSeq = $request->influencerSeq ?? null;
  33. $builder = $this->vendorModel->builder();
  34. $builder->where('IS_ACT', 'Y');
  35. // 키워드 검색
  36. if (!empty($keyword)) {
  37. $builder->groupStart()
  38. ->like('COMPANY_NAME', $keyword)
  39. ->orLike('DESCRIPTION', $keyword)
  40. ->orLike('TAGS', $keyword)
  41. ->groupEnd();
  42. }
  43. // 카테고리 필터
  44. if (!empty($category)) {
  45. $builder->where('CATEGORY', $category);
  46. }
  47. // 지역 필터
  48. if (!empty($region)) {
  49. $builder->where('REGION', $region);
  50. }
  51. // 정렬
  52. switch ($sortBy) {
  53. case 'partnership':
  54. $builder->orderBy('PARTNERSHIP_COUNT', 'DESC')
  55. ->orderBy('REG_DATE', 'DESC');
  56. break;
  57. case 'name':
  58. $builder->orderBy('COMPANY_NAME', 'ASC');
  59. break;
  60. case 'latest':
  61. default:
  62. $builder->orderBy('REG_DATE', 'DESC');
  63. break;
  64. }
  65. // 전체 개수
  66. $totalCount = $builder->countAllResults(false);
  67. // 페이징
  68. $offset = ($page - 1) * $size;
  69. $vendors = $builder->limit($size, $offset)->get()->getResultArray();
  70. // 파트너십 개수 추가
  71. foreach ($vendors as &$vendor) {
  72. $partnershipCount = $this->vendorInfluencerModel
  73. ->where('VENDOR_SEQ', $vendor['SEQ'])
  74. ->where('STATUS', 'APPROVED')
  75. ->where('IS_ACT', 'Y')
  76. ->countAllResults();
  77. $vendor['PARTNERSHIP_COUNT'] = $partnershipCount;
  78. // 인플루언서의 파트너십 상태 확인
  79. if ($influencerSeq) {
  80. $partnershipStatus = $this->vendorInfluencerModel
  81. ->where('VENDOR_SEQ', $vendor['SEQ'])
  82. ->where('INFLUENCER_SEQ', $influencerSeq)
  83. ->where('IS_ACT', 'Y')
  84. ->first();
  85. $vendor['PARTNERSHIP_STATUS'] = $partnershipStatus['STATUS'] ?? null;
  86. $vendor['REQUEST_TYPE'] = $partnershipStatus['REQUEST_TYPE'] ?? null;
  87. }
  88. }
  89. $totalPages = ceil($totalCount / $size);
  90. return $this->response->setJSON([
  91. 'success' => true,
  92. 'data' => [
  93. 'items' => $vendors,
  94. 'pagination' => [
  95. 'currentPage' => $page,
  96. 'totalPages' => $totalPages,
  97. 'totalCount' => $totalCount,
  98. 'pageSize' => $size,
  99. 'hasNext' => $page < $totalPages,
  100. 'hasPrev' => $page > 1
  101. ]
  102. ]
  103. ]);
  104. } catch (\Exception $e) {
  105. return $this->response->setStatusCode(500)->setJSON([
  106. 'success' => false,
  107. 'message' => '벤더사 검색 중 오류가 발생했습니다.',
  108. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  109. ]);
  110. }
  111. }
  112. /**
  113. * 승인 요청 생성
  114. */
  115. public function createRequest()
  116. {
  117. try {
  118. $request = $this->request->getJSON();
  119. $vendorSeq = $request->vendorSeq ?? null;
  120. $influencerSeq = $request->influencerSeq ?? null;
  121. $requestType = $request->requestType ?? 'INFLUENCER_REQUEST';
  122. $requestMessage = $request->requestMessage ?? '';
  123. $requestedBy = $request->requestedBy ?? null;
  124. $commissionRate = $request->commissionRate ?? null;
  125. $specialConditions = $request->specialConditions ?? '';
  126. // 필수 파라미터 검증
  127. if (!$vendorSeq || !$influencerSeq || !$requestedBy) {
  128. return $this->response->setStatusCode(400)->setJSON([
  129. 'success' => false,
  130. 'message' => '필수 파라미터가 누락되었습니다.'
  131. ]);
  132. }
  133. // 중복 요청 확인
  134. $existingRequest = $this->vendorInfluencerModel
  135. ->where('VENDOR_SEQ', $vendorSeq)
  136. ->where('INFLUENCER_SEQ', $influencerSeq)
  137. ->where('STATUS', 'PENDING')
  138. ->where('IS_ACT', 'Y')
  139. ->first();
  140. if ($existingRequest) {
  141. return $this->response->setStatusCode(409)->setJSON([
  142. 'success' => false,
  143. 'message' => '이미 처리 중인 요청이 있습니다.'
  144. ]);
  145. }
  146. // 요청 생성
  147. $data = [
  148. 'VENDOR_SEQ' => $vendorSeq,
  149. 'INFLUENCER_SEQ' => $influencerSeq,
  150. 'REQUEST_TYPE' => $requestType,
  151. 'STATUS' => 'PENDING',
  152. 'REQUEST_MESSAGE' => $requestMessage,
  153. 'REQUESTED_BY' => $requestedBy,
  154. 'COMMISSION_RATE' => $commissionRate,
  155. 'SPECIAL_CONDITIONS' => $specialConditions,
  156. 'EXPIRED_DATE' => date('Y-m-d H:i:s', strtotime('+7 days')),
  157. 'REG_DATE' => date('Y-m-d H:i:s')
  158. ];
  159. $insertId = $this->vendorInfluencerModel->insert($data);
  160. // 생성된 데이터 조회
  161. $createdRequest = $this->vendorInfluencerModel
  162. ->select('vim.*, v.COMPANY_NAME as vendorName, u.NICK_NAME as influencerName, req_user.NICK_NAME as requestedByName')
  163. ->from('VENDOR_INFLUENCER_MAPPING vim')
  164. ->join('VENDOR_LIST v', 'vim.VENDOR_SEQ = v.SEQ', 'left')
  165. ->join('USER_LIST u', 'vim.INFLUENCER_SEQ = u.SEQ', 'left')
  166. ->join('USER_LIST req_user', 'vim.REQUESTED_BY = req_user.SEQ', 'left')
  167. ->where('vim.SEQ', $insertId)
  168. ->get()
  169. ->getRowArray();
  170. return $this->response->setJSON([
  171. 'success' => true,
  172. 'message' => '승인 요청이 성공적으로 생성되었습니다.',
  173. 'data' => $createdRequest
  174. ]);
  175. } catch (\Exception $e) {
  176. return $this->response->setStatusCode(500)->setJSON([
  177. 'success' => false,
  178. 'message' => '승인 요청 생성 중 오류가 발생했습니다.',
  179. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  180. ]);
  181. }
  182. }
  183. /**
  184. * 매핑 리스트 조회 (벤더사용 승인요청 목록 포함)
  185. */
  186. public function getList()
  187. {
  188. try {
  189. $request = $this->request->getJSON();
  190. $vendorSeq = $request->vendorSeq ?? null;
  191. $influencerSeq = $request->influencerSeq ?? null;
  192. $status = $request->status ?? null;
  193. $requestType = $request->requestType ?? null;
  194. $keyword = $request->keyword ?? null;
  195. $category = $request->category ?? null;
  196. $sortBy = $request->sortBy ?? 'latest';
  197. $page = intval($request->page ?? 1);
  198. $size = intval($request->size ?? 12);
  199. $builder = $this->vendorInfluencerModel->builder();
  200. $builder->select('vim.*, v.COMPANY_NAME as vendorName, v.EMAIL as vendorEmail, v.LOGO as vendorLogo,
  201. u.NICK_NAME as influencerNickname, u.NAME as influencerName,
  202. u.EMAIL as influencerEmail, u.PHONE as influencerPhone,
  203. u.PROFILE_IMAGE as influencerAvatar, u.REGION as influencerRegion,
  204. u.PRIMARY_CATEGORY as influencerCategory,
  205. u.FOLLOWER_COUNT as followerCount,
  206. u.AVG_VIEWS as avgViews,
  207. u.ENGAGEMENT_RATE as engagementRate,
  208. u.DESCRIPTION as influencerDescription,
  209. u.SNS_CHANNELS as influencerSnsChannels,
  210. req_user.NICK_NAME as requestedByName,
  211. app_user.NICK_NAME as approvedByName,
  212. CASE WHEN vim.EXPIRED_DATE < NOW() AND vim.STATUS = "PENDING"
  213. THEN "EXPIRED" ELSE vim.STATUS END as displayStatus')
  214. ->from('VENDOR_INFLUENCER_MAPPING vim')
  215. ->join('VENDOR_LIST v', 'vim.VENDOR_SEQ = v.SEQ', 'left')
  216. ->join('USER_LIST u', 'vim.INFLUENCER_SEQ = u.SEQ', 'left')
  217. ->join('USER_LIST req_user', 'vim.REQUESTED_BY = req_user.SEQ', 'left')
  218. ->join('USER_LIST app_user', 'vim.APPROVED_BY = app_user.SEQ', 'left')
  219. ->where('vim.IS_ACT', 'Y');
  220. // 필터 적용
  221. if ($vendorSeq) {
  222. $builder->where('vim.VENDOR_SEQ', $vendorSeq);
  223. }
  224. if ($influencerSeq) {
  225. $builder->where('vim.INFLUENCER_SEQ', $influencerSeq);
  226. }
  227. if ($status) {
  228. $builder->where('vim.STATUS', $status);
  229. }
  230. if ($requestType) {
  231. $builder->where('vim.REQUEST_TYPE', $requestType);
  232. }
  233. // 키워드 검색 (인플루언서명)
  234. if (!empty($keyword)) {
  235. $builder->groupStart()
  236. ->like('u.NICK_NAME', $keyword)
  237. ->orLike('u.NAME', $keyword)
  238. ->groupEnd();
  239. }
  240. // 카테고리 필터 (인플루언서 카테고리)
  241. if (!empty($category)) {
  242. $builder->where('u.PRIMARY_CATEGORY', $category);
  243. }
  244. // 전체 개수
  245. $totalCount = $builder->countAllResults(false);
  246. // 정렬
  247. switch ($sortBy) {
  248. case 'oldest':
  249. $builder->orderBy('vim.REG_DATE', 'ASC');
  250. break;
  251. case 'expiring':
  252. $builder->orderBy('vim.EXPIRED_DATE', 'ASC');
  253. break;
  254. case 'latest':
  255. default:
  256. $builder->orderBy('vim.REG_DATE', 'DESC');
  257. break;
  258. }
  259. // 페이징
  260. $offset = ($page - 1) * $size;
  261. $items = $builder->limit($size, $offset)->get()->getResultArray();
  262. $totalPages = ceil($totalCount / $size);
  263. // 통계 데이터 계산 (벤더사용)
  264. $stats = [];
  265. if ($vendorSeq) {
  266. $statsBuilder = $this->vendorInfluencerModel->builder();
  267. $stats = [
  268. 'pending' => $statsBuilder->where('VENDOR_SEQ', $vendorSeq)->where('STATUS', 'PENDING')->where('IS_ACT', 'Y')->countAllResults(),
  269. 'approved' => $statsBuilder->resetQuery()->where('VENDOR_SEQ', $vendorSeq)->where('STATUS', 'APPROVED')->where('IS_ACT', 'Y')->countAllResults(),
  270. 'rejected' => $statsBuilder->resetQuery()->where('VENDOR_SEQ', $vendorSeq)->where('STATUS', 'REJECTED')->where('IS_ACT', 'Y')->countAllResults(),
  271. 'total' => $totalCount
  272. ];
  273. }
  274. return $this->response->setJSON([
  275. 'success' => true,
  276. 'data' => [
  277. 'items' => $items,
  278. 'pagination' => [
  279. 'currentPage' => $page,
  280. 'totalPages' => $totalPages,
  281. 'totalCount' => $totalCount,
  282. 'pageSize' => $size,
  283. 'hasNext' => $page < $totalPages,
  284. 'hasPrev' => $page > 1
  285. ],
  286. 'stats' => $stats
  287. ]
  288. ]);
  289. } catch (\Exception $e) {
  290. return $this->response->setStatusCode(500)->setJSON([
  291. 'success' => false,
  292. 'message' => '리스트 조회 중 오류가 발생했습니다.',
  293. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  294. ]);
  295. }
  296. }
  297. /**
  298. * 요청 취소
  299. */
  300. public function cancelRequest()
  301. {
  302. try {
  303. $request = $this->request->getJSON();
  304. $mappingSeq = $request->mappingSeq ?? null;
  305. $cancelledBy = $request->cancelledBy ?? null;
  306. $cancelReason = $request->cancelReason ?? '요청자에 의해 취소됨';
  307. // 필수 파라미터 검증
  308. if (!$mappingSeq || !$cancelledBy) {
  309. return $this->response->setStatusCode(400)->setJSON([
  310. 'success' => false,
  311. 'message' => '필수 파라미터가 누락되었습니다.'
  312. ]);
  313. }
  314. // 기존 요청 확인
  315. $existingMapping = $this->vendorInfluencerModel
  316. ->where('SEQ', $mappingSeq)
  317. ->where('STATUS', 'PENDING')
  318. ->where('IS_ACT', 'Y')
  319. ->first();
  320. if (!$existingMapping) {
  321. return $this->response->setStatusCode(404)->setJSON([
  322. 'success' => false,
  323. 'message' => '취소할 수 있는 요청을 찾을 수 없습니다.'
  324. ]);
  325. }
  326. // 권한 확인
  327. if ($existingMapping['REQUESTED_BY'] != $cancelledBy) {
  328. return $this->response->setStatusCode(403)->setJSON([
  329. 'success' => false,
  330. 'message' => '요청을 취소할 권한이 없습니다.'
  331. ]);
  332. }
  333. // 취소 처리
  334. $updateData = [
  335. 'STATUS' => 'CANCELLED',
  336. 'RESPONSE_MESSAGE' => $cancelReason,
  337. 'RESPONSE_DATE' => date('Y-m-d H:i:s'),
  338. 'APPROVED_BY' => $cancelledBy
  339. ];
  340. $this->vendorInfluencerModel->update($mappingSeq, $updateData);
  341. // 업데이트된 데이터 조회
  342. $updatedMapping = $this->vendorInfluencerModel
  343. ->select('vim.*, v.COMPANY_NAME as vendorName, u.NICK_NAME as influencerName, req_user.NICK_NAME as requestedByName')
  344. ->from('VENDOR_INFLUENCER_MAPPING vim')
  345. ->join('VENDOR_LIST v', 'vim.VENDOR_SEQ = v.SEQ', 'left')
  346. ->join('USER_LIST u', 'vim.INFLUENCER_SEQ = u.SEQ', 'left')
  347. ->join('USER_LIST req_user', 'vim.REQUESTED_BY = req_user.SEQ', 'left')
  348. ->where('vim.SEQ', $mappingSeq)
  349. ->get()
  350. ->getRowArray();
  351. return $this->response->setJSON([
  352. 'success' => true,
  353. 'message' => '요청이 취소되었습니다.',
  354. 'data' => $updatedMapping
  355. ]);
  356. } catch (\Exception $e) {
  357. return $this->response->setStatusCode(500)->setJSON([
  358. 'success' => false,
  359. 'message' => '요청 취소 중 오류가 발생했습니다.',
  360. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  361. ]);
  362. }
  363. }
  364. /**
  365. * 요청 승인/거부 처리
  366. */
  367. public function approveRequest()
  368. {
  369. try {
  370. $request = $this->request->getJSON();
  371. $mappingSeq = $request->mappingSeq ?? null;
  372. $action = $request->action ?? 'APPROVE'; // 'APPROVE' or 'REJECT'
  373. $processedBy = $request->processedBy ?? $request->approvedBy ?? null; // 기존 코드 호환성
  374. $responseMessage = $request->responseMessage ?? '';
  375. $commissionRate = $request->commissionRate ?? null;
  376. if (!$mappingSeq || !$processedBy) {
  377. return $this->response->setStatusCode(400)->setJSON([
  378. 'success' => false,
  379. 'message' => '필수 파라미터가 누락되었습니다.'
  380. ]);
  381. }
  382. // 처리자 벤더사 존재 확인
  383. $processingVendor = $this->vendorModel
  384. ->where('SEQ', $processedBy)
  385. ->first(); // IS_ACT 조건 제거해서 벤더사 존재 여부만 확인
  386. if (!$processingVendor) {
  387. return $this->response->setStatusCode(400)->setJSON([
  388. 'success' => false,
  389. 'message' => "처리자 SEQ {$processedBy}가 VENDOR_LIST 테이블에 존재하지 않습니다."
  390. ]);
  391. }
  392. // 벤더사 활성 상태 확인
  393. if ($processingVendor['IS_ACT'] !== 'Y') {
  394. return $this->response->setStatusCode(400)->setJSON([
  395. 'success' => false,
  396. 'message' => "처리자 SEQ {$processedBy}는 비활성 상태입니다. (IS_ACT: {$processingVendor['IS_ACT']})"
  397. ]);
  398. }
  399. // 기존 요청 확인
  400. $existingMapping = $this->vendorInfluencerModel
  401. ->where('SEQ', $mappingSeq)
  402. ->where('STATUS', 'PENDING')
  403. ->where('IS_ACT', 'Y')
  404. ->first();
  405. if (!$existingMapping) {
  406. return $this->response->setStatusCode(404)->setJSON([
  407. 'success' => false,
  408. 'message' => '처리할 수 있는 요청을 찾을 수 없습니다.'
  409. ]);
  410. }
  411. // 벤더사 권한 확인 (선택적)
  412. // 벤더사만 자신의 요청을 처리할 수 있도록 하려면 주석 해제
  413. // if ($existingMapping['VENDOR_SEQ'] !== $processingUser['SEQ']) {
  414. // return $this->response->setStatusCode(403)->setJSON([
  415. // 'success' => false,
  416. // 'message' => '해당 요청을 처리할 권한이 없습니다.'
  417. // ]);
  418. // }
  419. $status = $action === 'APPROVE' ? 'APPROVED' : 'REJECTED';
  420. $updateData = [
  421. 'STATUS' => $status,
  422. 'APPROVED_BY' => $processedBy,
  423. 'RESPONSE_MESSAGE' => $responseMessage,
  424. 'RESPONSE_DATE' => date('Y-m-d H:i:s')
  425. ];
  426. if ($commissionRate !== null && $action === 'APPROVE') {
  427. $updateData['COMMISSION_RATE'] = $commissionRate;
  428. }
  429. // 단계별 업데이트로 외래키 제약조건 문제 우회
  430. try {
  431. // 1단계: APPROVED_BY 없이 먼저 업데이트
  432. $firstUpdateData = [
  433. 'STATUS' => $status,
  434. 'RESPONSE_MESSAGE' => $responseMessage,
  435. 'RESPONSE_DATE' => date('Y-m-d H:i:s')
  436. ];
  437. if ($commissionRate !== null && $action === 'APPROVE') {
  438. $firstUpdateData['COMMISSION_RATE'] = $commissionRate;
  439. }
  440. log_message('debug', "1단계 업데이트 시도: " . json_encode($firstUpdateData));
  441. $result1 = $this->vendorInfluencerModel->update($mappingSeq, $firstUpdateData);
  442. log_message('debug', "1단계 업데이트 결과: " . ($result1 ? 'SUCCESS' : 'FAILED'));
  443. // 2단계: APPROVED_BY만 별도로 업데이트
  444. $secondUpdateData = [
  445. 'APPROVED_BY' => $processedBy
  446. ];
  447. log_message('debug', "2단계 업데이트 시도: " . json_encode($secondUpdateData));
  448. $result2 = $this->vendorInfluencerModel->update($mappingSeq, $secondUpdateData);
  449. log_message('debug', "2단계 업데이트 결과: " . ($result2 ? 'SUCCESS' : 'FAILED'));
  450. if (!$result1 || !$result2) {
  451. throw new \Exception("단계별 업데이트 실패: 1단계={$result1}, 2단계={$result2}");
  452. }
  453. } catch (\Exception $updateException) {
  454. log_message('error', "단계별 업데이트 실패: " . $updateException->getMessage());
  455. // 외래키 제약조건 오류인 경우 더 상세한 정보 제공
  456. if (strpos($updateException->getMessage(), 'foreign key constraint') !== false) {
  457. // 추가 디버깅: 직접 SQL로 사용자 존재 확인
  458. $db = \Config\Database::connect();
  459. $userCheck = $db->query("SELECT SEQ, NICK_NAME, EMAIL, IS_ACT FROM USER_LIST WHERE SEQ = ?", [$processedBy])->getRowArray();
  460. // 추가 디버깅: 현재 매핑 레코드 상태 확인
  461. $mappingCheck = $db->query("SELECT SEQ, STATUS, APPROVED_BY FROM VENDOR_INFLUENCER_MAPPING WHERE SEQ = ?", [$mappingSeq])->getRowArray();
  462. return $this->response->setStatusCode(500)->setJSON([
  463. 'success' => false,
  464. 'message' => '외래키 제약조건 오류가 발생했습니다.',
  465. 'error' => $updateException->getMessage(),
  466. 'debug_info' => [
  467. 'processedBy' => $processedBy,
  468. 'processingVendor' => $processingVendor,
  469. 'existingMapping' => $existingMapping,
  470. 'userCheck' => $userCheck,
  471. 'mappingCheck' => $mappingCheck,
  472. 'updateData' => $updateData
  473. ]
  474. ]);
  475. }
  476. throw $updateException; // 다른 예외는 기존 처리 방식 유지
  477. }
  478. $actionText = $action === 'APPROVE' ? '승인' : '거부';
  479. return $this->response->setJSON([
  480. 'success' => true,
  481. 'message' => "요청이 {$actionText}되었습니다.",
  482. 'data' => $updateData
  483. ]);
  484. } catch (\Exception $e) {
  485. return $this->response->setStatusCode(500)->setJSON([
  486. 'success' => false,
  487. 'message' => '요청 처리 중 오류가 발생했습니다.',
  488. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  489. ]);
  490. }
  491. }
  492. /**
  493. * 상세 조회
  494. */
  495. public function getDetail()
  496. {
  497. try {
  498. $request = $this->request->getJSON();
  499. $mappingSeq = $request->mappingSeq ?? null;
  500. if (!$mappingSeq) {
  501. return $this->response->setStatusCode(400)->setJSON([
  502. 'success' => false,
  503. 'message' => '매핑 SEQ가 필요합니다.'
  504. ]);
  505. }
  506. $detail = $this->vendorInfluencerModel
  507. ->select('vim.*, v.*, u.NICK_NAME as influencerName, u.EMAIL as influencerEmail')
  508. ->from('VENDOR_INFLUENCER_MAPPING vim')
  509. ->join('VENDOR_LIST v', 'vim.VENDOR_SEQ = v.SEQ', 'left')
  510. ->join('USER_LIST u', 'vim.INFLUENCER_SEQ = u.SEQ', 'left')
  511. ->where('vim.SEQ', $mappingSeq)
  512. ->where('vim.IS_ACT', 'Y')
  513. ->get()
  514. ->getRowArray();
  515. if (!$detail) {
  516. return $this->response->setStatusCode(404)->setJSON([
  517. 'success' => false,
  518. 'message' => '요청을 찾을 수 없습니다.'
  519. ]);
  520. }
  521. return $this->response->setJSON([
  522. 'success' => true,
  523. 'data' => $detail
  524. ]);
  525. } catch (\Exception $e) {
  526. return $this->response->setStatusCode(500)->setJSON([
  527. 'success' => false,
  528. 'message' => '상세 조회 중 오류가 발생했습니다.',
  529. 'error' => ENVIRONMENT === 'development' ? $e->getMessage() : null
  530. ]);
  531. }
  532. }
  533. }