{"id":2076,"date":"2015-07-16T17:00:00","date_gmt":"2015-07-16T17:00:00","guid":{"rendered":"https:\/\/www.couchbase.com\/blog\/?p=2076"},"modified":"2025-06-13T23:47:49","modified_gmt":"2025-06-14T06:47:49","slug":"making-a-game-api-server-using-nodejs-revisited","status":"publish","type":"post","link":"https:\/\/www.couchbase.com\/blog\/ko\/making-a-game-api-server-using-nodejs-revisited\/","title":{"rendered":"Node.js\ub97c \uc0ac\uc6a9\ud558\uc5ec \uac8c\uc784 API \uc11c\ubc84 \ub9cc\ub4e4\uae30: \uc7ac\uac80\ud1a0"},"content":{"rendered":"<p>\uba87 \ub144 \uc804, \ube0c\ub81b \ub85c\uc2a8\uc740 <a href=\"https:\/\/www.couchbase.com\/blog\/ko\/game-servers-and-couchbase-nodejs-part-1\/\">\ud6cc\ub96d\ud55c \ube14\ub85c\uadf8 \uc2dc\ub9ac\uc988<\/a> \uc5d0\uc11c \uac8c\uc784 \uc11c\ubc84 \ud504\ub808\uc784\uc6cc\ud06c \uac1c\ubc1c\uc744 \uc704\ud574 Couchbase Server\uc640 Node.js\ub97c \uc0ac\uc6a9\ud558\ub294 \ubc29\ubc95\uc5d0 \ub300\ud574 \uc124\uba85\ud588\uc2b5\ub2c8\ub2e4. \uadf8 \uc774\ud6c4\ub85c Couchbase\uc6a9 Node.js SDK\ub294 \ubc84\uc804 1.x\uc5d0\uc11c 2.x\ub85c \ud06c\uac8c \uc131\uc7a5\ud588\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\uc774 \uae00\uc5d0\uc11c\ub294 \uc6d0\ub798 \uc138 \uac1c\uc758 \uac8c\uc2dc\ubb3c\uc744 \ub2e4\uc2dc \uc0b4\ud3b4\ubcf4\uace0 \ucd5c\uc2e0 Node.js \ubc0f Express \ud504\ub808\uc784\uc6cc\ud06c \ud45c\uc900\uacfc \ucd5c\uc2e0 Node.js SDK \ubc84\uc804\uc758 Couchbase\uc5d0 \ub9de\uac8c \ubcc0\uacbd\ud574 \ubcf4\ub824\uace0 \ud569\ub2c8\ub2e4.<\/p>\n<h2>\uc804\uc81c \uc870\uac74<\/h2>\n<ul>\n<li>Node.js 0.12<\/li>\n<li>\uce74\uc6b0\uce58\ubca0\uc774\uc2a4 \uc11c\ubc84 4.0<\/li>\n<\/ul>\n<h2>\ud504\ub85c\uc81d\ud2b8 \uc900\ube44\ud558\uae30<\/h2>\n<p>Mac, Windows \ub610\ub294 Linux\ub97c \uc0ac\uc6a9\ud558\ub4e0 \uc0c8 Express \ud504\ub808\uc784\uc6cc\ud06c \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc744 \ub9cc\ub4e4\uba74 \uc774\ub4e4 \uac04\uc5d0 \uc77c\uad00\uc131\uc774 \uc720\uc9c0\ub418\uc5b4\uc57c \ud569\ub2c8\ub2e4. \ub77c\ub294 \uc0c8 \ub514\ub809\ud130\ub9ac\ub97c \ub9cc\ub4ed\ub2c8\ub2e4. <strong>gameapi-nodejs<\/strong> \ub370\uc2a4\ud06c\ud1b1\uc758 \ud130\ubbf8\ub110(Mac\/Linux) \ub610\ub294 \uba85\ub839 \ud504\ub86c\ud504\ud2b8(Windows)\uc5d0\uc11c \ub2e4\uc74c \uba85\ub839\uc744 \uc2e4\ud589\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-bash\">\r\nnpm \ucd08\uae30\ud654\r\n<\/code><\/pre>\n<p>\ubb3b\ub294 \uc9c8\ubb38\uc5d0 \ucd5c\uc120\uc744 \ub2e4\ud574 \ub2f5\ud558\uc138\uc694. \ubb3c\ub860 \uba85\ub839 \ud504\ub86c\ud504\ud2b8 \ub610\ub294 \ud130\ubbf8\ub110\uc5d0\uc11c \ud504\ub85c\uc81d\ud2b8 \ub514\ub809\ud130\ub9ac\uac00 \ud604\uc7ac \ub514\ub809\ud130\ub9ac\uc5ec\uc57c \uc131\uacf5\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uba85\ub839 \ub300\uc2e0 \uc774 \ud30c\uc77c\uc744 \uc218\ub3d9\uc73c\ub85c \ub9cc\ub4e4\uc5b4 \ucc44\uc6b8 \uc218\ub3c4 \uc788\uc2b5\ub2c8\ub2e4. \ud504\ub85c\uc81d\ud2b8 \ub514\ub809\ud130\ub9ac\uc5d0 package.json\uc774\ub77c\ub294 \uc0c8 \ud30c\uc77c\uc744 \ub9cc\ub4e4\uace0 \ub2e4\uc74c\uc744 \ucc44\uc6c1\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n  \"name\": \"gameapi-nodejs\",\r\n  \"version\": \"1.0.0\",\r\n  \"description\": \"An example of using the Node.js SDK for Couchbase to create a simple game server\",\r\n  \"author\": \"Couchbase, Inc.\",\r\n  \"license\": \"MIT\"\r\n}\r\n<\/code><\/pre>\n<p>\uc544\uc9c1 \ub05d\ub098\uc9c0 \uc54a\uc558\uc2b5\ub2c8\ub2e4. \uc774 \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc744 \uacc4\ud68d\ud558\uae30 \uc804\uc5d0 \ud504\ub85c\uc81d\ud2b8 \uc885\uc18d\uc131\uc744 \uc124\uce58\ud574\uc57c \ud569\ub2c8\ub2e4. \uba85\ub839 \ud504\ub86c\ud504\ud2b8 \ub610\ub294 \ud130\ubbf8\ub110\uc5d0\uc11c \ub2e4\uc74c \uba85\ub839\uc744 \uc2e4\ud589\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-bash\">\r\nnpm install body-parser couchbase express node-forge uuid --save\r\n<\/code><\/pre>\n<p>\uc774\ub807\uac8c \ud558\uba74 Express \ud504\ub808\uc784\uc6cc\ud06c, Couchbase Node.js SDK, \ube44\ubc00\ubc88\ud638 \ud574\uc2f1\uc744 \uc704\ud55c Forge, \uace0\uc720 \uac12 \uc0dd\uc131\uc744 \uc704\ud55c UUID, URL \uc778\ucf54\ub529 \ubc0f JSON POST \ub370\uc774\ud130\ub97c \ucc98\ub9ac\ud558\uae30 \uc704\ud55c \ubcf8\ubb38 \ud30c\uc11c \ubbf8\ub4e4\uc6e8\uc5b4\uac00 \uc124\uce58\ub429\ub2c8\ub2e4.<\/p>\n<h2>\ub370\uc774\ud130\ubca0\uc774\uc2a4 \uc900\ube44\ud558\uae30<\/h2>\n<p>\ucf54\ub529\uc744 \uc2dc\uc791\ud558\uae30 \uc804\uc5d0 \ub2e4\uc74c\uacfc \uac19\uc740 \ubc84\ud0b7\uc774 \uc788\ub294 Couchbase Server\ub97c \uc124\uce58\ud574\uc57c \ud569\ub2c8\ub2e4. <strong>\uac8c\uc784 \uc0d8\ud50c<\/strong> \uc0dd\uc131\ub418\uc5c8\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\uc774 \ud504\ub85c\uc81d\ud2b8\ub294 Couchbase 4.0\uc758 \uc8fc\uc694 \uae30\ub2a5\uc744 \uc0ac\uc6a9\ud560 \uc608\uc815\uc774\ubbc0\ub85c \uae30\ubcf8 \uc778\ub371\uc2a4\uac00 \uc0dd\uc131\ub418\ub3c4\ub85d \ubc84\ud0b7\uc744 \ucd94\uac00\ub85c \uad6c\uc131\ud574\uc57c \ud569\ub2c8\ub2e4. Linux, Mac \ub610\ub294 Windows \uc2dc\uc2a4\ud15c\uc744 \uc0ac\uc6a9\ud558\ub294 \uacbd\uc6b0, \uc774 \uc791\uc5c5\uc740 Couchbase \ucffc\ub9ac(CBQ) \ud074\ub77c\uc774\uc5b8\ud2b8\ub97c \ud1b5\ud574 \uc27d\uac8c \uc218\ud589\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<h3>Mac<\/h3>\n<p>Mac\uc5d0\uc11c CBQ\ub97c \uc5f4\ub824\uba74 \ud130\ubbf8\ub110\uc5d0\uc11c \ub2e4\uc74c\uc744 \uc2e4\ud589\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-bash\">\r\n.\/Applications\/Couchbase Server.app\/Contents\/Resources\/couchbase-core\/bin\/cbq\r\n<\/code><\/pre>\n<h3>Windows<\/h3>\n<p>Windows\uc5d0\uc11c CBQ\ub97c \uc5f4\ub824\uba74 \uba85\ub839 \ud504\ub86c\ud504\ud2b8\uc5d0\uc11c \ub2e4\uc74c\uc744 \uc2e4\ud589\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-bash\">\r\nC:\/\ud504\ub85c\uadf8\ub7a8 \ud30c\uc77c\/\uce74\uc6b0\uce58\ubca0\uc774\uc2a4\/\uc11c\ubc84\/bin\/cbq.exe\r\n<\/code><\/pre>\n<h3>\uae30\ubcf8 \uc0c9\uc778 \ub9cc\ub4e4\uae30<\/h3>\n<p>CBQ\ub97c \uc5f0 \uc0c1\ud0dc\uc5d0\uc11c \ub2e4\uc74c\uc744 \uc2e4\ud589\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-sql\">\r\nCREATE PRIMARY INDEX ON `gaming-sample` using GSI;\r\n<\/code><\/pre>\n<p>\uc774\uc81c \ubc84\ud0b7\uc744 \ub098\uba38\uc9c0 \ud504\ub85c\uc81d\ud2b8\uc5d0 \uc0ac\uc6a9\ud560 \uc900\ube44\uac00 \ub418\uc5c8\uc2b5\ub2c8\ub2e4!<\/p>\n<h2>\ud504\ub85c\uc81d\ud2b8 \uad6c\uc870<\/h2>\n<p>\ud504\ub85c\uc81d\ud2b8\ub294 \ub2e4\uc74c\uacfc \uac19\uc774 \uad6c\uc131\ub420 \uc608\uc815\uc785\ub2c8\ub2e4:<\/p>\n<table style=\"width: 500px\">\n<thead style=\"text-align: left\">\n<tr>\n<th style=\"width: 125px\">\ud56d\ubaa9<\/th>\n<th style=\"width: 125px\">\ubd80\ubaa8<\/th>\n<th style=\"width: 250px\">\uc124\uba85<\/th>\n<\/tr>\n<\/thead>\n<tbody>\n<tr>\n<td>\ubaa8\ub378<\/td>\n<td><\/td>\n<td>\ubaa8\ub4e0 \ub370\uc774\ud130\ubca0\uc774\uc2a4 \ud074\ub798\uc2a4 \ud30c\uc77c\uc740 \uc5ec\uae30\ub85c \uc774\ub3d9\ud569\ub2c8\ub2e4.<\/td>\n<\/tr>\n<tr>\n<td>\uacbd\ub85c<\/td>\n<td><\/td>\n<td>\ubaa8\ub4e0 API \uc5d4\ub4dc\ud3ec\uc778\ud2b8 \uc815\uc758\ub294 \uc5ec\uae30\ub85c \uc774\ub3d9\ud569\ub2c8\ub2e4.<\/td>\n<\/tr>\n<tr>\n<td>accountmodel.js<\/td>\n<td>\ubaa8\ub378<\/td>\n<td>\uacc4\uc815 \uc815\ubcf4 \uc0dd\uc131 \ubc0f \uac80\uc0c9<\/td>\n<\/tr>\n<tr>\n<td>\uc138\uc158\ubaa8\ub378.js<\/td>\n<td>\ubaa8\ub378<\/td>\n<td>\uc0ac\uc6a9\uc790 \uc778\uc99d \ubc0f \uc138\uc158 \uc815\ubcf4 \uc720\uc9c0 \uad00\ub9ac\ud558\uae30<\/td>\n<\/tr>\n<tr>\n<td>statemodel.js<\/td>\n<td>\ubaa8\ub378<\/td>\n<td>\uac8c\uc784 \uc0c1\ud0dc \uc815\ubcf4 \uc0dd\uc131, \uc5c5\ub370\uc774\ud2b8 \ubc0f \uac80\uc0c9<\/td>\n<\/tr>\n<tr>\n<td>routes.js<\/td>\n<td>\uacbd\ub85c<\/td>\n<td>GET, POST, PUT\uc758 \ubaa8\ub4e0 \uc5d4\ub4dc\ud3ec\uc778\ud2b8\uac00 \uc5ec\uae30\uc5d0 \uc788\uc2b5\ub2c8\ub2e4.<\/td>\n<\/tr>\n<tr>\n<td>app.js<\/td>\n<td><\/td>\n<td>\uc11c\ubc84 \uc124\uc815 \uc815\ubcf4<\/td>\n<\/tr>\n<tr>\n<td>config.json<\/td>\n<td><\/td>\n<td>\uc815\uc801 \ubcc0\uc218<\/td>\n<\/tr>\n<tr>\n<td>package.json<\/td>\n<td><\/td>\n<td>\uc885\uc18d\uc131 \uc815\ubcf4<\/td>\n<\/tr>\n<\/tbody>\n<\/table>\n<h2>\uae30\ubcf8 \uc0ac\ud56d<\/h2>\n<p>Node.js \uac8c\uc784 \uc11c\ubc84 \ub85c\uc9c1\uc5d0 \ub300\ud574 \uc790\uc138\ud788 \uc54c\uc544\ubcf4\uae30 \uc804\uc5d0 \uae30\ubcf8 Express \ud504\ub808\uc784\uc6cc\ud06c \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc744 \uad6c\uc131\ud558\ub294 \uac83\uc774 \uac00\uc7a5 \uc88b\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\ud504\ub85c\uc81d\ud2b8 \ub8e8\ud2b8\uc5d0\uc11c \ub2e4\uc74c\uacfc \uac19\uc740 \ud30c\uc77c\uc744 \ub9cc\ub4e4\uace0 \uc5fd\ub2c8\ub2e4. <strong>config.json<\/strong> \uc5d0 Couchbase \uc5f0\uacb0 \uc815\ubcf4\uc640 \uac19\uc740 \uc815\uc801 \uc815\ubcf4\ub97c \ubcf4\uad00\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc5f4\ub9ac\uba74 \ub2e4\uc74c\uc744 \ud3ec\ud568\ud558\uc138\uc694:<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n    \"couchbase\": {\r\n        \"server\": \"127.0.0.1:8091\",\r\n        \"bucket\": \"gaming-sample\"\r\n    }\r\n}\r\n<\/code><\/pre>\n<p>\ud504\ub85c\uc81d\ud2b8 \ub8e8\ud2b8\uc5d0\uc11c \ub2e4\uc74c\uacfc \uac19\uc740 \ud30c\uc77c\uc744 \ub9cc\ub4e4\uace0 \uc5fd\ub2c8\ub2e4. <strong>app.js<\/strong> \ud30c\uc77c\uc5d0\ub294 Node.js \uc11c\ubc84 \uc2e4\ud589\uc5d0 \ub300\ud55c \ubaa8\ub4e0 \uae30\ubcf8 \uc815\ubcf4\uac00 \ub4e4\uc5b4 \uc788\uc2b5\ub2c8\ub2e4. \ud30c\uc77c\uc744 \uc5f4\uba74 \ub2e4\uc74c\uc744 \ud3ec\ud568\ud558\uc138\uc694:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar express = require(\"express\");\r\nvar bodyParser = require(\"body-parser\");\r\nvar couchbase = require(\"couchbase\");\r\nvar config = require(\".\/config\");\r\nvar app = express();\r\n\r\napp.use(bodyParser.json());\r\napp.use(bodyParser.urlencoded({ extended: true }));\r\n\r\n\/\/ Global declaration of the Couchbase server and bucket to be used\r\nmodule.exports.bucket = (new couchbase.Cluster(config.couchbase.server)).openBucket(config.couchbase.bucket);\r\n\r\n\/\/ All endpoints to be used in this application\r\nvar routes = require(\".\/routes\/routes.js\")(app);\r\n\r\nvar server = app.listen(3000, function () {\r\n    console.log(\"Listening on port %s...\", server.address().port);\r\n});\r\n<\/code><\/pre>\n<p>\uc5ec\uae30\uc5d0 \ud45c\uc2dc\ub418\ub294 \ub0b4\uc6a9\uc744 \ubd84\uc11d\ud574 \ubcf4\uaca0\uc2b5\ub2c8\ub2e4. \ucc98\uc74c \uba87 \uc904\uc740 \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc5d0 \uc885\uc18d\uc131\uc744 \ud3ec\ud568\ud558\ub294 \uac83\uc785\ub2c8\ub2e4. \ud2b9\ubcc4\ud55c \uac83\uc740 \uc5c6\uc2b5\ub2c8\ub2e4. \uc911\uc694\ud55c \uac83\uc740 \ub2e4\uc74c\uacfc \uac19\uc2b5\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\napp.use(bodyParser.json());\r\napp.use(bodyParser.urlencoded({ extended: true }));\r\n<\/code><\/pre>\n<p>\uc989, \uc694\uccad \ubcf8\ubb38\uc5d0\uc11c JSON \ub370\uc774\ud130\uc640 URL\ub85c \uc778\ucf54\ub529\ub41c \ub370\uc774\ud130\ub97c \ud30c\uc2f1\ud558\uac8c \ub429\ub2c8\ub2e4. \ud2b9\ud788 POST \ubc0f PUT \uc694\uccad\uc774 \uadf8\ub807\uc2b5\ub2c8\ub2e4. \ub2e4\uc74c\uc73c\ub85c \ud560 \uc77c\uc740 \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc5d0\uc11c Couchbase \ud074\ub7ec\uc2a4\ud130\ub97c \ucd08\uae30\ud654\ud558\uace0 \uc560\ud50c\ub9ac\ucf00\uc774\uc158 \ub0b4\uc5d0\uc11c \uc0ac\uc6a9\ud560 \ub2e8\uc77c \ubc84\ud0b7\uc744 \uc5ec\ub294 \uac83\uc785\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nmodule.exports.bucket = (new couchbase.Cluster(config.couchbase.server)).openBucket(config.couchbase.bucket);\r\n<\/code><\/pre>\n<p>\ub2e4\uc74c\uc744 \ud3ec\ud568\ud569\ub2c8\ub2e4. <strong>module.exports.bucket<\/strong> \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc744 \ud1b5\ud574 \uc0ac\uc6a9\ud55c\ub2e4\ub294 \uc758\ubbf8\uc785\ub2c8\ub2e4. \uc774\uc81c \ub2e4\ub978 \uc790\ubc14\uc2a4\ud06c\ub9bd\ud2b8 \ud30c\uc77c\uc5d0\uc11c \ubc84\ud0b7\uc5d0 \uc561\uc138\uc2a4\ud558\ub824\uba74 \uadf8\ub0e5 \ud558\uba74 \ub429\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar bucket = require(\"relative\/path\/to\/app\").bucket;\r\n<\/code><\/pre>\n<p>\ub2e4\uc74c\uc5d0\ub294 \uace7 \uc0dd\uc131\ub420 \uc608\uc815\uc778 <strong>routes\/routes.js<\/strong> \ud30c\uc77c\uc744 \uc804\ub2ec\ud558\uace0 <strong>\uc571\u00a0<\/strong>\ubcc0\uc218\ub97c \uc778\uc218 \uc911 \ud558\ub098\ub85c \uc0ac\uc6a9\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc774\uac83\uc774 \ubb34\uc5c7\uc744 \ud558\ub294\uc9c0\ub294 \uace7 \ubd84\uba85\ud574\uc9c8 \uac83\uc785\ub2c8\ub2e4.<\/p>\n<p>\ub9c8\uc9c0\ub9c9\uc73c\ub85c <strong>app.listen<\/strong> \ud3ec\ud2b8 3000\uc5d0\uc11c \uc694\uccad\uc744 \uc218\uc2e0 \ub300\uae30\ud558\ub3c4\ub85d Node.js\uc5d0 \uc9c0\uc2dc\ud558\uace0 \uc788\uc2b5\ub2c8\ub2e4. \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc740 \uac00\uc7a5 \uae30\ubcf8\uc801\uc778 \uc0c1\ud0dc\uc5d0\uc11c \uac70\uc758 \uc0ac\uc6a9\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc0dd\uc131 \ubc0f \uc5f4\uae30 <strong>routes\/routes.js<\/strong> \ub97c \ud074\ub9ad\ud558\uace0 \ub2e4\uc74c \uc904\uc744 \ucd94\uac00\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar appRouter = function(app) {\r\n\r\n    app.get(\"\/\", function(req, res) {\r\n        res.status(403).send(\"Not a valid endpoint\");\r\n    });\r\n\r\n}\r\n\r\nmodule.exports = appRouter;\r\n<\/code><\/pre>\n<p>\uc774\uc81c \ub2e4\uc74c\uc744 \uc2e4\ud589\ud558\uc5ec \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc744 \uc2e4\ud589\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. <strong>\ub178\ub4dc app.js<\/strong> \uba85\ub839 \ud504\ub86c\ud504\ud2b8 \ub610\ub294 \ud130\ubbf8\ub110\uc5d0\uc11c \uc785\ub825\ud569\ub2c8\ub2e4. \ucc29\ub959 \uc704\uce58\u00a0<strong>https:\/\/localhost:3000<\/strong> \ub97c \uc2e4\ud589\ud558\uba74 \"\uc720\ud6a8\ud55c \uc5d4\ub4dc\ud3ec\uc778\ud2b8\uac00 \uc544\ub2d9\ub2c8\ub2e4\"\ub77c\ub294 \uba54\uc2dc\uc9c0\uac00 \ud45c\uc2dc\ub429\ub2c8\ub2e4.<\/p>\n<h2>API \ub370\uc774\ud130 \ubaa8\ub378<\/h2>\n<p>\uc911\uc694\ud55c \ucf54\ub4dc\ub97c \uc0b4\ud3b4\ubcf4\uae30 \uc804\uc5d0 Couchbase\uc5d0\uc11c \ub370\uc774\ud130\uac00 \uc5b4\ub5bb\uac8c \ubcf4\uc774\ub294\uc9c0 \uc544\ub294 \uac83\uc774 \uac00\uc7a5 \uc88b\uc2b5\ub2c8\ub2e4. \ud55c \uc0ac\uc6a9\uc790\uc5d0\uac8c\ub294 \ub2e4\uc74c\uacfc \uac19\uc774 \ubcf4\uc774\ub294 4\uac1c\uc758 \ubb38\uc11c\uac00 \uc788\uc2b5\ub2c8\ub2e4:<\/p>\n<h3>\uc0ac\uc6a9\uc790 \ubb38\uc11c<\/h3>\n<p>\uc0ac\uc6a9\uc790 \ubb38\uc11c\uc5d0\ub294 \uc0ac\uc6a9\uc790\uc5d0 \ub300\ud55c \ubaa8\ub4e0 \uc815\ubcf4\uac00 \uc800\uc7a5\ub429\ub2c8\ub2e4. \uc774 \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc758 \uacbd\uc6b0 \ud574\ub2f9 \uc815\ubcf4\ub294 \uc774\ub984, \uc0ac\uc6a9\uc790 \uc774\ub984, \ube44\ubc00\ubc88\ud638\uc785\ub2c8\ub2e4.<br \/>\n\uc774 \ubb38\uc11c\uc758 \uc774\ub984 \uc55e\uc5d0 \ub2e4\uc74c\uacfc \uac19\uc740 \uc811\ub450\uc0ac\uac00 \ubd99\uc2b5\ub2c8\ub2e4. <strong>user::<\/strong> \uc5d0 \uace0\uc720\ud55c UID \uac12\uc744 \ucd94\uac00\ud569\ub2c8\ub2e4. \uc774 \ubb38\uc11c \uc774\ub984 \uc9c0\uc815 \uc804\ub7b5\uc740 \ubcf5\ud569 \ud0a4\ub77c\ub294 \uac83\uc744 \uc0ac\uc6a9\ud569\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n    \"type\": \"user\",\r\n    \"uid\", \"\",\r\n    \"name\": \"\",\r\n    \"username\": \"\",\r\n    \"password\": \"\"\r\n}\r\n<\/code><\/pre>\n<h3>\uc0ac\uc6a9\uc790 \uc774\ub984 \ubb38\uc11c<\/h3>\n<p>\uc0ac\uc6a9\uc790\uba85 \ubb38\uc11c\uc5d0\ub294 \uc0ac\uc6a9\uc790 \ubb38\uc11c\uc5d0 \uc788\ub294 uid \uac12\ub9cc \uc800\uc7a5\ub429\ub2c8\ub2e4. \uc0ac\uc6a9\uc790\uba85 \ubb38\uc11c\uc758 \ubaa9\uc801\uc740 \ub85c\uadf8\uc778 \ubc29\ubc95 \ubb38\uc11c\ucc98\ub7fc \uc0dd\uac01\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc608\ub97c \ub4e4\uc5b4 \uc0ac\uc6a9\uc790\uac00 \uc0ac\uc6a9\uc790 \uc774\ub984\uacfc \ube44\ubc00\ubc88\ud638\ub97c \uc785\ub825\ud558\ub294 \uac04\ub2e8\ud55c \ub85c\uadf8\uc778\uc744 \ub098\ud0c0\ub0bc \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc774 \ubb38\uc11c\uc5d0\ub294 \uc5f0\uacb0 UID\uac00 \ud3ec\ud568\ub418\uc5b4 \uc788\uae30 \ub54c\ubb38\uc5d0 \uc0ac\uc6a9\uc790 \ubb38\uc11c\uc5d0 \uc5f0\uacb0\ub420 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc0ac\uc6a9\uc790 \uc774\ub984 \ubb38\uc11c \uc55e\uc5d0\ub294\u00a0<strong>\uc0ac\uc6a9\uc790 \uc774\ub984::<\/strong> \uc744 \uc785\ub825\ud558\uba74 \uc2e4\uc81c \uc0ac\uc6a9\uc790 \uc544\uc774\ub514\uac00 \ucd94\uac00\ub429\ub2c8\ub2e4. \ud398\uc774\uc2a4\ubd81\uc774\ub098 \ud2b8\uc704\ud130\ub97c \ub85c\uadf8\uc778 \uc218\ub2e8\uc73c\ub85c \uc0ac\uc6a9\ud558\uace0 \uc544\uc774\ub514 \ud544\ub4dc\ub97c \ud1b5\ud574 \uc5f0\uacb0\ud558\ub294 \uacbd\uc6b0\uc5d0\ub3c4 \ube44\uc2b7\ud55c \uc804\ub7b5\uc744 \uc0ac\uc6a9\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n    \"type\": \"username\",\r\n    \"uid\": \"\"\r\n}\r\n<\/code><\/pre>\n<h3>\uc138\uc158 \ubb38\uc11c<\/h3>\n<p>\uc138\uc158 \ubb38\uc11c\ub294 \uc790\ub3d9 \ub9cc\ub8cc\ub418\ub294 \ubb38\uc11c\ub85c, \uc0ac\uc6a9\uc790\uac00 \ubcf4\ub2e4 \uc548\uc804\ud55c \uc815\ubcf4\ub85c \uc774\ub3d9\ud558\ub294 \uacbd\ub85c \uc5ed\ud560\uc744 \ud569\ub2c8\ub2e4. \uc774\ub860\uc0c1\uc73c\ub85c\ub294 \uc0ac\uc6a9\uc790 \ud504\ub7f0\ud2b8\uc5d4\ub4dc\uc5d0\uc11c \uc2dc\ub4dc \uac12\uc744 \uc800\uc7a5\ud558\uc5ec \ubcf4\ud638\ub41c \uc5d4\ub4dc\ud3ec\uc778\ud2b8 \uac04\uc5d0 \uc804\ub2ec\ud569\ub2c8\ub2e4. \uc774\ub97c \ud1b5\ud574 \uc0ac\uc6a9\uc790\uc5d0\uac8c \uc5f0\uacb0\ub41c UID\uc5d0 \uc561\uc138\uc2a4\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n    \"type\": \"session\",\r\n    \"sid\": \"\",\r\n    \"uid\": \"\"\r\n}\r\n<\/code><\/pre>\n<h3>\uc0c1\ud0dc \ubb38\uc11c<\/h3>\n<p>\uc0c1\ud0dc \ubb38\uc11c\uc5d0\ub294 \uac8c\uc784\uc758 \ud2b9\uc815 \uc0c1\ud0dc\uc5d0 \ub300\ud55c \uc815\ubcf4\uac00 \uc800\uc7a5\ub429\ub2c8\ub2e4. \uc608\ub97c \ub4e4\uc5b4 \uac8c\uc784 \uce90\ub9ad\ud130\uc758 \uc0dd\uba85\ub825\uc774 5\uac1c, \ubb3c\uc57d\uc774 20\uac1c \ub0a8\uc740 \uacbd\uc6b0 \ud574\ub2f9 \uc815\ubcf4\uac00 \uc5ec\uae30\uc5d0 \uc800\uc7a5\ub429\ub2c8\ub2e4. \ub450 \uac1c\uc758 \ud65c\uc131 \uac8c\uc784 \uc138\uc158\uc774 \uac19\uc740 \uacc4\uc815\uc73c\ub85c \uc800\uc7a5\ub418\ub294 \uac83\uc744 \ubc29\uc9c0\ud558\uae30 \uc704\ud55c \ubc84\uc804 \uc815\ubcf4\ub3c4 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-json\">\r\n{\r\n    \"type\": \"state\",\r\n    \"uid\": \"\",\r\n    \"states\": {\r\n        \"state-name\": {\r\n            \"version\": 1,\r\n            \"data\": { }\r\n        }\r\n    }\r\n}\r\n<\/code><\/pre>\n<h2>\uacc4\uc815 \ubaa8\ub378 \ub9cc\ub4e4\uae30<\/h2>\n<p>\uacc4\uc815 \ubaa8\ub378\uc740 \uc138 \uac00\uc9c0 \ud2b9\ubcc4\ud55c \uc6a9\ub3c4\ub85c \uc0ac\uc6a9\ub429\ub2c8\ub2e4:<\/p>\n<ul>\n<li>\uc0ac\uc6a9\uc790 \uacc4\uc815 \ub9cc\ub4e4\uae30<\/li>\n<li>\uc0ac\uc6a9\uc790 \uacc4\uc815 \ubcf5\uad6c\ud558\uae30<\/li>\n<li>\ud574\uc2dc\ub41c \ube44\ubc00\ubc88\ud638\uc640 \ud574\uc2dc\ub418\uc9c0 \uc54a\uc740 \ube44\ubc00\ubc88\ud638 \ube44\uad50\ud558\uae30<\/li>\n<\/ul>\n<p>\ucf54\ub529\uc744 \uc2dc\uc791\ud558\uae30 \uc804\uc5d0 \uc778\ud074\ub8e8\ub4dc\ub97c \uc21c\uc11c\ub300\ub85c \uc815\ub9ac\ud574\uc57c \ud569\ub2c8\ub2e4. \ub2e4\uc74c\uc744\u00a0<strong>models\/accountmodel.js<\/strong> file:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar uuid = require(\"uuid\");\r\nvar forge = require(\"node-forge\");\r\nvar db = require(\"..\/app\").bucket;\r\nvar N1qlQuery = require('couchbase').N1qlQuery;\r\n<\/code><\/pre>\n<p>\uc704\uc5d0\uc11c \ubcf8 \ub370\uc774\ud130 \ubaa8\ub378\uc5d0 \ub530\ub77c \uc0ac\uc6a9\uc790 \uacc4\uc815\uc744 \ub9cc\ub4e4\ub824\uba74 \uc774\ub984, \uc0ac\uc6a9\uc790 \uc544\uc774\ub514, \ube44\ubc00\ubc88\ud638\uac00 \ud544\uc694\ud569\ub2c8\ub2e4. \uc774 \uc815\ubcf4\uac00 \uc900\ube44\ub418\uba74 \ube44\ubc00\ubc88\ud638\ub294 \ud574\uc2dc\ub418\uace0 \uc0ac\uc6a9\uc790 \uc815\ubcf4\ub294 \ub2e4\uc74c\uacfc \uac19\uc774 \ucc38\uc870 \ubb38\uc11c\uc640 \ud568\uaed8 \uc800\uc7a5\ub429\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nAccountModel.create = function(params, callback) {\r\n    var userDoc = {\r\n        type: \"user\",\r\n        uid: uuid.v4(),\r\n        name: params.name,\r\n        username: params.username,\r\n        password: forge.md.sha1.create().update(params.password).digest().toHex()\r\n    };\r\n    var referenceDoc = {\r\n        type: \"username\",\r\n        uid: userDoc.uid\r\n    };\r\n    db.insert(\"username::\" + userDoc.username, referenceDoc, function(error) {\r\n        if(error) {\r\n            callback(error, null);\r\n            return;\r\n        }\r\n        db.insert(\"user::\" + userDoc.uid, userDoc, function(error, result) {\r\n            if(error) {\r\n                callback(error, null);\r\n                return;\r\n            }\r\n            callback(null, {message: \"success\", data: result});\r\n        });\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>\uc704 \ucf54\ub4dc\uc5d0\uc11c \uba3c\uc800 \uc0c8 \ucc38\uc870 \ubb38\uc11c\ub97c \uc0bd\uc785\ud558\ub824\uace0 \uc2dc\ub3c4\ud569\ub2c8\ub2e4. \uc2e4\ud328\ud558\uba74(\uc774\ubbf8 \uc874\uc7ac\ud558\ub294 \ubb38\uc11c\uc77c \uc218\ub3c4 \uc788\uc74c) \ub450 \ubb38\uc11c \ubaa8\ub450 \uc800\uc7a5\ub418\uc9c0 \uc54a\uace0 \ub300\uc2e0 \uc624\ub958 \uba54\uc2dc\uc9c0\uac00 \ubc18\ud658\ub429\ub2c8\ub2e4. \uc131\uacf5 \ub610\ub294 \uc2e4\ud328 \uc5ec\ubd80\uc5d0 \uad00\uacc4\uc5c6\uc774 \ub77c\uc6b0\ud305 \ud30c\uc77c\uc758 \ucf5c\ubc31\uc774 \uc2e4\ud589\ub418\uc5b4 \uc694\uccad\uc790\uc5d0\uac8c \uc5b4\ub5a4 \uc885\ub958\uc758 \ub2f5\ubcc0\uc774\ub4e0 \ud45c\uc2dc\ud569\ub2c8\ub2e4.<\/p>\n<p>\uc0ac\uc6a9\uc790 \ub370\uc774\ud130\ub97c \uc77d\uc744 \ub54c \ub450 \uac00\uc9c0 \uc911 \ud558\ub098\uac00 \ubc1c\uc0dd\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc77c\uc885\uc758 \uac04\ub2e8\ud55c \ub85c\uadf8\uc778\uc744 \uc218\ud589\ud558\uae30 \ub54c\ubb38\uc5d0 \uc0ac\uc6a9\uc790 \uc774\ub984\uc744 \uc804\ub2ec\ud558\uac70\ub098 \uc0ac\uc6a9\uc790 ID\ub97c \uc804\ub2ec\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \ubaa9\uc801\uc5d0 \ub530\ub77c \ub2e4\ub985\ub2c8\ub2e4. \uc0ac\uc6a9\uc790 \uc774\ub984\uc73c\ub85c\ub9cc \ub85c\uadf8\uc778\ud55c\ub2e4\uace0 \uac00\uc815\ud558\uba74 \ub2e4\uc74c\uacfc \uac19\uc740 \ud568\uc218\ub97c \ud638\ucd9c\ud558\uace0 \uc2f6\uc744 \uac83\uc785\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nAccountModel.getByUsername = function(params, callback) {\r\n    var query = N1qlQuery.fromString(\r\n        \"select users.* from `gaming-sample` as usernames \" +\r\n        \"join `gaming-sample` as users on keys (\"user::\" || usernames.uid) \" +\r\n        \"where meta(usernames).id = $1\"\r\n    );\r\n    db.query(query, [\"username::\" + params.username], function(error, result) {\r\n        if(error) {\r\n            return callback(error, null);\r\n        }\r\n        callback(null, result);\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>Couchbase 4.0\uc758 \uc0c8\ub85c\uc6b4 \uae30\ub2a5\uc778 N1QL \ucffc\ub9ac\ub97c \uc5b4\ub5bb\uac8c \uc0ac\uc6a9\ud558\uace0 \uc788\ub294\uc9c0 \uc8fc\ubaa9\ud558\uc138\uc694. \uc774 \ucffc\ub9ac\ub294 \uae30\uc874 SQL\uacfc \ub9e4\uc6b0 \uc720\uc0ac\ud558\uba70 \uc560\ud50c\ub9ac\ucf00\uc774\uc158 \uacc4\uce35\uc5d0\uc11c \ub370\uc774\ud130\ub97c \ud06c\ub7f0\uce58\ud558\uac70\ub098 \ud3ec\ub9f7\ud560 \ud544\uc694\uac00 \uc5c6\ub294 \ud3b8\ub9ac\ud568\uc744 \uc81c\uacf5\ud569\ub2c8\ub2e4. \uc774 \ubaa8\ub4e0 \uc791\uc5c5\uc740 Couchbase Server\uac00 \uc790\ub3d9\uc73c\ub85c \uc218\ud589\ud569\ub2c8\ub2e4. \ud558\uc9c0\ub9cc \uc774\uc804 \ubc84\uc804\uc758 Couchbase \ubc0f \uae30\ud0c0 NoSQL \ud50c\ub7ab\ud3fc\uc5d0\uc11c\uc640 \uac19\uc774 \ub370\uc774\ud130\ub97c \uc694\uccad\ud560 \uc218 \uc788\ub294 \uc635\uc158\ub3c4 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\uc704\uc758 N1QL \ubb38\uc5d0\uc11c\ub294 \uc77c\ubc18 \ud14d\uc2a4\ud2b8 \uc0ac\uc6a9\uc790 \uc774\ub984\uc5d0 \ubcf5\ud569 \ud0a4\uac00 \ucd94\uac00\ub41c \ubb38\uc11c\ub97c \uc120\ud0dd\ud558\uace0 \uc788\uc2b5\ub2c8\ub2e4. \uc0ac\uc6a9\uc790 \uc774\ub984 \ubb38\uc11c(\uc678\ub798 \ud0a4)\uc640 \uc0ac\uc6a9\uc790 \ubb38\uc11c(\uae30\ubcf8 \ud0a4)\uc758 uid \uc18d\uc131\uc744 \uc0ac\uc6a9\ud558\uc5ec \uc870\uc778\uc774 \uc774\ub8e8\uc5b4\uc9c0\uace0 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\uc774\uc81c \ube44\ubc00\ubc88\ud638\ub97c \ud655\uc778\ud558\ub294 \ub2e8\uacc4\ub85c \ub118\uc5b4\uac11\ub2c8\ub2e4. \uc5ec\uae30\uc11c\ub294 \ub370\uc774\ud130\ubca0\uc774\uc2a4\ub97c \ud638\ucd9c\ud558\uc9c0 \uc54a\uc2b5\ub2c8\ub2e4. \uc6d0\uc2dc \ube44\ubc00\ubc88\ud638\ub97c \uac00\uc838\uc640\uc11c \ud574\uc2f1\ud55c \ub2e4\uc74c \ud574\uc2dc\ub97c \uc800\uc7a5\ub41c \ube44\ubc00\ubc88\ud638\uc640 \ube44\uad50\ud558\uae30\ub9cc \ud558\uba74 \ub429\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\nAccountModel.validatePassword = function(rawPassword, hashedPassword) {\r\n    return forge.md.sha1.create().update(rawPassword).digest().toHex() === hashedPassword ? true : false;\r\n};\r\n<\/code><\/pre>\n<p>\ub9cc\ub4e4\ub824\uba74 <strong>models\/accountmodel.js<\/strong> \uacbd\ub85c \ud30c\uc77c\uc5d0\uc11c \uc0ac\uc6a9\ud560 \uc218 \uc788\ub3c4\ub85d \ud558\ub824\uba74 \ub2e4\uc74c\uacfc \uac19\uc774 \ub0b4\ubcf4\ub0b4\uc57c \ud569\ub2c8\ub2e4.<br \/>\n<strong>models\/accountmodel.js<\/strong> \ucf54\ub4dc:<\/p>\n<pre><code class=\"language-javascript\">\r\nmodule.exports = AccountModel;\r\n<\/code><\/pre>\n<h2>\uc138\uc158 \ubaa8\ub378 \ub9cc\ub4e4\uae30<\/h2>\n<p>\uc138\uc158 \ubaa8\ub378\uc740 \uc138 \uac00\uc9c0 \ud2b9\ubcc4\ud55c \uc6a9\ub3c4\ub85c \uc0ac\uc6a9\ub429\ub2c8\ub2e4:<\/p>\n<ul>\n<li>\uc0ac\uc6a9\uc790 \uc138\uc158 \ub9cc\ub4e4\uae30<\/li>\n<li>\uc0ac\uc6a9\uc790 \uc138\uc158 \ub2e4\uc2dc \uac00\uc838\uc624\uae30<\/li>\n<li>\uc0ac\uc6a9\uc790 \uc138\uc158 \uc720\ud6a8\uc131 \uac80\uc0ac<\/li>\n<\/ul>\n<p>\ucf54\ub529\uc744 \uc2dc\uc791\ud558\uae30 \uc804\uc5d0 \uc778\ud074\ub8e8\ub4dc\ub97c \uc21c\uc11c\ub300\ub85c \uc815\ub9ac\ud574\uc57c \ud569\ub2c8\ub2e4. \ub2e4\uc74c\uc744\u00a0<strong>models\/sessionmodel.js<\/strong> file:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar uuid = require(\"uuid\");\r\nvar db = require(\"..\/app\").bucket;\r\n<\/code><\/pre>\n<h3>\uc138\uc158 \ub9cc\ub4e4\uae30<\/h3>\n<p>\uc0ac\uc6a9\uc790\uac00 \uc138\uc158\uc744 \uc0dd\uc131\ud558\ub824\uba74 \uc544\uc774\ub514\ub97c \uc81c\uacf5\ud574\uc57c \ud569\ub2c8\ub2e4. \uc774\ub97c \ud1b5\ud574 \uace0\uc720\ud55c \uc138\uc158 ID\uac00 \uc0dd\uc131\ub418\uace0 \ub9cc\ub8cc\uc640 \ud568\uaed8 \ub370\uc774\ud130\ubca0\uc774\uc2a4\uc5d0 \uc0bd\uc785\ub429\ub2c8\ub2e4. \ubb38\uc11c\uac00 \ub9cc\ub8cc\ub418\uba74 \uc0ac\uc6a9\uc790\uc758 \uac1c\uc785 \uc5c6\uc774 \uc790\ub3d9\uc73c\ub85c Couchbase\uc5d0\uc11c \uc81c\uac70\ub418\uc5b4 \uc0ac\uc6a9\uc790\uac00 \ub85c\uadf8\uc544\uc6c3\ub429\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\nSessionModel.create = function(uid, callback) {\r\n    var sessionDoc = {\r\n        type: \"session\",\r\n        sid: uuid.v4(),\r\n        uid: uid\r\n    };\r\n    db.insert(\"session::\" + sessionDoc.sid, sessionDoc, {\"expiry\": 3600}, function(error, result) {\r\n        if(error) {\r\n            callback(error, null);\r\n            return;\r\n        }\r\n        callback(null, sessionDoc.sid);\r\n    });\r\n};\r\n<\/code><\/pre>\n<h3>\uc0ac\uc6a9\uc790 \uc778\uc99d\ud558\uae30<\/h3>\n<p>\uc0ac\uc6a9\uc790 \uc138\uc158\uc774 \uc0dd\uc131\ub418\uba74 \uc0ac\uc6a9\uc790\uac00 \ubcf4\ud638\ub41c \uc5d4\ub4dc\ud3ec\uc778\ud2b8\uc5d0 \ub3c4\ub2ec\ud560 \ub54c\ub9c8\ub2e4 \uc138\uc158 ID\ub97c \uc0ac\uc6a9\ud574\uc57c \ud569\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\nSessionModel.authenticate = function(req, res, next) {\r\n    if(!req.headers.authorization) {\r\n        next(\"Must be authorized to use\");\r\n    }\r\n    var authInfo = req.headers.authorization.split(\" \");\r\n    if(authInfo[0] === \"Bearer\") {\r\n        var sid = authInfo[1];\r\n        SessionModel.get(sid, function(error, result) {\r\n            if(error) {\r\n                return next(error);\r\n            }\r\n            SessionModel.refresh(sid, function() {});\r\n            req.uid = result.value.uid;\r\n            next();\r\n        });\r\n    }\r\n};\r\n<\/code><\/pre>\n<p>\uc704 \ucf54\ub4dc\uc5d0\uc11c \ud568\uc218\ub294 \uc138\uc158 ID\ub97c \ubc1b\uc544 \uc138\uc158\uc774 \uc774\ubbf8 \uc874\uc7ac\ud558\ub294\uc9c0 \uc870\ud68c\ud558\ub294 \ub370 \uc0ac\uc6a9\ud569\ub2c8\ub2e4. \uc138\uc158\uc774 \uc874\uc7ac\ud558\uba74 \uc138\uc158 \ub9cc\ub8cc \uc2dc\uac04\uc744 \uc7ac\uc124\uc815\ud558\uace0 \uc138\uc158\uacfc \uc5f0\uacb0\ub41c UID\ub97c \ubc18\ud658\ud569\ub2c8\ub2e4. \uc138\uc158 \uc815\ubcf4\ub97c \uac00\uc838\uc624\ub294 \ubc29\ubc95\uc740 \ub2e4\uc74c\uacfc \uac19\uc2b5\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nSessionModel.get = function(sid, callback) {\r\n    db.get(\"session::\" + sid, function(error, result) {\r\n        if(error) {\r\n            callback(error, null);\r\n            return;\r\n        }\r\n        callback(null, result);\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>\ub098\uc058\uc9c0 \uc54a\uc8e0? \ub9c8\uc9c0\ub9c9\uc73c\ub85c \uc138\uc158 \uc7ac\uc124\uc815\ub3c4 \ube44\uc2b7\ud55c \ubc29\uc2dd\uc73c\ub85c \uc774\ub8e8\uc5b4\uc9d0\uc744 \uc54c \uc218 \uc788\uc2b5\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\nSessionModel.refresh = function(sid, callback) {\r\n    db.touch(\"session::\" + sid, 3600, function(error, result) {\r\n        if(error) {\r\n            callback(error, null);\r\n        }\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>\ud130\uce58 \ubc29\uc2dd\uc740 \uc2dc\uac04\uc744 \ucd94\uac00\ud558\uc9c0 \uc54a\uace0 \ub300\uc2e0 \uc2dc\uac04\uc744 \ucd08\uae30\ud654\ud569\ub2c8\ub2e4. \uc774 \uacbd\uc6b0 \ud0c0\uc774\uba38\uac00 1\uc2dc\uac04\uc73c\ub85c \uc7ac\uc124\uc815\ub429\ub2c8\ub2e4.<\/p>\n<p>\ub9cc\ub4e4\ub824\uba74 <strong>models\/sessionmodel.js<\/strong> \uacbd\ub85c \ud30c\uc77c\uc5d0\uc11c \uc0ac\uc6a9\ud560 \uc218 \uc788\ub3c4\ub85d \ud558\ub824\uba74 \ub2e4\uc74c\uacfc \uac19\uc774 \ub0b4\ubcf4\ub0b4\uc57c \ud569\ub2c8\ub2e4.\u00a0<strong>models\/sessionmodel.js<\/strong> \ucf54\ub4dc:<\/p>\n<pre><code class=\"language-javascript\">\r\nmodule.exports = SessionModel;\r\n<\/code><\/pre>\n<h2>\uc0c1\ud0dc \ubaa8\ub378 \ub9cc\ub4e4\uae30<\/h2>\n<p>\uc0c1\ud0dc \ubaa8\ub378\uc740 \ub450 \uac00\uc9c0 \ud2b9\ubcc4\ud55c \uc6a9\ub3c4\ub85c \uc0ac\uc6a9\ub429\ub2c8\ub2e4:<\/p>\n<ul>\n<li>\uc800\uc7a5 \uc0c1\ud0dc \uc0dd\uc131 \ub610\ub294 \uc5c5\ub370\uc774\ud2b8\ud558\uae30<\/li>\n<li>\uc774\ub984\uc73c\ub85c \uc800\uc7a5 \uc0c1\ud0dc \uac00\uc838\uc624\uae30<\/li>\n<\/ul>\n<p>\ucf54\ub529\uc744 \uc2dc\uc791\ud558\uae30 \uc804\uc5d0 \uc778\ud074\ub8e8\ub4dc\ub97c \uc21c\uc11c\ub300\ub85c \uc815\ub9ac\ud574\uc57c \ud569\ub2c8\ub2e4. \ub2e4\uc74c\uc744\u00a0<strong>models\/statemodel.js<\/strong> file:<\/p>\n<pre><code class=\"language-javascript\">\r\nvar uuid = require(\"uuid\");\r\nvar couchbase = require(\"couchbase\");\r\nvar N1qlQuery = require('couchbase').N1qlQuery;\r\nvar db = require(\"..\/app\").bucket;\r\n<\/code><\/pre>\n<h3>\uc800\uc7a5 \uc0c1\ud0dc \ub9cc\ub4e4\uae30<\/h3>\n<p>\uc800\uc7a5 \uc0c1\ud0dc\ub97c \ub9cc\ub4e4\uac70\ub098 \uc5c5\ub370\uc774\ud2b8\ud558\ub294 \ubaa9\uc801\uc740 \uba3c\uc800 \uc800\uc7a5 \uc0c1\ud0dc\uac00 \uc874\uc7ac\ud558\ub294\uc9c0 \ud655\uc778\ud558\ub294 \uac83\uc785\ub2c8\ub2e4. \uc874\uc7ac\ud558\uc9c0 \uc54a\uc73c\uba74 \uc0dd\uc131\ud569\ub2c8\ub2e4. \uc874\uc7ac\ud55c\ub2e4\uba74 \uc874\uc7ac\ud558\ub294 \uc815\ubcf4\ub97c \uac00\uc838\uc640\uc11c \ubcc0\uacbd\ud55c \ub2e4\uc74c \ub370\uc774\ud130\ubca0\uc774\uc2a4\uc5d0 \ud604\uc7ac \uc874\uc7ac\ud558\ub294 \ubaa8\ub4e0 \uc815\ubcf4\ub97c \ub300\uccb4\ud569\ub2c8\ub2e4. \uc774 \ubaa8\ub4e0 \uc791\uc5c5\uc740 \uac8c\uc784 \uc800\uc7a5 \ubc84\uc804 \uac04\uc758 \ucda9\ub3cc\uc744 \ud53c\ud558\uae30 \uc704\ud574 \uc0c1\ud0dc \ubc84\uc804\uc744 \ub298\ub9ac\uba74\uc11c \uc218\ud589\ub429\ub2c8\ub2e4. \uc2e4\uc81c\ub85c \uce74\uc6b0\uce58\ubca0\uc774\uc2a4\uc5d0 \uc800\uc7a5\ud560 \ub54c \ucda9\ub3cc\uc744 \ubc29\uc9c0\ud558\uae30 \uc704\ud55c \uac83\uc774 \uc544\ub2c8\ub77c, \ub450 \uac1c\uc758 \uae30\uae30\uc5d0\uc11c \uac8c\uc784\uc744 \uc120\ud0dd\ud574\uc11c \ud6e8\uc52c \uc624\ub798\ub41c \uc800\uc7a5\uc73c\ub85c \uac8c\uc784 \ub370\uc774\ud130\ub97c \ub36e\uc5b4\uc4f0\uc9c0 \uc54a\ub3c4\ub85d \ud558\uae30 \uc704\ud55c \uac83\uc785\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\nStateModel.save = function(uid, name, preVer, data, callback) {\r\n    db.get(\"user::\" + uid + \"::state\", function(error, result) {\r\n        if(error &amp;&amp; error.code !== couchbase.errors.keyNotFound) {\r\n            callback(error, null);\r\n            return;\r\n        }\r\n        var stateDoc = {\r\n            type: \"state\",\r\n            uid: uid,\r\n            states: {}\r\n        };\r\n        if(result != null &amp;&amp; result.value) {\r\n            stateDoc = result.value;\r\n        }\r\n        var stateBlock = {\r\n            version: 0,\r\n            data: null\r\n        };\r\n        if(stateDoc.states[name]) {\r\n            stateBlock = stateDoc.states[name];\r\n        } else {\r\n            stateDoc.states[name] = stateBlock;\r\n        }\r\n        if(stateBlock.version !== preVer) {\r\n            return callback({\"status\": \"error\", \"message\": \"Your version does not match the server version\"});\r\n        } else {\r\n            stateBlock.version++;\r\n            stateBlock.data = data;\r\n        }\r\n        var stateOptions = {};\r\n        if(result != null &amp;&amp; result.value) {\r\n            stateOptions.cas = result.cas;\r\n        }\r\n        db.upsert(\"user::\" + uid + \"::state\", stateDoc, stateOptions, function(error, result) {\r\n            if(error) {\r\n                return callback(error, null);\r\n            }\r\n            callback(null, stateBlock);\r\n        });\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>\ucc38\uc870 <strong>\uc5c5\uc11c\ud2b8<\/strong> \ub97c \uc785\ub825\ud558\uc138\uc694. \uce74\uc6b0\uce58\ubca0\uc774\uc2a4\uc5d0\uc11c\ub294 \uc874\uc7ac\ud558\uc9c0 \uc54a\uc73c\uba74 \uc0dd\uc131\ud558\uace0, \uc874\uc7ac\ud558\uba74 \ub300\uccb4\ud55c\ub2e4\ub294 \ub73b\uc785\ub2c8\ub2e4. \uac8c\uc784\uc758 \uc800\uc7a5 \uc0c1\ud0dc\uc640 \uac19\uc740 \uc791\uc5c5\uc5d0 \ub9e4\uc6b0 \ud3b8\ub9ac\ud569\ub2c8\ub2e4.<\/p>\n<h3>\uc0c1\ud0dc \uac00\uc838\uc624\uae30<\/h3>\n<p>\uc774\ub807\uac8c \ud558\uba74 \uc0dd\uc131\ud55c \ubaa8\ub4e0 \uc800\uc7a5 \uc0c1\ud0dc\ub97c \uac00\uc838\uc62c \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\nStateModel.getByUserIdAndName = function(uid, name, callback) {\r\n    db.get(\"user::\" + uid + \"::state\", function(error, result) {\r\n        if(error) {\r\n            if(error.code !== couchbase.errors.keyNotFound) {\r\n                return callback(null, {});\r\n            } else {\r\n                return callback(error, null);\r\n            }\r\n        }\r\n        if(!result.value.states[name]) {\r\n            return callback({\"status\": \"error\", \"message\": \"State does not exist\"}, null);\r\n        }\r\n        callback(null, result.value.states[name]);\r\n    });\r\n};\r\n<\/code><\/pre>\n<p>\uc774 \uac1c\ub150\uc740 \uc0ac\uc6a9\uc790 ID\ub97c \uae30\ubc18\uc73c\ub85c \ubb38\uc11c \uc870\ud68c\ub97c \uc218\ud589\ud55c\ub2e4\ub294 \uac83\uc785\ub2c8\ub2e4. \ud2b9\uc815 \uc544\uc774\ub514\uc5d0 \ub300\ud55c \uc0c1\ud0dc \ubb38\uc11c\uac00 \uc874\uc7ac\ud558\uba74 \uc5f0\uad00 \ubc30\uc5f4\uc5d0\uc11c \uc870\ud68c\ub97c \uc218\ud589\ud558\uc5ec \uc0c1\ud0dc \uc774\ub984\uc774 \uc874\uc7ac\ud558\ub294\uc9c0 \ud655\uc778\ud569\ub2c8\ub2e4. \uc874\uc7ac\ud55c\ub2e4\uba74 \uadf8 \uc774\ub984\uc5d0 \ub300\ud574 \uc874\uc7ac\ud558\ub294 \ubaa8\ub4e0 \uc0c1\ud0dc \ucf58\ud150\uce20\ub97c \ubc18\ud658\ud569\ub2c8\ub2e4.<\/p>\n<p>\ub9cc\ub4e4\ub824\uba74 <strong>models\/statemodel.js<\/strong> \uacbd\ub85c \ud30c\uc77c\uc5d0\uc11c \uc0ac\uc6a9\ud560 \uc218 \uc788\ub3c4\ub85d \ud558\ub824\uba74 \ub2e4\uc74c\uacfc \uac19\uc774 \ub0b4\ubcf4\ub0b4\uc57c \ud569\ub2c8\ub2e4.\u00a0<strong>models\/statemodel.js<\/strong> \ucf54\ub4dc:<\/p>\n<pre><code class=\"language-javascript\">\r\nmodule.exports = StateModel;\r\n<\/code><\/pre>\n<h2>API \uacbd\ub85c \ub9cc\ub4e4\uae30<\/h2>\n<p>\uc704\uc5d0\uc11c \ud544\uc694\ud55c \ubaa8\ub4e0 \ub370\uc774\ud130 \ubaa8\ub378\uc744 \ub9cc\ub4e4\uc5c8\uc73c\ubbc0\ub85c \uc774\uc81c \uc774\ub97c \uc0ac\uc6a9\uc790\uac00 \uc561\uc138\uc2a4\ud560 \uc218 \uc788\ub294 \uacbd\ub85c\uc640 \ud568\uaed8 \uc5f0\uacb0\ud560 \ucc28\ub840\uc785\ub2c8\ub2e4. \ub2e4\uc2dc <strong>routes\/routes.js<\/strong> \ud30c\uc77c\uc5d0 \uacc4\uc815 \ubaa8\ub378 \uacbd\ub85c\ub97c \ucd94\uac00\ud558\ub294 \uac83\ubd80\ud130 \uc2dc\uc791\ud558\uaca0\uc2b5\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\napp.post(\"\/api\/user\", function(req, res) {\r\n    if(!req.body.name) {\r\n        return res.status(400).send({\"status\": \"error\", \"message\": \"A name is required\"});\r\n    } else if(!req.body.username) {\r\n        return res.status(400).send({\"status\": \"error\", \"message\": \"A username is required\"});\r\n    } else if(!req.body.password) {\r\n        return res.status(400).send({\"status\": \"error\", \"message\": \"A password is required\"});\r\n    }\r\n    AccountModel.create(req.body, function(error, result) {\r\n        if(error) {\r\n            return res.status(400).send(error);\r\n        }\r\n        res.send(result);\r\n    });\r\n});\r\n<\/code><\/pre>\n<p>\uc704\uc758 \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \uc774\ub984, \uc0ac\uc6a9\uc790 \uc774\ub984, \ube44\ubc00\ubc88\ud638 \ubcf8\ubb38 \ub9e4\uac1c\ubcc0\uc218\uac00 \ud3ec\ud568\ub41c POST \uc694\uccad\uc744 \uae30\ub300\ud569\ub2c8\ub2e4. HTTP \uc694\uccad\uc744 \ud1b5\ud574 \ub370\uc774\ud130\ub97c \uc0dd\uc131\ud558\uac70\ub098 \uc0bd\uc785\ud560 \ub54c POST\ub97c \uc0ac\uc6a9\ud558\ub294 \uac83\uc774 \uac00\uc7a5 \uc88b\uc740 \ubc29\ubc95\uc774\uae30 \ub54c\ubb38\uc5d0 POST\ub97c \uc218\uc2e0 \ub300\uae30\ud558\uace0 \uc788\uc2b5\ub2c8\ub2e4. \uc774 \uc138 \uac00\uc9c0\uac00 \ubaa8\ub450 \uc874\uc7ac\ud558\uba74\u00a0<strong>AccountModel.create()<\/strong> \uba54\uc11c\ub4dc\uac00 \ud638\ucd9c\ub418\uc5b4 \uba54\uc11c\ub4dc\uc758 \uc131\uacf5 \uc5ec\ubd80\uc5d0 \ub530\ub77c \uc624\ub958 \ub610\ub294 \uacb0\uacfc\ub97c \ubc18\ud658\ud569\ub2c8\ub2e4. \ud544\uc218 \ub9e4\uac1c\ubcc0\uc218 \uc911 \ud558\ub098 \uc774\uc0c1\uc774 \uc874\uc7ac\ud558\uc9c0 \uc54a\uc73c\uba74 \uc624\ub958\uac00 \ubc18\ud658\ub429\ub2c8\ub2e4. \uc624\ub958 \ucf54\ub4dc \ubaa9\ub85d\uc740 \ub2e4\uc74c\uacfc \uac19\uc2b5\ub2c8\ub2e4. <a href=\"https:\/\/en.wikipedia.org\/wiki\/List_of_HTTP_status_codes\">\uc5ec\uae30<\/a>.<\/p>\n<p>\uc774 \uc608\uc81c\uc5d0\uc11c\ub294 \uc0ac\uc6a9\uc790 \uc815\ubcf4\ub97c \uac00\uc838\uc624\uae30 \uc704\ud55c \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \uadf8\ub2e4\uc9c0 \uc911\uc694\ud558\uc9c0 \uc54a\uc73c\ubbc0\ub85c \uc0ac\uc6a9\uc790 \uc778\uc99d\uacfc \uc138\uc158 \uc0dd\uc131\uc73c\ub85c \ubc14\ub85c \ub118\uc5b4\uac00\uaca0\uc2b5\ub2c8\ub2e4. \uc774 \uc608\uc81c\uc5d0\uc11c\ub294 <strong>routes\/routes.js<\/strong> \ud30c\uc77c\uc5d0 \ub2e4\uc74c\uc744 \ucd94\uac00\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\napp.get(\"\/api\/auth\", function(req, res, next) {\r\n    if(!req.query.username) {\r\n        return next(JSON.stringify({\"status\": \"error\", \"message\": \"A username must be provided\"}));\r\n    }\r\n    if(!req.query.password) {\r\n        return next(JSON.stringify({\"status\": \"error\", \"message\": \"A password must be provided\"}));\r\n    }\r\n    AccountModel.getByUsername(req.query, function(error, user) {\r\n        if(error) {\r\n            return res.status(400).send(error);\r\n        }\r\n        if(!AccountModel.validatePassword(req.query.password, user[0].password)) {\r\n            return res.send({\"status\": \"error\", \"message\": \"The password entered is invalid\"});\r\n        }\r\n        SessionModel.create(user[0].uid, function(error, result) {\r\n            if(error) {\r\n                return res.status(400).send(error);\r\n            }\r\n            res.setHeader(\"Authorization\", \"Bearer \" + result);\r\n            res.send(user);\r\n        });\r\n    });\r\n});\r\n<\/code><\/pre>\n<p>\uc778\uc99d \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \uc0ac\uc6a9\uc790 \uc774\ub984\uacfc \ube44\ubc00\ubc88\ud638\ub97c \uc694\uad6c\ud569\ub2c8\ub2e4. \ub450 \uac00\uc9c0\uac00 \ubaa8\ub450 \ubc1c\uacac\ub418\uba74 \uc0ac\uc6a9\uc790\ub97c \uc870\ud68c\ud569\ub2c8\ub2e4. \uc0ac\uc6a9\uc790\ub97c \ucc3e\uc73c\uba74 \ube44\ubc00\ubc88\ud638 \ube44\uad50\uac00 \uc774\ub8e8\uc5b4\uc9c0\uace0 \uc131\uacf5\ud558\uba74 \uc138\uc158\uc774 \uc0dd\uc131\ub429\ub2c8\ub2e4.<\/p>\n<p>\ub9c8\uc9c0\ub9c9\uc73c\ub85c \uc720\uc6a9\ud55c \ub450 \uac00\uc9c0 API \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \uc800\uc7a5 \uc0c1\ud0dc\ub97c \uac00\uc838\uc624\uace0 \uc0dd\uc131\ud558\ub294 \ub370 \uc0ac\uc6a9\ub429\ub2c8\ub2e4. \uc800\uc7a5 \uc0c1\ud0dc \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub97c \ub9cc\ub4dc\ub294 \uac83\ubd80\ud130 \uc2dc\uc791\ud574\uc11c, \uc5ec\ub7ec\ubd84\uc758\u00a0<strong>routes\/routes.js<\/strong>\ub97c \ud074\ub9ad\ud558\uace0 \ub2e4\uc74c\uc744 \ucd94\uac00\ud569\ub2c8\ub2e4:<\/p>\n<pre><code class=\"language-javascript\">\r\napp.put(\"\/api\/state\/:name\", SessionModel.authenticate, function(req, res, next) {\r\n    StateModel.save(req.uid, req.params.name, parseInt(req.query.preVer, 10), req.body, function(error, result) {\r\n        if(error) {\r\n            return res.send(error);\r\n        }\r\n        res.send(result);\r\n    });\r\n});\r\n<\/code><\/pre>\n<p>\uc704\uc758 \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \uc0c1\ud0dc \uc774\ub984\uc744 \ub098\ud0c0\ub0b4\ub294 URL \ud30c\ub77c\ubbf8\ud130, \ud604\uc7ac \uc0c1\ud0dc \ubc84\uc804\uc744 \ub098\ud0c0\ub0b4\ub294 \ucffc\ub9ac \ud30c\ub77c\ubbf8\ud130, \uc800\uc7a5\ud560 \uac00\uce58\uac00 \uc788\ub294 \uac8c\uc784 \ub370\uc774\ud130\ub97c \ub098\ud0c0\ub0b4\ubbc0\ub85c \uc0c1\uc0c1\ud560 \uc218 \uc788\ub294 \ubaa8\ub4e0 JSON\uc744 \ud3ec\ud568\ud560 \uc218 \uc788\ub294 \uc694\uccad \ubcf8\ubb38\uc744 \uae30\ub300\ud569\ub2c8\ub2e4.<\/p>\n<p>\ub9c8\uc9c0\ub9c9\uc73c\ub85c \uc800\uc7a5\ub41c \uc0c1\ud0dc\ub97c \uac00\uc838\uc624\ub294 \uc791\uc5c5\ub9cc \ub0a8\uc558\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-javascript\">\r\napp.get(\"\/api\/state\/:name\", SessionModel.authenticate, function(req, res, next) {\r\n    StateModel.getByUserIdAndName(req.uid, req.params.name, function(error, result) {\r\n        if(error) {\r\n            return res.send(error);\r\n        }\r\n        res.send(result);\r\n    });\r\n});\r\n<\/code><\/pre>\n<p>\uc704\uc758 \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 \ud2b9\uc815 \uc800\uc7a5 \uc0c1\ud0dc\ub97c \ub098\ud0c0\ub0b4\ub294 URL \ub9e4\uac1c\ubcc0\uc218\ub97c \ucc3e\uc744 \uac83\uc73c\ub85c \uc608\uc0c1\ud569\ub2c8\ub2e4. \ubb3c\ub860 \uc0ac\uc6a9\uc790\uac00 \uba3c\uc800 \uc778\uc99d\ub418\uae30\ub97c \uae30\ub300\ud569\ub2c8\ub2e4.<\/p>\n<h2>API \ud14c\uc2a4\ud2b8<\/h2>\n<p>\uc5d0\uc11c \ub9cc\ub4e0 API \uc5d4\ub4dc\ud3ec\uc778\ud2b8\ub294 <strong>routes\/routes.js<\/strong> \ud30c\uc77c\uc740 \uba87 \uac00\uc9c0 \ubc29\ubc95\uc73c\ub85c \ud14c\uc2a4\ud2b8\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc81c\uac00 \uac00\uc7a5 \uc88b\uc544\ud558\ub294 \ub450 \uac00\uc9c0 \ud14c\uc2a4\ud2b8 \ubc29\ubc95\uc740 Chrome\uc6a9 Postman \ud655\uc7a5 \ud504\ub85c\uadf8\ub7a8 \ub610\ub294 cURL\uc744 \uc0ac\uc6a9\ud558\ub294 \uac83\uc785\ub2c8\ub2e4. cURL\uc744 \uc0ac\uc6a9\ud558\uc5ec \uc9c1\uc811 \ud14c\uc2a4\ud2b8\ud574 \ubcf4\uc138\uc694:<\/p>\n<pre><code class=\"language-bash\">\r\ncurl -X POST https:\/\/localhost:3000\/api\/user --data \"name=Nic%20Raboy&amp;username=nraboy&amp;password=12345\"\r\n\r\nRESPONSE:\r\n\r\n{\r\n    \"message\": \"success\",\r\n    \"data\": {\r\n        \"cas\": \"16588775686144\"\r\n    }\r\n}\r\n<\/code><\/pre>\n<p>\uc704\uc5d0\uc11c \uc0c8 \uc0ac\uc6a9\uc790 \uacc4\uc815\uc744 \ub9cc\ub4e4\uc5c8\uc2b5\ub2c8\ub2e4.<\/p>\n<pre><code class=\"language-bash\">\r\ncurl -X GET \"https:\/\/localhost:3000\/api\/auth?username=nraboy&amp;password=12345\"\r\n\r\nRESPONSE:\r\n\r\n[\r\n    {\r\n        \"name\": \"Nic Raboy\",\r\n        \"password\": \"8cb2237d0679ca88db6464eac60da96345513964\",\r\n        \"type\": \"user\",\r\n        \"uid\":\"c3e834b0-867e-4a43-aede-b15a4e139adc\",\r\n        \"username\": \"nraboy\"\r\n    }\r\n]\r\n<\/code><\/pre>\n<p>\uc704\uc5d0\uc11c \uc0ac\uc6a9\uc790 \uc138\uc158\uc744 \ub9cc\ub4e4\uc5c8\uc2b5\ub2c8\ub2e4. \ub2e4\ub978 \uc5d4\ub4dc\ud3ec\uc778\ud2b8\uc5d0\ub3c4 \ub3d9\uc77c\ud55c \uc804\ub7b5\uc744 \uc801\uc6a9\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<h2>\uacb0\ub860<\/h2>\n<p>Node.js\uc640 \uce74\uc6b0\uce58\ubca0\uc774\uc2a4 \uc11c\ubc84 SDK\ub97c \uc0ac\uc6a9\ud558\uba74 \uac8c\uc784\uc6a9 API \ubc31\uc5d4\ub4dc\ub97c \uc27d\uac8c \ub9cc\ub4e4 \uc218 \uc788\uc2b5\ub2c8\ub2e4. \uc774\uc81c \uce74\uc6b0\uce58\ubca0\uc774\uc2a4 4.0\uc5d0\uc11c\ub294 \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc758 \ub370\uc774\ud130 \ucffc\ub9ac\ub97c \uc704\ud55c \uc635\uc158\uc73c\ub85c N1QL\uc744 \uc790\uc720\ub86d\uac8c \uc0ac\uc6a9\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4.<\/p>\n<p>\uc774 \ubb38\uc11c\uc5d0\uc11c \uc124\uba85\ud55c \uc804\uccb4 Node.js \uc560\ud50c\ub9ac\ucf00\uc774\uc158\uc740 \ub2e4\uc74c \ub9c1\ud06c\uc5d0\uc11c \ubb34\ub8cc\ub85c \ub2e4\uc6b4\ub85c\ub4dc\ud560 \uc218 \uc788\uc2b5\ub2c8\ub2e4. <a href=\"https:\/\/github.com\/couchbaselabs\/gameapi-nodejs\">\uce74\uc6b0\uce58\ubca0\uc774\uc2a4 \ub7a9 \uae43\ud5c8\ube0c\u00a0<\/a>\ub9ac\ud3ec\uc9c0\ud1a0\ub9ac\uc5d0 \uc800\uc7a5\ud569\ub2c8\ub2e4.<\/p>","protected":false},"excerpt":{"rendered":"<p>A few years ago, Brett Lawson made a great blog series on using Couchbase Server and Node.js for the development a game server framework. Since then, the Node.js SDK for Couchbase has grown significantly from version 1.x to 2.x. In [&hellip;]<\/p>","protected":false},"author":63,"featured_media":13873,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_acf_changed":false,"inline_featured_image":false,"footnotes":""},"categories":[1822,1812],"tags":[1393],"ppma_author":[9032],"class_list":["post-2076","post","type-post","status-publish","format-standard","has-post-thumbnail","hentry","category-node-js","category-n1ql-query","tag-api"],"acf":[],"yoast_head":"<!-- This site is optimized with the Yoast SEO Premium plugin v25.8 (Yoast SEO v25.8) - https:\/\/yoast.com\/wordpress\/plugins\/seo\/ -->\n<title>Node.js Game Server Framework &amp; Development Architecture<\/title>\n<meta name=\"description\" content=\"This article explains development of the latest Node.js game server and Express Framework standards as well as the latest Node.js SDK version of Couchbase.\" \/>\n<meta name=\"robots\" content=\"index, follow, max-snippet:-1, max-image-preview:large, max-video-preview:-1\" \/>\n<link rel=\"canonical\" href=\"https:\/\/www.couchbase.com\/blog\/ko\/making-a-game-api-server-using-nodejs-revisited\/\" \/>\n<meta property=\"og:locale\" content=\"ko_KR\" \/>\n<meta property=\"og:type\" content=\"article\" \/>\n<meta property=\"og:title\" content=\"Making a Game API Server Using Node.js: Revisited\" \/>\n<meta property=\"og:description\" content=\"This article explains development of the latest Node.js game server and Express Framework standards as well as the latest Node.js SDK version of Couchbase.\" \/>\n<meta property=\"og:url\" content=\"https:\/\/www.couchbase.com\/blog\/ko\/making-a-game-api-server-using-nodejs-revisited\/\" \/>\n<meta property=\"og:site_name\" content=\"The Couchbase Blog\" \/>\n<meta property=\"article:author\" content=\"https:\/\/www.facebook.com\/thepolyglotdeveloper\" \/>\n<meta property=\"article:published_time\" content=\"2015-07-16T17:00:00+00:00\" \/>\n<meta property=\"article:modified_time\" content=\"2025-06-14T06:47:49+00:00\" \/>\n<meta name=\"author\" content=\"Nic Raboy, Developer Advocate, Couchbase\" \/>\n<meta name=\"twitter:card\" content=\"summary_large_image\" \/>\n<meta name=\"twitter:creator\" content=\"@nraboy\" \/>\n<meta name=\"twitter:label1\" content=\"Written by\" \/>\n\t<meta name=\"twitter:data1\" content=\"Nic Raboy, Developer Advocate, Couchbase\" \/>\n\t<meta name=\"twitter:label2\" content=\"Est. reading time\" \/>\n\t<meta name=\"twitter:data2\" content=\"14\ubd84\" \/>\n<script type=\"application\/ld+json\" class=\"yoast-schema-graph\">{\"@context\":\"https:\/\/schema.org\",\"@graph\":[{\"@type\":\"Article\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#article\",\"isPartOf\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/\"},\"author\":{\"name\":\"Nic Raboy, Developer Advocate, Couchbase\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/bb545ebe83bb2d12f91095811d0a72e1\"},\"headline\":\"Making a Game API Server Using Node.js: Revisited\",\"datePublished\":\"2015-07-16T17:00:00+00:00\",\"dateModified\":\"2025-06-14T06:47:49+00:00\",\"mainEntityOfPage\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/\"},\"wordCount\":2387,\"commentCount\":5,\"publisher\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage\"},\"thumbnailUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"keywords\":[\"API\"],\"articleSection\":[\"Node.js\",\"SQL++ \/ N1QL Query\"],\"inLanguage\":\"ko-KR\",\"potentialAction\":[{\"@type\":\"CommentAction\",\"name\":\"Comment\",\"target\":[\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#respond\"]}]},{\"@type\":\"WebPage\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/\",\"url\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/\",\"name\":\"Node.js Game Server Framework & Development Architecture\",\"isPartOf\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#website\"},\"primaryImageOfPage\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage\"},\"thumbnailUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"datePublished\":\"2015-07-16T17:00:00+00:00\",\"dateModified\":\"2025-06-14T06:47:49+00:00\",\"description\":\"This article explains development of the latest Node.js game server and Express Framework standards as well as the latest Node.js SDK version of Couchbase.\",\"breadcrumb\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#breadcrumb\"},\"inLanguage\":\"ko-KR\",\"potentialAction\":[{\"@type\":\"ReadAction\",\"target\":[\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/\"]}]},{\"@type\":\"ImageObject\",\"inLanguage\":\"ko-KR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage\",\"url\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"contentUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png\",\"width\":1800,\"height\":630},{\"@type\":\"BreadcrumbList\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#breadcrumb\",\"itemListElement\":[{\"@type\":\"ListItem\",\"position\":1,\"name\":\"Home\",\"item\":\"https:\/\/www.couchbase.com\/blog\/\"},{\"@type\":\"ListItem\",\"position\":2,\"name\":\"Making a Game API Server Using Node.js: Revisited\"}]},{\"@type\":\"WebSite\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#website\",\"url\":\"https:\/\/www.couchbase.com\/blog\/\",\"name\":\"The Couchbase Blog\",\"description\":\"Couchbase, the NoSQL Database\",\"publisher\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\"},\"potentialAction\":[{\"@type\":\"SearchAction\",\"target\":{\"@type\":\"EntryPoint\",\"urlTemplate\":\"https:\/\/www.couchbase.com\/blog\/?s={search_term_string}\"},\"query-input\":{\"@type\":\"PropertyValueSpecification\",\"valueRequired\":true,\"valueName\":\"search_term_string\"}}],\"inLanguage\":\"ko-KR\"},{\"@type\":\"Organization\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#organization\",\"name\":\"The Couchbase Blog\",\"url\":\"https:\/\/www.couchbase.com\/blog\/\",\"logo\":{\"@type\":\"ImageObject\",\"inLanguage\":\"ko-KR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/\",\"url\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png\",\"contentUrl\":\"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png\",\"width\":218,\"height\":34,\"caption\":\"The Couchbase Blog\"},\"image\":{\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/\"}},{\"@type\":\"Person\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/bb545ebe83bb2d12f91095811d0a72e1\",\"name\":\"Nic Raboy, Developer Advocate, Couchbase\",\"image\":{\"@type\":\"ImageObject\",\"inLanguage\":\"ko-KR\",\"@id\":\"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/image\/8863514d8bed0cf6080f23db40e00354\",\"url\":\"https:\/\/secure.gravatar.com\/avatar\/bedeb68368d4681aca4c74fe5f697f0c423b80d498ec50fd915ba018b72c101f?s=96&d=mm&r=g\",\"contentUrl\":\"https:\/\/secure.gravatar.com\/avatar\/bedeb68368d4681aca4c74fe5f697f0c423b80d498ec50fd915ba018b72c101f?s=96&d=mm&r=g\",\"caption\":\"Nic Raboy, Developer Advocate, Couchbase\"},\"description\":\"Nic Raboy is an advocate of modern web and mobile development technologies. He has experience in Java, JavaScript, Golang and a variety of frameworks such as Angular, NativeScript, and Apache Cordova. Nic writes about his development experiences related to making web and mobile development easier to understand.\",\"sameAs\":[\"https:\/\/www.thepolyglotdeveloper.com\",\"https:\/\/www.facebook.com\/thepolyglotdeveloper\",\"https:\/\/x.com\/nraboy\"],\"url\":\"https:\/\/www.couchbase.com\/blog\/ko\/author\/nic-raboy-2\/\"}]}<\/script>\n<!-- \/ Yoast SEO Premium plugin. -->","yoast_head_json":{"title":"Node.js Game Server Framework & Development Architecture","description":"\uc774 \ubb38\uc11c\uc5d0\uc11c\ub294 \ucd5c\uc2e0 Node.js \uac8c\uc784 \uc11c\ubc84 \ubc0f Express \ud504\ub808\uc784\uc6cc\ud06c \ud45c\uc900\uacfc \ucd5c\uc2e0 Node.js SDK \ubc84\uc804\uc758 Couchbase \uac1c\ubc1c\uc5d0 \ub300\ud574 \uc124\uba85\ud569\ub2c8\ub2e4.","robots":{"index":"index","follow":"follow","max-snippet":"max-snippet:-1","max-image-preview":"max-image-preview:large","max-video-preview":"max-video-preview:-1"},"canonical":"https:\/\/www.couchbase.com\/blog\/ko\/making-a-game-api-server-using-nodejs-revisited\/","og_locale":"ko_KR","og_type":"article","og_title":"Making a Game API Server Using Node.js: Revisited","og_description":"This article explains development of the latest Node.js game server and Express Framework standards as well as the latest Node.js SDK version of Couchbase.","og_url":"https:\/\/www.couchbase.com\/blog\/ko\/making-a-game-api-server-using-nodejs-revisited\/","og_site_name":"The Couchbase Blog","article_author":"https:\/\/www.facebook.com\/thepolyglotdeveloper","article_published_time":"2015-07-16T17:00:00+00:00","article_modified_time":"2025-06-14T06:47:49+00:00","author":"Nic Raboy, Developer Advocate, Couchbase","twitter_card":"summary_large_image","twitter_creator":"@nraboy","twitter_misc":{"Written by":"Nic Raboy, Developer Advocate, Couchbase","Est. reading time":"14\ubd84"},"schema":{"@context":"https:\/\/schema.org","@graph":[{"@type":"Article","@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#article","isPartOf":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/"},"author":{"name":"Nic Raboy, Developer Advocate, Couchbase","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/bb545ebe83bb2d12f91095811d0a72e1"},"headline":"Making a Game API Server Using Node.js: Revisited","datePublished":"2015-07-16T17:00:00+00:00","dateModified":"2025-06-14T06:47:49+00:00","mainEntityOfPage":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/"},"wordCount":2387,"commentCount":5,"publisher":{"@id":"https:\/\/www.couchbase.com\/blog\/#organization"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage"},"thumbnailUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","keywords":["API"],"articleSection":["Node.js","SQL++ \/ N1QL Query"],"inLanguage":"ko-KR","potentialAction":[{"@type":"CommentAction","name":"Comment","target":["https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#respond"]}]},{"@type":"WebPage","@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/","url":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/","name":"Node.js Game Server Framework & Development Architecture","isPartOf":{"@id":"https:\/\/www.couchbase.com\/blog\/#website"},"primaryImageOfPage":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage"},"thumbnailUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","datePublished":"2015-07-16T17:00:00+00:00","dateModified":"2025-06-14T06:47:49+00:00","description":"\uc774 \ubb38\uc11c\uc5d0\uc11c\ub294 \ucd5c\uc2e0 Node.js \uac8c\uc784 \uc11c\ubc84 \ubc0f Express \ud504\ub808\uc784\uc6cc\ud06c \ud45c\uc900\uacfc \ucd5c\uc2e0 Node.js SDK \ubc84\uc804\uc758 Couchbase \uac1c\ubc1c\uc5d0 \ub300\ud574 \uc124\uba85\ud569\ub2c8\ub2e4.","breadcrumb":{"@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#breadcrumb"},"inLanguage":"ko-KR","potentialAction":[{"@type":"ReadAction","target":["https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/"]}]},{"@type":"ImageObject","inLanguage":"ko-KR","@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#primaryimage","url":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","contentUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/sites\/1\/2022\/11\/couchbase-nosql-dbaas.png","width":1800,"height":630},{"@type":"BreadcrumbList","@id":"https:\/\/www.couchbase.com\/blog\/making-a-game-api-server-using-nodejs-revisited\/#breadcrumb","itemListElement":[{"@type":"ListItem","position":1,"name":"Home","item":"https:\/\/www.couchbase.com\/blog\/"},{"@type":"ListItem","position":2,"name":"Making a Game API Server Using Node.js: Revisited"}]},{"@type":"WebSite","@id":"https:\/\/www.couchbase.com\/blog\/#website","url":"https:\/\/www.couchbase.com\/blog\/","name":"\uce74\uc6b0\uce58\ubca0\uc774\uc2a4 \ube14\ub85c\uadf8","description":"NoSQL \ub370\uc774\ud130\ubca0\uc774\uc2a4, Couchbase","publisher":{"@id":"https:\/\/www.couchbase.com\/blog\/#organization"},"potentialAction":[{"@type":"SearchAction","target":{"@type":"EntryPoint","urlTemplate":"https:\/\/www.couchbase.com\/blog\/?s={search_term_string}"},"query-input":{"@type":"PropertyValueSpecification","valueRequired":true,"valueName":"search_term_string"}}],"inLanguage":"ko-KR"},{"@type":"Organization","@id":"https:\/\/www.couchbase.com\/blog\/#organization","name":"\uce74\uc6b0\uce58\ubca0\uc774\uc2a4 \ube14\ub85c\uadf8","url":"https:\/\/www.couchbase.com\/blog\/","logo":{"@type":"ImageObject","inLanguage":"ko-KR","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/","url":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png","contentUrl":"https:\/\/www.couchbase.com\/blog\/wp-content\/uploads\/2023\/04\/admin-logo.png","width":218,"height":34,"caption":"The Couchbase Blog"},"image":{"@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/logo\/image\/"}},{"@type":"Person","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/bb545ebe83bb2d12f91095811d0a72e1","name":"Nic Raboy, \uac1c\ubc1c\uc790 \uc639\ud638\uc790, Couchbase","image":{"@type":"ImageObject","inLanguage":"ko-KR","@id":"https:\/\/www.couchbase.com\/blog\/#\/schema\/person\/image\/8863514d8bed0cf6080f23db40e00354","url":"https:\/\/secure.gravatar.com\/avatar\/bedeb68368d4681aca4c74fe5f697f0c423b80d498ec50fd915ba018b72c101f?s=96&d=mm&r=g","contentUrl":"https:\/\/secure.gravatar.com\/avatar\/bedeb68368d4681aca4c74fe5f697f0c423b80d498ec50fd915ba018b72c101f?s=96&d=mm&r=g","caption":"Nic Raboy, Developer Advocate, Couchbase"},"description":"\ub2c9 \ub77c\ubcf4\uc774\ub294 \ucd5c\uc2e0 \uc6f9 \ubc0f \ubaa8\ubc14\uc77c \uac1c\ubc1c \uae30\uc220\uc744 \uc639\ud638\ud558\ub294 \uc0ac\ub78c\uc785\ub2c8\ub2e4. \uadf8\ub294 Java, JavaScript, Golang \ubc0f Angular, NativeScript, Apache Cordova\uc640 \uac19\uc740 \ub2e4\uc591\ud55c \ud504\ub808\uc784\uc6cc\ud06c\uc5d0 \ub300\ud55c \uacbd\ud5d8\uc774 \uc788\uc2b5\ub2c8\ub2e4. Nic\uc740 \uc6f9 \ubc0f \ubaa8\ubc14\uc77c \uac1c\ubc1c\uc744 \ubcf4\ub2e4 \uc27d\uac8c \uc774\ud574\ud560 \uc218 \uc788\ub3c4\ub85d \uc790\uc2e0\uc758 \uac1c\ubc1c \uacbd\ud5d8\uc5d0 \ub300\ud574 \uae00\uc744 \uc4f0\uace0 \uc788\uc2b5\ub2c8\ub2e4.","sameAs":["https:\/\/www.thepolyglotdeveloper.com","https:\/\/www.facebook.com\/thepolyglotdeveloper","https:\/\/x.com\/nraboy"],"url":"https:\/\/www.couchbase.com\/blog\/ko\/author\/nic-raboy-2\/"}]}},"authors":[{"term_id":9032,"user_id":63,"is_guest":0,"slug":"nic-raboy-2","display_name":"Nic Raboy, Developer Advocate, Couchbase","avatar_url":"https:\/\/secure.gravatar.com\/avatar\/bedeb68368d4681aca4c74fe5f697f0c423b80d498ec50fd915ba018b72c101f?s=96&d=mm&r=g","author_category":"","last_name":"Raboy","first_name":"Nic","job_title":"","user_url":"https:\/\/www.thepolyglotdeveloper.com","description":"\ub2c9 \ub77c\ubcf4\uc774\ub294 \ucd5c\uc2e0 \uc6f9 \ubc0f \ubaa8\ubc14\uc77c \uac1c\ubc1c \uae30\uc220\uc744 \uc639\ud638\ud558\ub294 \uc0ac\ub78c\uc785\ub2c8\ub2e4. \uadf8\ub294 Java, JavaScript, Golang \ubc0f Angular, NativeScript, Apache Cordova\uc640 \uac19\uc740 \ub2e4\uc591\ud55c \ud504\ub808\uc784\uc6cc\ud06c\uc5d0 \ub300\ud55c \uacbd\ud5d8\uc774 \uc788\uc2b5\ub2c8\ub2e4. Nic\uc740 \uc6f9 \ubc0f \ubaa8\ubc14\uc77c \uac1c\ubc1c\uc744 \ubcf4\ub2e4 \uc27d\uac8c \uc774\ud574\ud560 \uc218 \uc788\ub3c4\ub85d \uc790\uc2e0\uc758 \uac1c\ubc1c \uacbd\ud5d8\uc5d0 \ub300\ud574 \uae00\uc744 \uc4f0\uace0 \uc788\uc2b5\ub2c8\ub2e4."}],"_links":{"self":[{"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/posts\/2076","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/users\/63"}],"replies":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/comments?post=2076"}],"version-history":[{"count":0,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/posts\/2076\/revisions"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/media\/13873"}],"wp:attachment":[{"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/media?parent=2076"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/categories?post=2076"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/tags?post=2076"},{"taxonomy":"author","embeddable":true,"href":"https:\/\/www.couchbase.com\/blog\/ko\/wp-json\/wp\/v2\/ppma_author?post=2076"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}