Auth.php 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911
  1. <?php
  2. namespace App\Controllers;
  3. require_once __DIR__ . '/../Libraries/autoload.php';
  4. use App\Models\UserListModel;
  5. use CodeIgniter\RESTful\ResourceController;
  6. use Google\Client;
  7. use Google\Service\Oauth2;
  8. use App\Libraries\JwtLib\JWT;
  9. use App\Libraries\JwtLib\Key;
  10. class Auth extends ResourceController
  11. {
  12. private const FRONTEND_BASE_URL = "http://localhost:3000";
  13. //private const FRONTEND_BASE_URL = 'https://shopdeli.mycafe24.com';
  14. protected $userModel;
  15. public function __construct()
  16. {
  17. $this->userModel = new UserListModel();
  18. }
  19. //구글 로그인 콜백(인증)
  20. public function callback()
  21. {
  22. if (isset($_GET['code'])) {
  23. $code = $_GET['code'];
  24. // Google 클라이언트 설정
  25. $client = new Client();
  26. $client->setClientId('373780605211-diojebh7mug45urv9rnqdil6n0b1ogge.apps.googleusercontent.com');
  27. $client->setClientSecret('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');
  28. $client->setRedirectUri('https://shopdeli.mycafe24.com/auth/callback');
  29. $client->setAccessType('offline'); // 인증 요청 전에 지정!
  30. $client->authenticate($code);
  31. $googleAccessToken = $client->getAccessToken();
  32. // 별도의 변수로 분리
  33. $googleOAuthAccessToken = $googleAccessToken['access_token'] ?? null;
  34. $googleOAuthRefreshToken = $googleAccessToken['refresh_token'] ?? null;
  35. // 업데이트: setAccessToken에는 전체 배열을 넣을 수 있지만
  36. // accessToken 만 넣으려면 아래처럼 쓸 수 있습니다.
  37. if ($googleOAuthAccessToken) {
  38. $client->setAccessToken($googleOAuthAccessToken);
  39. }
  40. // 사용자 정보 가져오기
  41. $oauth2 = new Oauth2($client);
  42. $userInfo = $oauth2->userinfo->get();
  43. // 사용자 정보 변수 지정
  44. $id = $userInfo->id;
  45. $authenticatedEmail = $userInfo->email;
  46. $name = $userInfo->name;
  47. // DB Connection (CodeIgniter 4)
  48. $db = \Config\Database::connect();
  49. // 1. USER_LIST에서 이메일로 조회
  50. $builder = $db->table('USER_LIST');
  51. $user = $builder->where('EMAIL', $authenticatedEmail)->get()->getRowArray();
  52. if ($user) {
  53. $jwtSecret = env('JWT_SECRET');
  54. $kid = env('JWT_KID');
  55. if (empty($jwtSecret) || empty($kid)) {
  56. return $this->failServerError('환경변수가 누락되었습니다. 관리자에게 문의하세요.');
  57. }
  58. if (!class_exists('\App\Libraries\JwtLib\JWT')) {
  59. return $this->failServerError('JWT 라이브러리를 찾을 수 없습니다.');
  60. }
  61. $issuedAt = time();
  62. $accessExpire = $issuedAt + 60 * 15; // 15분
  63. $refreshExpire = $issuedAt + 60 * 60 * 24 * 14; // 14일
  64. $accessPayload = [
  65. 'iat' => $issuedAt,
  66. 'exp' => $accessExpire,
  67. 'sub' => $id,
  68. 'name' => $name ?? '',
  69. ];
  70. $refreshPayload = [
  71. 'iat' => $issuedAt,
  72. 'exp' => $refreshExpire,
  73. 'sub' => $id,
  74. ];
  75. try {
  76. $accessToken = JWT::encode($accessPayload, $jwtSecret, 'HS256', $kid);
  77. $refreshToken = JWT::encode($refreshPayload, $jwtSecret, 'HS256', $kid);
  78. // (선택) DB의 리프레시 토큰 값 업데이트
  79. $db = \Config\Database::connect();
  80. $builder = $db->table('USER_LIST');
  81. $builder->where('ID', $id)->update(['REFRESH_TOKEN' => $refreshToken]);
  82. } catch (\Throwable $e) {
  83. return $this->failServerError('JWT 생성 오류: ' . $e->getMessage());
  84. }
  85. unset($user['PASSWORD']); // 혹시 비번이 있다면 노출 방지
  86. $query = http_build_query([
  87. "accessToken" => $accessToken,
  88. "refreshToken" => $refreshToken,
  89. "user" => urlencode(json_encode($user)),
  90. ]);
  91. //return redirect()->to(base_url("auth/popupClose?$query"));
  92. // 개발진행중 풀 URL
  93. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  94. } else {
  95. // 회원이 없으면 회원가입
  96. $type = 1;
  97. // ID는 auto_increment라면 생략
  98. $authData = [
  99. 'ID' => $id,
  100. 'NAME' => $name,
  101. 'EMAIL' => $authenticatedEmail,
  102. 'TYPE' => $type,
  103. 'JOIN' => '1',
  104. 'GOOGLE_REFRESH_TOKEN' => $googleOAuthRefreshToken ?? null
  105. ];
  106. $query = http_build_query([
  107. "user" => urlencode(json_encode($authData)),
  108. ]);
  109. //return redirect()->to(base_url("auth/popupClose?$query"));
  110. // 개발진행중 풀 URL
  111. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  112. }
  113. } else {
  114. echo "인증코드가 없습니다.";
  115. }
  116. }
  117. public function join()
  118. {
  119. $postData = $this->request->getJSON(true);
  120. // 필수값 추출
  121. $id = $postData['ID'] ?? null;
  122. $password = $postData['PASSWORD'] ?? null;
  123. $name = $postData['NAME'] ?? null;
  124. $nick_name = $postData['NICK_NAME'] ?? null;
  125. $phone = $postData['PHONE'] ?? null;
  126. $email = $postData['EMAIL'] ?? null;
  127. $sns_type = $postData['SNS_TYPE'] ?? null;
  128. $sns_link_id = $postData['SNS_LINK_ID'] ?? null;
  129. $add_info1 = $postData['ADD_INFO1'] ?? null;
  130. $google_refresh_token = $postData['GOOGLE_REFRESH_TOKEN'] ?? null;
  131. $type = $postData['TYPE'] ?? null;
  132. // 필수값 검증
  133. if (empty($name) || empty($email)) {
  134. return redirect()->back()->with('error', '필수 정보를 입력해 주세요.');
  135. }
  136. // insert용 데이터 준비
  137. $insertData = [
  138. 'ID' => $id,
  139. 'PASSWORD' => $password,
  140. 'NAME' => $name,
  141. 'NICK_NAME' => $nick_name,
  142. 'PHONE' => $phone,
  143. 'EMAIL' => $email,
  144. 'SNS_TYPE' => $sns_type,
  145. 'SNS_LINK_ID' => $sns_link_id,
  146. 'ADD_INFO1' => $add_info1,
  147. 'GOOGLE_REFRESH_TOKEN' => $google_refresh_token,
  148. 'TYPE' => $type,
  149. ];
  150. if (!empty($password)) {
  151. $insertData['PASSWORD'] = password_hash($password, PASSWORD_DEFAULT);
  152. }
  153. // DB 연결 및 INSERT
  154. try {
  155. $db = \Config\Database::connect();
  156. $builder = $db->table('USER_LIST');
  157. $result = $builder->insert($insertData);
  158. if (!$result) {
  159. $error = $db->error();
  160. return $this->response->setJSON([
  161. 'status' => 'fail',
  162. 'message' => 'DB Error: '.$error['message']
  163. ])->setStatusCode(500);
  164. }
  165. } catch (\Throwable $e) {
  166. return $this->response->setJSON([
  167. 'status' => 'fail',
  168. 'message' => 'Exception: ' . $e->getMessage()
  169. ])->setStatusCode(500);
  170. }
  171. // 회원가입 성공 시, JSON 응답으로 결과 반환 (200 OK)
  172. return $this->response->setJSON([
  173. 'status' => 'success',
  174. 'message' => '회원가입이 완료되었습니다.'
  175. ])->setStatusCode(200);
  176. }
  177. public function joinVendor()
  178. {
  179. $postData = $this->request->getJSON(true);
  180. // 필수값 추출
  181. $id = $postData['ID'] ?? null;
  182. $password = $postData['PASSWORD'] ?? null;
  183. $name = $postData['NAME'] ?? null;
  184. $company_name = $postData['COMPANY_NAME'] ?? null;
  185. $company_number = $postData['COMPANY_NUMBER'] ?? null;
  186. $hp = $postData['HP'] ?? null;
  187. $email = $postData['EMAIL'] ?? null;
  188. // 필수값 검증
  189. if (empty($name) || empty($email)) {
  190. return redirect()->back()->with('error', '필수 정보를 입력해 주세요.');
  191. }
  192. // insert용 데이터 준비
  193. $insertData = [
  194. 'ID' => $id,
  195. 'PASSWORD' => $password,
  196. 'NAME' => $name,
  197. 'COMPANY_NAME' => $company_name,
  198. 'HP' => $hp,
  199. 'EMAIL' => $email,
  200. 'COMPANY_NUMBER' => $company_number
  201. ];
  202. if (!empty($password)) {
  203. $insertData['PASSWORD'] = password_hash($password, PASSWORD_DEFAULT);
  204. }
  205. // DB 연결 및 INSERT
  206. try {
  207. $db = \Config\Database::connect();
  208. $builder = $db->table('VENDOR_LIST');
  209. $result = $builder->insert($insertData);
  210. if (!$result) {
  211. $error = $db->error();
  212. return $this->response->setJSON([
  213. 'status' => 'fail',
  214. 'message' => 'DB Error: '.$error['message']
  215. ])->setStatusCode(500);
  216. }
  217. } catch (\Throwable $e) {
  218. return $this->response->setJSON([
  219. 'status' => 'fail',
  220. 'message' => 'Exception: ' . $e->getMessage()
  221. ])->setStatusCode(500);
  222. }
  223. // 회원가입 성공 시, JSON 응답으로 결과 반환 (200 OK)
  224. return $this->response->setJSON([
  225. 'status' => 'success',
  226. 'message' => '회원가입이 완료되었습니다.'
  227. ])->setStatusCode(200);
  228. }
  229. //구글 로그인 환경 회원 탈퇴
  230. public function withdrawal()
  231. {
  232. // 1. 요청에서 사용자 SEQ 추출 (예: POST or GET)
  233. $postData = $this->request->getJSON(true);
  234. $seq = $postData['SEQ'];
  235. $googleAccessToken = $postData['GOOGLE_REFRESH_TOKEN'];
  236. // 2. 사용자 정보 조회 (구글 토큰 포함)
  237. $user = $this->userModel->find($seq);
  238. if (!$user) {
  239. return $this->response->setJSON(['error' => 'User not found'])->setStatusCode(404);
  240. }
  241. // 3. 구글 인증 연결 해제(access_token 필요)
  242. if ($googleAccessToken) {
  243. $this->revokeGoogleAccess($googleAccessToken);
  244. }
  245. // 4. USER_LIST에서 사용자 삭제
  246. $this->userModel->delete($seq);
  247. // 5. 응답 반환
  248. return $this->response->setJSON([
  249. 'result' => 'User account deleted and Google link revoked'
  250. ])->setStatusCode(200);
  251. }
  252. /***********************************
  253. * 카카오 간편로그인 / 가입
  254. **********************************/
  255. //카카오 인증
  256. public function kakaoLogin()
  257. {
  258. $clientId = '1f8376b18a02a00f2e4e5594f9ace6d4'; // 카카오 REST API 키로 변경
  259. $redirectUri = urlencode('https://shopdeli.mycafe24.com/auth/kakao'); // 콜백 URL (ex: https://도메인/auth/kakaoCallback)
  260. $url = "https://kauth.kakao.com/oauth/authorize?client_id={$clientId}&redirect_uri={$redirectUri}&response_type=code";
  261. return redirect()->to($url);
  262. }
  263. //카카오 인증후 진행
  264. public function kakao()
  265. {
  266. $code = $this->request->getGet('code');
  267. $clientId = '1f8376b18a02a00f2e4e5594f9ace6d4'; // 카카오 REST API 키
  268. $redirectUri = 'https://shopdeli.mycafe24.com/auth/kakao'; // 콜백 URL
  269. $tokenUrl = "https://kauth.kakao.com/oauth/token";
  270. // 토큰 요청
  271. $tokenData = [
  272. 'grant_type' => 'authorization_code',
  273. 'scopes' => 'offline_access',
  274. 'client_id' => $clientId,
  275. 'redirect_uri' => $redirectUri,
  276. 'code' => $code,
  277. ];
  278. $client = \Config\Services::curlrequest();
  279. $tokenResponse = $client->post($tokenUrl, [
  280. 'form_params' => $tokenData
  281. ]);
  282. $tokenResult = json_decode($tokenResponse->getBody(), true);
  283. $accessTokenKakao = $tokenResult['access_token'];
  284. $refreshTokenKakao = $tokenResult['refresh_token'];
  285. // 사용자 정보 요청
  286. $userUrl = "https://kapi.kakao.com/v2/user/me";
  287. $userResponse = $client->get($userUrl, [
  288. 'headers' => [
  289. 'Authorization' => 'Bearer ' . $accessTokenKakao,
  290. ]
  291. ]);
  292. $userInfo = json_decode($userResponse->getBody(), true);
  293. $userInfo['access_token'] = $accessTokenKakao;
  294. $userInfo['refresh_token'] = $refreshTokenKakao; //회원탈퇴시 이용 로그인시 마다 신규 리프래시 토큰 발급 받게 됨
  295. // 여기에 회원가입/로그인 처리를 구현하세요
  296. // 예: $userInfo['kakao_account']['email'], $userInfo['properties']['nickname']
  297. //DB 커넥션
  298. $db = \Config\Database::connect();
  299. // 1. USER_LIST에서 이메일로 조회
  300. $id = $userInfo['id'];
  301. $email = $userInfo['kakao_account']['email'];
  302. $builder = $db->table('USER_LIST');
  303. $user = $builder->where('EMAIL', $email)->get()->getRowArray();
  304. if($user){
  305. $jwtSecret = env('JWT_SECRET');
  306. $kid = env('JWT_KID');
  307. if (empty($jwtSecret) || empty($kid)) {
  308. return $this->failServerError('환경변수가 누락되었습니다. 관리자에게 문의하세요.');
  309. }
  310. if (!class_exists('\App\Libraries\JwtLib\JWT')) {
  311. return $this->failServerError('JWT 라이브러리를 찾을 수 없습니다.');
  312. }
  313. $issuedAt = time();
  314. $accessExpire = $issuedAt + 60 * 15; // 15분
  315. $refreshExpire = $issuedAt + 60 * 60 * 24 * 14; // 14일
  316. $accessPayload = [
  317. 'iat' => $issuedAt,
  318. 'exp' => $accessExpire,
  319. 'sub' => $id,
  320. //'name' => $name ?? '',
  321. ];
  322. $refreshPayload = [
  323. 'iat' => $issuedAt,
  324. 'exp' => $refreshExpire,
  325. 'sub' => $id,
  326. ];
  327. try {
  328. $accessToken = JWT::encode($accessPayload, $jwtSecret, 'HS256', $kid);
  329. $refreshToken = JWT::encode($refreshPayload, $jwtSecret, 'HS256', $kid);
  330. // (선택) DB의 리프레시 토큰 값 업데이트
  331. $db = \Config\Database::connect();
  332. $builder = $db->table('USER_LIST');
  333. $builder->where('ID', $id)->update([
  334. 'REFRESH_TOKEN' => $refreshToken,
  335. 'KAKAO_REFRESH_TOKEN' => $userInfo['refresh_token']
  336. ]);
  337. } catch (\Throwable $e) {
  338. return $this->failServerError('JWT 생성 오류: ' . $e->getMessage());
  339. }
  340. unset($user['PASSWORD']); // 혹시 비번이 있다면 노출 방지
  341. $query = http_build_query([
  342. "accessToken" => $accessToken,
  343. "refreshToken" => $refreshToken,
  344. "user" => urlencode(json_encode($user)),
  345. ]);
  346. //return redirect()->to(base_url("auth/popupClose?$query"));
  347. // 개발진행중 풀 URL
  348. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  349. }else{
  350. // 회원이 없으면 회원가입
  351. $type = 1;
  352. // ID는 auto_increment라면 생략
  353. $authData = [
  354. 'ID' => $id,
  355. //'NAME' => $name,
  356. 'EMAIL' => $email,
  357. 'TYPE' => $type,
  358. 'JOIN' => '1',
  359. 'KAKAO_REFRESH_TOKEN' => $userInfo['refresh_token'] ?? null
  360. ];
  361. $query = http_build_query([
  362. "user" => urlencode(json_encode($authData)),
  363. ]);
  364. //return redirect()->to(base_url("auth/popupClose?$query"));
  365. // 개발진행중 풀 URL
  366. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  367. }
  368. }
  369. //카카오 회원탈퇴
  370. public function kakaoWithdrawal(){
  371. // 1. 요청에서 사용자 SEQ 추출 (예: POST or GET)
  372. $postData = $this->request->getJSON(true);
  373. $seq = $postData['SEQ'];
  374. $refreshTokenKaKao = $postData['KAKAO_REFRESH_TOKEN'];
  375. // 2. 사용자 정보 조회 (구글 토큰 포함)
  376. $user = $this->userModel->find($seq);
  377. if (!$user) {
  378. return $this->response->setJSON(['error' => 'User not found'])->setStatusCode(404);
  379. }
  380. // 3. 카카오 언링크 처리
  381. $tokenUrl = "https://kauth.kakao.com/oauth/token";
  382. $tokenData = [
  383. 'grant_type' => 'refresh_token',
  384. 'client_id' => '1f8376b18a02a00f2e4e5594f9ace6d4',
  385. 'refresh_token' => $refreshTokenKaKao, // DB에서 불러온 값
  386. ];
  387. $client = \Config\Services::curlrequest();
  388. $tokenResponse = $client->post($tokenUrl, [
  389. 'form_params' => $tokenData
  390. ]);
  391. $tokenResult = json_decode($tokenResponse->getBody(), true);
  392. $accessTokenKakao = $tokenResult['access_token'];
  393. // (2) 발급받은 access_token으로 연결 끊기 요청
  394. $userUnlinkUrl = "https://kapi.kakao.com/v1/user/unlink";
  395. $response = $client->post($userUnlinkUrl, [
  396. 'headers' => [
  397. 'Authorization' => 'Bearer ' . $accessTokenKakao,
  398. ]
  399. ]);
  400. // 4. USER_LIST에서 사용자 삭제
  401. $this->userModel->delete($seq);
  402. // 5. 응답 반환
  403. return $this->response->setJSON([
  404. 'result' => 'User account deleted and Google link revoked'
  405. ])->setStatusCode(200);
  406. }
  407. /***********************************
  408. * 내아버 간편로그인 / 가입
  409. **********************************/
  410. //네이버 인증
  411. public function naverLogin()
  412. {
  413. $client_id = 'tPw7dRu1r7yY89O5gN7n';
  414. $redirect_uri = urlencode('https://shopdeli.mycafe24.com/auth/naver'); // ex) https://your.site/naver-callback
  415. $state = bin2hex(random_bytes(8));
  416. session()->set('naver_state', $state);
  417. $naver_auth_url = "https://nid.naver.com/oauth2.0/authorize?"
  418. . "response_type=code"
  419. . "&client_id={$client_id}"
  420. . "&client_icon=https://ndevthumb-phinf.pstatic.net/20250708_43/1751954347202gc3db_JPEG/x49C3CDfcWcI20250708145907.jpeg"
  421. . "&redirect_uri={$redirect_uri}"
  422. . "&state={$state}";
  423. // 네이버 인증/동의화면으로 리디렉트
  424. return redirect()->to($naver_auth_url);
  425. }
  426. public function naver(){
  427. $client_id = 'tPw7dRu1r7yY89O5gN7n';
  428. $client_secret = 'Pgan4lv9l9';
  429. $state = $this->request->getGet('state');
  430. $code = $this->request->getGet('code');
  431. if ($state !== session()->get('naver_state')) {
  432. exit('CSRF 실패');
  433. }
  434. // 토큰 발급
  435. $token_url = "https://nid.naver.com/oauth2.0/token"
  436. . "?grant_type=authorization_code"
  437. . "&client_id={$client_id}"
  438. . "&client_secret={$client_secret}"
  439. . "&code={$code}"
  440. . "&state={$state}";
  441. // cURL로 토큰 요청
  442. $ch = curl_init();
  443. curl_setopt($ch, CURLOPT_URL, $token_url);
  444. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  445. $token_response = curl_exec($ch);
  446. curl_close($ch);
  447. // JSON 파싱
  448. $token_data = json_decode($token_response, true);
  449. $access_token = $token_data['access_token'] ?? null;
  450. // 회원 정보 요청
  451. if ($access_token) {
  452. $headers = [
  453. "Authorization: Bearer {$access_token}"
  454. ];
  455. $ch = curl_init();
  456. curl_setopt($ch, CURLOPT_URL, "https://openapi.naver.com/v1/nid/me");
  457. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  458. curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
  459. $user_info = curl_exec($ch);
  460. curl_close($ch);
  461. $user_info_arr = json_decode($user_info, true);
  462. $user_info_arr['refresh_token'] = $token_data['refresh_token'];
  463. $user_info_arr['access_token'] = $token_data['access_token'];
  464. //DB 커넥션
  465. $db = \Config\Database::connect();
  466. // 1. USER_LIST에서 이메일로 조회
  467. $id = $user_info_arr['response']['id'];
  468. $email = $user_info_arr['response']['email'];
  469. $name = $user_info_arr['response']['name'];
  470. $phone = $user_info_arr['response']['mobile'];
  471. $builder = $db->table('USER_LIST');
  472. $user = $builder->where('EMAIL', $email)->get()->getRowArray();
  473. if($user){
  474. $jwtSecret = env('JWT_SECRET');
  475. $kid = env('JWT_KID');
  476. if (empty($jwtSecret) || empty($kid)) {
  477. return $this->failServerError('환경변수가 누락되었습니다. 관리자에게 문의하세요.');
  478. }
  479. if (!class_exists('\App\Libraries\JwtLib\JWT')) {
  480. return $this->failServerError('JWT 라이브러리를 찾을 수 없습니다.');
  481. }
  482. $issuedAt = time();
  483. $accessExpire = $issuedAt + 60 * 15; // 15분
  484. $refreshExpire = $issuedAt + 60 * 60 * 24 * 14; // 14일
  485. $accessPayload = [
  486. 'iat' => $issuedAt,
  487. 'exp' => $accessExpire,
  488. 'sub' => $id,
  489. 'name' => $name ?? '',
  490. ];
  491. $refreshPayload = [
  492. 'iat' => $issuedAt,
  493. 'exp' => $refreshExpire,
  494. 'sub' => $id,
  495. ];
  496. try {
  497. $accessToken = JWT::encode($accessPayload, $jwtSecret, 'HS256', $kid);
  498. $refreshToken = JWT::encode($refreshPayload, $jwtSecret, 'HS256', $kid);
  499. // (선택) DB의 리프레시 토큰 값 업데이트
  500. $db = \Config\Database::connect();
  501. $builder = $db->table('USER_LIST');
  502. $builder->where('ID', $id)->update([
  503. 'REFRESH_TOKEN' => $refreshToken,
  504. 'NAVER_REFRESH_TOKEN' => $user_info_arr['refresh_token']
  505. ]);
  506. } catch (\Throwable $e) {
  507. return $this->failServerError('JWT 생성 오류: ' . $e->getMessage());
  508. }
  509. unset($user['PASSWORD']); // 혹시 비번이 있다면 노출 방지
  510. $query = http_build_query([
  511. "accessToken" => $accessToken,
  512. "refreshToken" => $refreshToken,
  513. "user" => urlencode(json_encode($user)),
  514. ]);
  515. //return redirect()->to(base_url("auth/popupClose?$query"));
  516. // 개발진행중 풀 URL
  517. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  518. }else{
  519. // 회원이 없으면 회원가입
  520. $type = 1;
  521. // ID는 auto_increment라면 생략
  522. $authData = [
  523. 'ID' => $id,
  524. 'NAME' => $name,
  525. 'PHONE' => $phone,
  526. 'EMAIL' => $email,
  527. 'TYPE' => $type,
  528. 'JOIN' => '1',
  529. 'NAVER_REFRESH_TOKEN' => $user_info_arr['refresh_token'] ?? null
  530. ];
  531. $query = http_build_query([
  532. "user" => urlencode(json_encode($authData)),
  533. ]);
  534. //return redirect()->to(base_url("auth/popupClose?$query"));
  535. // 개발진행중 풀 URL
  536. return redirect()->to(self::FRONTEND_BASE_URL."/auth/popupClose?$query");
  537. }
  538. } else {
  539. // 오류 처리
  540. }
  541. }
  542. public function naverWithdrawal($refreshToken)
  543. {
  544. // 1. 요청에서 사용자 SEQ 추출 (예: POST or GET)
  545. $postData = $this->request->getJSON(true);
  546. $seq = $postData['SEQ'];
  547. $refreshTokenNaver = $postData['NAVER_REFRESH_TOKEN'];
  548. // 2. 사용자 정보 조회 (구글 토큰 포함)
  549. $user = $this->userModel->find($seq);
  550. if (!$user) {
  551. return $this->response->setJSON(['error' => 'User not found'])->setStatusCode(404);
  552. }
  553. $clientId = 'tPw7dRu1r7yY89O5gN7n';
  554. $clientSecret = 'Pgan4lv9l9';
  555. // 1. 리프레시 토큰으로 엑세스 토큰 재발급 요청
  556. $tokenUrl = "https://nid.naver.com/oauth2.0/token";
  557. $tokenParams = [
  558. 'grant_type' => 'refresh_token',
  559. 'client_id' => $clientId,
  560. 'client_secret' => $clientSecret,
  561. 'refresh_token' => $refreshTokenNaver
  562. ];
  563. $ch = curl_init();
  564. curl_setopt($ch, CURLOPT_URL, $tokenUrl);
  565. curl_setopt($ch, CURLOPT_POST, true);
  566. curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($tokenParams));
  567. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  568. $tokenResponse = curl_exec($ch);
  569. $tokenError = curl_error($ch);
  570. curl_close($ch);
  571. if ($tokenError) {
  572. return [
  573. 'success' => false,
  574. 'message' => "토큰 재발급 오류: " . $tokenError,
  575. ];
  576. }
  577. $tokenData = json_decode($tokenResponse, true);
  578. if (empty($tokenData['access_token'])) {
  579. return [
  580. 'success' => false,
  581. 'message' => '엑세스 토큰 재발급 실패: ' . ($tokenData['error_description'] ?? '알 수 없는 오류'),
  582. ];
  583. }
  584. $accessToken = $tokenData['access_token'];
  585. // 2. 엑세스 토큰으로 연결끊기
  586. $withdrawUrl = "https://nid.naver.com/oauth2.0/token";
  587. $withdrawParams = [
  588. 'grant_type' => 'delete',
  589. 'client_id' => $clientId,
  590. 'client_secret' => $clientSecret,
  591. 'access_token' => $accessToken,
  592. 'service_provider' => 'NAVER',
  593. ];
  594. $ch2 = curl_init();
  595. curl_setopt($ch2, CURLOPT_URL, $withdrawUrl);
  596. curl_setopt($ch2, CURLOPT_POST, true);
  597. curl_setopt($ch2, CURLOPT_POSTFIELDS, http_build_query($withdrawParams));
  598. curl_setopt($ch2, CURLOPT_RETURNTRANSFER, true);
  599. $withdrawResponse = curl_exec($ch2);
  600. $withdrawError = curl_error($ch2);
  601. curl_close($ch2);
  602. if ($withdrawError) {
  603. return [
  604. 'success' => false,
  605. 'message' => "연결해제 요청 오류: " . $withdrawError,
  606. ];
  607. }
  608. $withdrawData = json_decode($withdrawResponse, true);
  609. if (isset($withdrawData['result']) && $withdrawData['result'] === 'success') {
  610. return [
  611. 'success' => true,
  612. 'message' => '네이버 연동 해제 완료',
  613. ];
  614. // 4. USER_LIST에서 사용자 삭제
  615. $this->userModel->delete($seq);
  616. // 5. 응답 반환
  617. return $this->response->setJSON([
  618. 'result' => 'User account deleted and Google link revoked'
  619. ])->setStatusCode(200);
  620. } else {
  621. return [
  622. 'success' => false,
  623. 'message' => $withdrawData['error'] ?? '연동 해제 실패',
  624. ];
  625. }
  626. }
  627. /***********************************
  628. * @param $refreshTokenGoogle
  629. * @return void
  630. **********************************/
  631. // 구글 연결 가입정보 연결 해제(회원가입중 페이지 빠져나올경우 리프래시 토큰이 사라짐 고객이 직접 계정에서 해제 필요)
  632. protected function revokeGoogleAccess($refreshTokenGoogle)
  633. {
  634. // 1. 리프레시 토큰으로 엑세스 토큰 발급
  635. $client_id = '373780605211-diojebh7mug45urv9rnqdil6n0b1ogge.apps.googleusercontent.com';
  636. $client_secret = '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';
  637. $refresh_token = $refreshTokenGoogle;
  638. $token_url = 'https://oauth2.googleapis.com/token';
  639. $params = [
  640. 'client_id' => $client_id,
  641. 'client_secret' => $client_secret,
  642. 'refresh_token' => $refresh_token,
  643. 'grant_type' => 'refresh_token',
  644. ];
  645. $ch = curl_init();
  646. curl_setopt($ch, CURLOPT_URL, $token_url);
  647. curl_setopt($ch, CURLOPT_POST, true);
  648. curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($params));
  649. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  650. $response = curl_exec($ch);
  651. curl_close($ch);
  652. $data = json_decode($response, true);
  653. // 2. 새 엑세스 토큰으로 연결 끊기
  654. if (!empty($data['access_token'])) {
  655. $accessToken = $data['access_token'];
  656. $revoke_url = 'https://accounts.google.com/o/oauth2/revoke?token=' . urlencode($accessToken);
  657. $ch = curl_init($revoke_url);
  658. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  659. curl_exec($ch);
  660. curl_close($ch);
  661. } else {
  662. // TODO : 엑세스 토큰 발급 실패 처리 필요
  663. }
  664. }
  665. /***********************************
  666. * 아이디 체크
  667. **********************************/
  668. public function checkId()
  669. {
  670. $postData = $this->request->getJSON(true);
  671. // 필수값 검증
  672. if (empty($id) || empty($type)) {
  673. return $this->response->setJSON([
  674. 'status' => 'fail',
  675. 'message' => '필수 정보가 누락되었습니다.'
  676. ])->setStatusCode(400);
  677. }
  678. $db = \Config\Database::connect();
  679. try {
  680. // type에 따라 다른 테이블에서 ID 중복 체크
  681. if ($type === 'vendor') {
  682. // 벤더 회원가입의 경우 VENDOR_LIST 테이블 체크
  683. $builder = $db->table('VENDOR_LIST');
  684. $existingUser = $builder->where('ID', $id)->get()->getRowArray();
  685. if ($existingUser) {
  686. return $this->response->setJSON([
  687. 'status' => 'fail',
  688. 'message' => '이미 사용 중인 아이디입니다.'
  689. ])->setStatusCode(200);
  690. }
  691. } elseif ($type === 'influencer') {
  692. // 인플루언서(일반회원가입)의 경우 USER_LIST 테이블 체크
  693. $builder2 = $db->table('USER_LIST');
  694. $existingUser = $builder2->where('ID', $id)->get()->getRowArray();
  695. if ($existingUser) {
  696. return $this->response->setJSON([
  697. 'status' => 'fail',
  698. 'message' => '이미 사용 중인 아이디입니다.'
  699. ])->setStatusCode(200);
  700. }
  701. } else {
  702. return $this->response->setJSON([
  703. 'status' => 'fail',
  704. 'message' => '유효하지 않은 회원 유형입니다.'
  705. ])->setStatusCode(400);
  706. }
  707. // ID가 사용 가능한 경우
  708. return $this->response->setJSON([
  709. 'status' => 'success',
  710. 'message' => '사용 가능한 아이디입니다.'
  711. ])->setStatusCode(200);
  712. } catch (\Throwable $e) {
  713. return $this->response->setJSON([
  714. 'status' => 'fail',
  715. 'message' => 'DB 오류: ' . $e->getMessage()
  716. ])->setStatusCode(500);
  717. }
  718. }
  719. }