{"id":905,"date":"2026-05-20T09:55:28","date_gmt":"2026-05-20T01:55:28","guid":{"rendered":"https:\/\/junai.ai\/blog\/nodejs-auth-jwt-19\/"},"modified":"2026-05-20T09:55:28","modified_gmt":"2026-05-20T01:55:28","slug":"nodejs-auth-jwt-19","status":"publish","type":"post","link":"https:\/\/junai.ai\/blog\/nodejs-auth-jwt-19\/","title":{"rendered":"JWT \uc778\uc99d \u2014 \ub85c\uadf8\uc778\u00b7\ud1a0\ud070 \ubc1c\uae09\u00b7\uac80\uc99d"},"content":{"rendered":"\n<!-- WordPress REST API \ubc1c\ud589\uc6a9 HTML (\uc790\ub3d9 \uc0dd\uc131) -->\n<!-- WP-FEATURED-MEDIA-ID: 852 -->\n<div style=\"max-width:800px;margin:0 auto;\">\n<style>\n:root {--color-primary:#059669;--color-accent:#10b981;--color-bg:#fafbfc;--color-bg-card:#fff;--color-text:#1a202c;--color-text-muted:#64748b;--hero-start:#064e3b;--hero-end:#059669;}\n*{box-sizing:border-box;}\n.container{max-width:760px;margin:0 auto;padding:0 24px 80px;}\n.hero{background:linear-gradient(135deg,var(--hero-start) 0%,var(--hero-end) 100%);color:#fff;padding:80px 24px 60px;text-align:center;}\n.hero .eyebrow{display:inline-block;font-size:14px;color:#6ee7b7;font-weight:700;letter-spacing:0.1em;text-transform:uppercase;margin-bottom:14px;}\n.hero h1{font-size:36px;margin:0 0 16px;line-height:1.3;font-weight:800;}\n.hero p{color:#d1fae5;font-size:18px;max-width:640px;margin:0 auto;line-height:1.6;}\n.hero img{width:100%;max-width:640px;height:auto;margin:32px auto 0;border-radius:10px;display:block;}\narticle{padding-top:48px;}\narticle h2{font-size:26px;margin:56px 0 20px;padding-left:14px;border-left:5px solid var(--color-accent);line-height:1.4;}\narticle h3{font-size:19px;margin:32px 0 12px;color:var(--color-primary);}\narticle p{margin:16px 0;}\narticle strong{color:var(--color-primary);font-weight:700;}\narticle code{background:#d1fae5;padding:2px 8px;border-radius:4px;font-family:'SF Mono',Menlo,Consolas,monospace;font-size:14px;color:#065f46;}\n.databox{background:#d1fae5;border-left:4px solid var(--color-accent);padding:16px 20px;margin:24px 0;border-radius:0 8px 8px 0;font-size:15.5px;}\n.databox strong{color:var(--color-primary);}\n.warnbox{background:linear-gradient(135deg,#fef3c7 0%,#fde68a 100%);padding:16px 20px;margin:24px 0;border-radius:8px;font-size:15.5px;}\n.tablewrap{overflow-x:auto;-webkit-overflow-scrolling:touch;margin:22px 0;}\ntable{width:100%;border-collapse:collapse;font-size:15px;background:var(--color-bg-card);}\nth,td{padding:11px 12px;text-align:left;border-bottom:1px solid #e2e8f0;vertical-align:top;}\nth{background:#f1f5f9;font-weight:700;color:#0f172a;}\ntd:first-child,th:first-child{font-weight:700;}\n@media (max-width:560px){.tablewrap table,.tablewrap thead,.tablewrap tbody,.tablewrap tr,.tablewrap th,.tablewrap td{display:block;width:auto;}.tablewrap thead{display:none;}.tablewrap tr{margin:0 0 14px;border:1px solid #e2e8f0;border-radius:10px;overflow:hidden;}.tablewrap td{border:none;border-bottom:1px solid #f1f5f9;padding:9px 14px;}.tablewrap td:first-child{background:#f1f5f9;font-weight:800;font-size:15.5px;}.tablewrap td:last-child{border-bottom:none;}.tablewrap td[data-label]::before{content:attr(data-label) \" \u2014 \";font-weight:700;color:var(--color-primary);}}\n.code-block{background:#0f172a;color:#e2e8f0;padding:16px 20px;border-radius:8px;font-family:'SF Mono',Menlo,Consolas,monospace;font-size:14px;line-height:1.6;margin:20px 0;overflow-x:auto;white-space:pre;}\n.cta{background:linear-gradient(135deg,#059669 0%,#10b981 100%);color:#fff;padding:28px 24px;border-radius:12px;margin:48px 0 0;text-align:center;}\n.cta h3{color:#fff;margin:0 0 8px;font-size:20px;}\n.cta p{color:#d1fae5;margin:0;font-size:15.5px;}\n.footer-nav{margin-top:32px;padding-top:20px;border-top:1px solid #e2e8f0;font-size:14px;color:var(--color-text-muted);}\n.footer-nav a{color:var(--color-primary);text-decoration:none;}\n@media (max-width:480px){.hero h1{font-size:26px;}.hero p{font-size:16px;}article h2{font-size:21px;}article h3{font-size:17px;}body{font-size:16px;}}\n<\/style>\n<section class=\"hero\">\n  <span class=\"eyebrow\">Node.js \uad50\uc7ac \u00b7 19\ud3b8 \u00b7 JWT \uc778\uc99d<\/span>\n  <h1>JWT \uc778\uc99d \u2014 \ub85c\uadf8\uc778\u00b7\ud1a0\ud070 \ubc1c\uae09\u00b7\uac80\uc99d<\/h1>\n  <p>Auth.js \ucc98\ub7fc \ud480\uc2a4\ud0dd \uc194\ub8e8\uc158\uc774 \ubabb \ub2ff\ub294 \uacf3 \u2014 \ubaa8\ubc14\uc77c API\u00b7\uc11c\ub4dc\ud30c\ud2f0 \ubc1c\uae09\uc758 \ud45c\uc900.<\/p>\n  <img decoding=\"async\" src=\"https:\/\/junai.ai\/blog\/wp-content\/uploads\/2026\/05\/hero-5-83.jpg\" alt=\"JWT \ud1a0\ud070 \uad6c\uc870\uc640 \ub85c\uadf8\uc778 \ud750\ub984 \ucee8\uc149 \uc77c\ub7ec\uc2a4\ud2b8\">\n<\/section>\n\n<div class=\"container\">\n<article>\n\n<p>Next.js 19\ud3b8\uc758 Auth.js \uac00 \uac19\uc740 \uc0ac\uc774\ud2b8 \uc548 \uc778\uc99d\uc758 \ub2f5\uc774\ub77c\uba74, <strong>JWT<\/strong>(JSON Web Token) \ub294 \uadf8 \ubc94\uc704 \ubc16 \u2014 \ubaa8\ubc14\uc77c \uc571 API, \uc678\ubd80 \uc11c\ube44\uc2a4 \ubc1c\uae09, \ub9c8\uc774\ud06c\ub85c\uc11c\ube44\uc2a4 \uac04 \ud638\ucd9c\uc5d0\uc11c \ud45c\uc900. Node \uc9c1\uc811 \uad6c\ud604\uc774 \ud45c\uc900\uc774\ub2e4.<\/p>\n\n<p>\uc774\ubc88 \ud3b8\uc740 JWT \uc758 \uc815\uccb4\u00b7\ub85c\uadf8\uc778 \ud750\ub984\u00b7\uc2e4\uc804 \uad6c\ud604\u00b7refresh \ud328\ud134\uae4c\uc9c0. \ubcf4\uc548 \uc0ac\uace0 \uc904\uc774\ub294 5\uac00\uc9c0 \ud568\uc815\ub3c4.<\/p>\n\n<h2>1. JWT \uc758 \uc815\uccb4 \u2014 \uc810 \ub450 \uac1c\ub85c \ub098\ub25c \ubb38\uc790\uc5f4<\/h2>\n\n<p>JWT \ub294 \uadf8\ub0e5 <strong>3 \ubd80\ubd84\uc774 \uc810\uc73c\ub85c \uc5f0\uacb0\ub41c \ubb38\uc790\uc5f4<\/strong>.<\/p>\n\n<div class=\"code-block\">eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiIxMjMiLCJleHAiOjE3MTAwMDAwMDB9.X3sigX\n\u2514\u2500 header \u2500\u2518.\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500 payload \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518.\u2514\u2500signature\u2500\u2518<\/div>\n\n<p>\uac01 \ubd80\ubd84\uc758 \uc815\uccb4:<\/p>\n\n<div class=\"tablewrap\">\n<table>\n<thead><tr><th>\ubd80\ubd84<\/th><th>\ub0b4\uc6a9<\/th><th>\uc778\ucf54\ub529<\/th><\/tr><\/thead>\n<tbody>\n<tr><td>header<\/td><td data-label=\"\ub0b4\uc6a9\"><code>{\"alg\":\"HS256\"}<\/code><\/td><td data-label=\"\uc778\ucf54\ub529\">Base64url<\/td><\/tr>\n<tr><td>payload<\/td><td data-label=\"\ub0b4\uc6a9\"><code>{\"sub\":\"123\",\"exp\":1710000000,...}<\/code><\/td><td data-label=\"\uc778\ucf54\ub529\">Base64url<\/td><\/tr>\n<tr><td>signature<\/td><td data-label=\"\ub0b4\uc6a9\">HMAC(header.payload, SECRET)<\/td><td data-label=\"\uc778\ucf54\ub529\">Base64url<\/td><\/tr>\n<\/tbody>\n<\/table>\n<\/div>\n\n<div class=\"warnbox\">\n<strong>\uac00\uc7a5 \ud070 \uc624\ud574 \u2014 \uc554\ud638\ud654 \uc544\ub2d8<\/strong> \u2014 JWT payload \ub294 <strong>\uadf8\ub0e5 Base64<\/strong>. \ub204\uad6c\ub098 \ub514\ucf54\ub529\ud574 \ub0b4\uc6a9\uc744 \ubcfc \uc218 \uc788\ub2e4 (11\ud3b8 Buffer \ucc55\ud130\uc758 base64 toString \ud55c \uc904). signature \ub294 \uc704\ubcc0\uc870 \ubc29\uc9c0\uc77c \ubfd0, \ube44\ubc00 \ub370\uc774\ud130 \uc228\uae30\ub294 \uac8c \uc544\ub2d8. <strong>\ube44\ubc00\ubc88\ud638\u00b7\ubbfc\uac10 \uc815\ubcf4\ub97c payload \uc5d0 \uc808\ub300 \ub123\uc9c0 \ub9d0 \uac83<\/strong>.\n<\/div>\n\n<h2>2. \uc124\uce58\uc640 \ud5ec\ud37c<\/h2>\n\n<div class=\"code-block\">$ npm install bcrypt jsonwebtoken\n$ npm install -D @types\/bcrypt @types\/jsonwebtoken<\/div>\n\n<div class=\"code-block\">\/\/ auth\/tokens.js\nimport jwt from &#8216;jsonwebtoken&#8217;;\n\nconst SECRET = process.env.JWT_SECRET;     \/\/ 17\ud3b8 \uac80\uc99d\ub41c \uac12\nconst EXP    = process.env.JWT_EXPIRES || &#8217;15m&#8217;;\n\nexport function signAccessToken(userId) {\n  return jwt.sign({ sub: userId }, SECRET, { expiresIn: EXP });\n}\n\nexport function verifyToken(token) {\n  try {\n    return jwt.verify(token, SECRET);   \/\/ \ub9cc\ub8cc\u00b7\uc11c\uba85 \uc790\ub3d9 \uac80\uc99d\n  } catch {\n    return null;\n  }\n}<\/div>\n\n<p><code>sign<\/code> \uc73c\ub85c \ubc1c\uae09, <code>verify<\/code> \ub85c \uac80\uc99d. \uac80\uc99d \uc2e4\ud328(\ub9cc\ub8cc\u00b7\uc798\ubabb\ub41c \uc11c\uba85) \ub294 <code>throw<\/code> \ub77c try\/catch \ub610\ub294 null \ubc18\ud658.<\/p>\n\n<h2>3. \ub85c\uadf8\uc778\u00b7\ubc1c\uae09 \u2014 bcrypt \ud574\uc2dc\uc640 \ud568\uaed8<\/h2>\n\n<p>\ube44\ubc00\ubc88\ud638\ub294 <strong>\uc808\ub300 \ud3c9\ubb38 \uc800\uc7a5 \uae08\uc9c0<\/strong>. <code>bcrypt<\/code> \ub85c \ud574\uc2dc.<\/p>\n\n<div class=\"code-block\">\/\/ \ud68c\uc6d0\uac00\uc785\nimport bcrypt from &#8216;bcrypt&#8217;;\nimport { pool } from &#8216;@\/db&#8217;;\n\nrouter.post(&#8216;\/signup&#8217;, async (req, res) =&gt; {\n  const { email, password } = req.body;\n  const hash = await bcrypt.hash(password, 12);  \/\/ 12 \ub77c\uc6b4\ub4dc \uad8c\uc7a5\n  const { rows: [user] } = await pool.query(\n    &#8216;INSERT INTO users (email, password_hash) VALUES ($1, $2) RETURNING id, email&#8217;,\n    [email, hash]\n  );\n  res.status(201).json(user);\n});\n\n\/\/ \ub85c\uadf8\uc778\nrouter.post(&#8216;\/login&#8217;, async (req, res) =&gt; {\n  const { email, password } = req.body;\n  const { rows: [user] } = await pool.query(\n    &#8216;SELECT id, password_hash FROM users WHERE email = $1&#8217;,\n    [email]\n  );\n  if (!user) return res.status(401).json({ error: &#8216;\uc798\ubabb\ub41c \uc790\uaca9&#8217; });\n\n  const valid = await bcrypt.compare(password, user.password_hash);\n  if (!valid) return res.status(401).json({ error: &#8216;\uc798\ubabb\ub41c \uc790\uaca9&#8217; });\n\n  const token = signAccessToken(user.id);\n  res.json({ token });\n});<\/div>\n\n<div class=\"databox\">\n<strong>\uc65c bcrypt 12 \ub77c\uc6b4\ub4dc?<\/strong> \u2014 \ub77c\uc6b4\ub4dc \uc218\uac00 \ub192\uc744\uc218\ub85d \ubb34\ucc28\ubcc4 \ub300\uc785 \uacf5\uaca9\uc774 \ub290\ub824\uc9c4\ub2e4. 12 \ub294 2026 \uae30\uc900 \ud45c\uc900 (\uc11c\ubc84\uc5d0\uc11c ~250ms). \ub108\ubb34 \ub0ae\uc73c\uba74(8) \ubcf4\uc548 \uc57d\ud568, \ub108\ubb34 \ub192\uc73c\uba74(15) \uc0ac\uc6a9\uc790\uac00 \ub85c\uadf8\uc778 1\ucd08 \uae30\ub2e4\ub9bc. 12 \uac00 \uc2a4\uc704\ud2b8 \uc2a4\ud3ff. \ub77c\uc774\ube0c\ub7ec\ub9ac \uc635\uc158 <strong>argon2id<\/strong> \uac00 \ub354 \uc548\uc804\ud55c \ud6c4\uc18d\uc774\uc9c0\ub9cc bcrypt \uac00 \ud638\ud658\uc131 \uc88b\uc74c.\n<\/div>\n\n<h2>4. \uac80\uc99d \ubbf8\ub4e4\uc6e8\uc5b4 \u2014 Express \uac00\ub4dc<\/h2>\n\n<p>14\ud3b8\uc758 \ubbf8\ub4e4\uc6e8\uc5b4 \ud328\ud134 \uc751\uc6a9.<\/p>\n\n<div class=\"code-block\">\/\/ middleware\/auth.js\nimport { verifyToken } from &#8216;.\/tokens.js&#8217;;\n\nexport function requireAuth(req, res, next) {\n  const auth = req.headers.authorization;\n  if (!auth?.startsWith(&#8216;Bearer &#8216;)) {\n    return res.status(401).json({ error: &#8216;\ud1a0\ud070 \uc5c6\uc74c&#8217; });\n  }\n  const payload = verifyToken(auth.slice(7));\n  if (!payload) {\n    return res.status(401).json({ error: &#8216;\ub9cc\ub8cc \ub610\ub294 \uc798\ubabb\ub41c \ud1a0\ud070&#8217; });\n  }\n  req.user = { id: payload.sub };\n  next();\n}\n\n\/\/ \uc0ac\uc6a9\nrouter.get(&#8216;\/me&#8217;, requireAuth, (req, res) =&gt; {\n  res.json({ id: req.user.id });\n});<\/div>\n\n<p>\ud074\ub77c\uc774\uc5b8\ud2b8\ub294 \ub9e4 \uc694\uccad <code>Authorization: Bearer eyJ...<\/code> \ud5e4\ub354\ub85c \ud1a0\ud070\uc744 \ubcf4\ub0b8\ub2e4. \ud45c\uc900 HTTP \uc778\uc99d \ubc29\uc2dd.<\/p>\n\n<h2>5. Refresh \ud1a0\ud070 \u2014 \uc9e7\uc740 access + \uae34 refresh<\/h2>\n\n<p>\ud55c \ud1a0\ud070\ub9cc \uc4f0\uba74 \u2014 \uae38\uac8c \ub450\uba74 \ud0c8\ucde8 \uc704\ud5d8, \uc9e7\uac8c \ub450\uba74 \uc0ac\uc6a9\uc790\uac00 \ub9e4 \uc2dc\uac04 \uc7ac\ub85c\uadf8\uc778. \ub2f5\uc774 <strong>2 \ud1a0\ud070<\/strong>.<\/p>\n\n<div class=\"tablewrap\">\n<table>\n<thead><tr><th>\ud1a0\ud070<\/th><th>\uc218\uba85<\/th><th>\uc800\uc7a5 \uc704\uce58<\/th><th>\uc6a9\ub3c4<\/th><\/tr><\/thead>\n<tbody>\n<tr><td>Access<\/td><td data-label=\"\uc218\uba85\">15\ubd84<\/td><td data-label=\"\uc704\uce58\">\uba54\ubaa8\ub9ac\u00b7localStorage<\/td><td data-label=\"\uc6a9\ub3c4\">\ub9e4 API \uc694\uccad<\/td><\/tr>\n<tr><td>Refresh<\/td><td data-label=\"\uc218\uba85\">14\uc77c<\/td><td data-label=\"\uc704\uce58\"><strong>httpOnly \ucfe0\ud0a4<\/strong><\/td><td data-label=\"\uc6a9\ub3c4\">access \ub9cc\ub8cc \uc2dc \uac31\uc2e0<\/td><\/tr>\n<\/tbody>\n<\/table>\n<\/div>\n\n<p>access \uac00 \ub9cc\ub8cc\ub418\uba74 \ud074\ub77c\uc774\uc5b8\ud2b8\uac00 <code>POST \/auth\/refresh<\/code> \ud638\ucd9c \u2192 \uc11c\ubc84\uac00 refresh \ud1a0\ud070(\ucfe0\ud0a4) \uac80\uc99d \u2192 \uc0c8 access \ubc1c\uae09. \ud0c8\ucde8\ub3fc\ub3c4 refresh \ub294 httpOnly \ub77c JS \uac00 \ubabb \uc77d\uc5b4 XSS \ub85c \ubabb \ube7c\uac04\ub2e4.<\/p>\n\n<div class=\"code-block\">\/\/ \ub85c\uadf8\uc778 \uc2dc \ub458 \ub2e4 \ubc1c\uae09\nconst access  = signAccessToken(user.id);      \/\/ 15m\nconst refresh = signRefreshToken(user.id);     \/\/ 14d\n\nres.cookie(&#8216;refreshToken&#8217;, refresh, {\n  httpOnly: true,\n  secure: true,\n  sameSite: &#8216;strict&#8217;,\n  maxAge: 14 * 24 * 60 * 60 * 1000,\n});\nres.json({ access });   \/\/ \ubcf8\ubb38\uc5d4 access \ub9cc<\/div>\n\n<div class=\"warnbox\">\n<strong>\ud754\ud55c \ud568\uc815 5\uac00\uc9c0<\/strong> \u2014 \u2460 \ube44\ubc00\ubc88\ud638 \ud3c9\ubb38 \uc800\uc7a5(\ubc94\uc8c4) \u2461 payload \uc5d0 \ube44\ubc00 \ub123\uc74c(\ub178\ucd9c\ub428) \u2462 refresh \ud1a0\ud070 localStorage \uc800\uc7a5(XSS \ucde8\uc57d) \u2463 sameSite \uc548 \uc124\uc815(CSRF) \u2464 HS256 secret \uc9e7\uc74c(32\uc790+ \ud544\uc218). \ud55c \uc904\ub85c \uc0ac\uace0 \uac00\ub2a5. \ucc98\uc74c\ubd80\ud130 \uc815\uc11d\uc73c\ub85c.\n<\/div>\n\n<h3>\uc694\uc57d \u2014 19\ud3b8 \uc88c\ud45c<\/h3>\n\n<p>\uc5ec\uae30\uae4c\uc9c0 \uc815\ub9ac. JWT \ub294 3\ubd80\ubd84 Base64 \ubb38\uc790\uc5f4 \u2014 payload \ub294 \ub204\uad6c\ub098 \ub514\ucf54\ub529 \uac00\ub2a5, \ube44\ubc00 X. \ube44\ubc00\ubc88\ud638\ub294 <strong>bcrypt 12 \ub77c\uc6b4\ub4dc<\/strong>\ub85c \ud574\uc2dc. \ub85c\uadf8\uc778 \uc2dc access(15\ubd84) + refresh(14\uc77c\u00b7httpOnly \ucfe0\ud0a4) \ub450 \ud1a0\ud070 \ubc1c\uae09. \ubbf8\ub4e4\uc6e8\uc5b4\ub85c <code>Authorization: Bearer<\/code> \uac80\uc99d. 5\uac00\uc9c0 \ud568\uc815\ub9cc \ud53c\ud558\uba74 \ud45c\uc900 \uc778\uc99d \uc2dc\uc2a4\ud15c \uc644\uc131. \ub2e4\uc74c \ud3b8\uc5d0\uc11c <strong>\ud30c\uc77c \uc5c5\ub85c\ub4dc<\/strong>\ub85c \uba40\ud2f0\ud30c\ud2b8 \ud3fc.<\/p>\n\n<div class=\"cta\">\n<h3>\ub2e4\uc74c \ud3b8 \uc608\uace0 \u2014 \ud30c\uc77c \uc5c5\ub85c\ub4dc<\/h3>\n<p>multer \ub85c \uc774\ubbf8\uc9c0\u00b7\ud30c\uc77c \ubc1b\uae30. 20\ud3b8.<\/p>\n<\/div>\n\n<div class=\"footer-nav\">\n\uc2dc\ub9ac\uc988 \u00b7 <a href=\"https:\/\/junai.ai\/blog\/category\/nodejs\/\">\uc27d\uac8c \ubc30\uc6b0\ub294 Node.js<\/a> \u00b7 \uc774\uc804: <a href=\"https:\/\/junai.ai\/blog\/nodejs-db-pg-18\/\">Ch.18 PostgreSQL<\/a>\n<\/div>\n\n<\/article>\n<\/div>\n<\/div>\n","protected":false},"excerpt":{"rendered":"<p>Node \uc5d0\uc11c JWT \uc778\uc99d \uad6c\ud604 \u2014 bcrypt \ube44\ubc00\ubc88\ud638 \ud574\uc2dc\u00b7jsonwebtoken \ud1a0\ud070 \ubc1c\uae09\u00b7\uac80\uc99d\u00b7refresh \ud328\ud134\u00b7\ubbf8\ub4e4\uc6e8\uc5b4 \uac00\ub4dc. \uad50\uc7ac 19\ud3b8.<\/p>\n","protected":false},"author":1,"featured_media":852,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[24],"tags":[],"class_list":["post-905","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-nodejs"],"aioseo_notices":[],"_links":{"self":[{"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/posts\/905","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/comments?post=905"}],"version-history":[{"count":0,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/posts\/905\/revisions"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/media\/852"}],"wp:attachment":[{"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/media?parent=905"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/categories?post=905"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/junai.ai\/blog\/wp-json\/wp\/v2\/tags?post=905"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}