Technology:
- NodeJs 17.6.0
- Express 4.18.1
- cors 2.8.5
- crypto-js 4.1.1 # 加解密套件
- jsonwebtoken 8.5.1 # Json Web Token 的功能套件
- sequelize 6.20.2 # ORM 套件
- mysql2 2.3.3 # MySQL client for Node.js
- MySQL 8.0 # 使用的資料庫
專案完成後的檔案結構
./專案目錄
├── app/
│ ├── config/
│ │ └── db.config.js
│ ├── middleware/
│ │ ├── auth.jwt.js
│ │ ├── index.js
│ │ └── verify.signup.js
│ ├── models/
│ │ ├── index.js
│ │ ├── role.model.js
│ │ ├── todo.model.js
│ │ └── user.model.js
│ ├── routes/
│ │ ├── auth.routes.js
│ │ ├── todo.routes.js
│ │ └── user.routes.js
│ └── services/
│ ├── auth.service.js
│ ├── todo.service.js
│ └── user.service.js
├── node_modules/
├── .env
├── .gitignore
├── package.json
├── README.md
├── server.js
└── yarn-lock
專案完成後所提供的 API 端點
Methods | Urls | Actions |
---|---|---|
POST | /api/auth/signup | 註冊新使用者帳號 |
POST | /api/auth/signin | 使用者帳號登入 |
GET | /api/todos | get all Todos |
GET | /api/todos/:id | get Todo by id |
GET | /api/todos/done | find all done Todos |
GET | /api/todos/title=[keyword ] |
find all Todos whick title contains keyword |
POST | /api/todos | add New Todo |
PUT | /api/todos/:id | update Todo by id |
DELETE | /api/todos/:id | remove Todo by id |
DELETE | /api/todos | remove all Todos |
設置專案環境
$ node --version # 檢測環境已裝妥 node.js (若已安裝會顯示目前安裝的版本)
v17.6.0
$ mkdir nodejs-webapi-jwt-mysql && cd nodejs-webapi-jwt-mysql # 建立一個專案目錄
$ npm init # 産生一專案設定檔 package.json
$ touch server.js # 産生一個新檔案
package.json 預設的內容
{
"name": "nodejs-webapi-jwt-mysql",
"version": "1.0.0",
"description": "Node.js + express + JWT + MySQL",
"main": "server.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [
"node.js",
"express",
"jwt",
"mysql"
],
"author": "calvin",
"license": "ISC"
}
安裝相依套件
$ yarn add express dotenv sequelize mysql2 # 加入相依套件
$ yarn add --dev nodemon # 加入開發時期相依套件
$ yarn add crypto-js jsonwebtoken # 使用者密碼加解密 / Json Web Token
$ yarn add cors # 可跨網域提供 Webapi service
建立 git 初始版本
$ git init
$ echo 'node_modules/' > .gitignore # 新增 git ignore 設定檔,並設定 node_modules/ 目錄不加入版控
$ echo 'yarn.lock' >> .gitignore
$ git add . && git commit -m "Initial commit" 建立 git 初始版本的資訊
開啟 VSCode
$ code .
設定專案啟動指令(如第七行的指令設定),當輸入 npm start 時系統自動以 node 來執行 server.js 程式,並即時監測 server.js 檔案有變化存檔時馬上重新啟動 node server.js 來執行該程式。
|
|
在 server.js 中加入此行程式,並在 vscode Terminal 中輸入 npm start
console.log("Hi NodeJS...")
顯示結果如下
> nodejs-webapi-jwt-mysql@1.0.0 start
> nodemon server.js
[nodemon] 2.0.18
[nodemon] to restart at any time, enter `rs`
[nodemon] watching path(s): *.*
[nodemon] watching extensions: js,mjs,json
[nodemon] starting `node server.js`
Hi NodeJS...
[nodemon] clean exit - waiting for changes before restart
建立一個 Express 應用程式
使用以下程式覆蓋 server.js 檔案
const express = require("express");
const app = express();
app.get('/', function (req, res) {
res.send('Hello World')
});
app.get("/api/test", (req, res) => {
console.log("test is successful");
res.send("test is successful");
});
const PORT = process.env.PORT || 5000;
app.listen(PORT, () => {
console.log(`Backend server is running on port ${PORT}`);
})
存檔後,開啟 瀏覽器,輸入 localhost:5000/api/test,在 vscode terminal 視窗中會顯示:
Backend server is running on port 5000
test is successful
使用 Node.js 連結 MySQL
若你還沒有安裝 MySQL 可以參考這筆記先將資料庫管理系統備妥 使用 Docker 執行 MySQL
在程式中使用了 sequelize 這個套件的功能來連結 mysql 資料庫,直接透過 Sequelize constructor 來給定連結資料庫的參數,包含 “資料庫名稱“、”User Id"、“Password”、“Host Name"、“資料庫類別“(dialect)等。
|
|
存檔後,在 vscode terminal 視窗中會顯示:
Connection has been established successfully.
Backend server is running on port 5000
表示連結成功。
專案目錄
接下來要陸續完成相關程式,為使程式架構更顯清晰,專案目錄規劃如下:
./專案目錄
├── app/ # 程式目錄
│ ├── config/ # 設置連結 MySQL 資料庫的參數
│ ├── middleware/
│ ├── models/
│ ├── routes/
│ └── services/ # 業務邏輯
├── node_modules/
├── .env # 程式中的相關“設定值”
├── .gitignore
├── package.json
├── README.md
├── server.js # 主程式
└── yarn.loc
彈性管理參數值
管理程式中的“資料庫連結設定值”
在 app/config/目錄中新增一支 db.config.js 程式內容如下:
module.exports = {
HOST: "localhost", // Host Name
USER: "mysql", // User Name
PASSWORD: "12345", // Password
DB: "testdb", // Database Name
dialect: "mysql", // 資料庫類別
pool: {
max: 5, // 連結池中最大的 connection 數
min: 0,
acquire: 30000, // 連結 Timeout 時間(毫秒)
idle: 10000 // 連結被釋放的 idle 時間(毫秒)
}
};
管理程式中的“設定值”
使用 dotenv 套件的功能來管理程式中相關的設定值
- 首先在專案根目錄下建立一個名為 “.env” 的檔案,內容如下:
JWT_SEC=Jason-Web-Token-Secret-key-jaslkdjfhjwkej01kd1954
- 在程式中先匯入 dotenv 套件(第1行),再“啟動它”(第2行),使用時透過 “process.env.JWT_SEC” 語法取得 JWT_SEC 的設定值
- 第四行程式碼中的 process.env.PORT 為相同的原則可在 .evn 檔案中加入 PORT 的設定值調整
const dotenv = require("dotenv");
dotenv.config();
// ...
const PORT = process.env.PORT || 5000; // port 預設為 5000 ,並可以在 .env 檔案中進行客製化 (如:PORT=5001)
app.listen(PORT, () => {
console.log(`Backend server is running on port ${PORT}`);
})
定義 Sequelize 模型
在 app/models/目錄中新增一支 user.model.js 程式內容如下,這個設定資料可搭合 sequelize.sync() 功能自動在 MySQL 資料庫中建立一個名為 users 的資料表(table),共有三個皆為 string 型態的欄位,分別為 username、email、password。
module.exports = (sequelize, Sequelize) => {
const User = sequelize.define("users", {
username: {
type: Sequelize.STRING
},
email: {
type: Sequelize.STRING
},
password: {
type: Sequelize.STRING
}
});
return User;
};
同時在 initial Sequelize 後,我們不需要編寫 CRUD 函數,Sequelize 支持所有這些函數
- 建立一個新的 user: create(object)
- 透過 id 找到一個 user: findByPk(id)
- 透過 email 找到一個 user: findOne({ where: { email: … } })
- 取得所有使用者: findAll()
- 透過 username 找到符合的 user: findAll({ where: { username: … } })
在 app/models/目錄中新增一支 role.model.js 程式內容如下
module.exports = (sequelize, Sequelize) => {
const Role = sequelize.define("roles", {
id: {
type: Sequelize.INTEGER,
primaryKey: true
},
name: {
type: Sequelize.STRING
}
});
return Role;
};
在 app/models/目錄中新增一支 role.model.js 程式內容如下
module.exports = (sequelize, Sequelize) => {
const Role = sequelize.define("roles", {
id: {
type: Sequelize.INTEGER,
primaryKey: true
},
name: {
type: Sequelize.STRING
}
});
return Role;
};
初始化 Sequelize
在 app/models/目錄中新增一支 index.js 程式內容如下
const config = require("../config/db.config.js"); // 引入資料庫連結設定檔
const Sequelize = require("sequelize");
const sequelize = new Sequelize( // 由資料庫連結設定檔的設定值來備置 Sequelize
config.DB,
config.USER,
config.PASSWORD,
{
host: config.HOST,
dialect: config.dialect,
operatorsAliases: 0,
pool: {
max: config.pool.max,
min: config.pool.min,
acquire: config.pool.acquire,
idle: config.pool.idle
}
}
);
const db = {};
db.Sequelize = Sequelize;
db.sequelize = sequelize;
db.user = require("../models/user.model.js")(sequelize, Sequelize);
db.role = require("../models/role.model.js")(sequelize, Sequelize);
// 設定兩資料表的對應關係(多對多,所以會多出一個新的表 user_roles)
// 一個使用者可能有多個角色
// 一個角色也可能有多個使用者
db.role.belongsToMany(db.user, {
through: "user_roles",
foreignKey: "roleId",
otherKey: "userId"
});
db.user.belongsToMany(db.role, {
through: "user_roles",
foreignKey: "userId",
otherKey: "roleId"
});
db.ROLES = ["user", "admin"];
module.exports = db;
執行程式産生資料表與資料
在 server.js 主程式中加入以下程式
const db = require("./app/models"); // 引入 app/models/index.js 匯出的程式碼(即 sequelize model 定義檔)
const Role = db.role;
// 呼叫 sync function 將會依 model 定義內容産生資料表,force 參數值為 true 將會重建已存在的資料表
db.sequelize.sync({ force: true }).then(() => {
console.log('Drop and Resync Database with { force: true }');
initial(); // 産生資料表後,呼叫 initial function 為 roles table 新增二筆初始資料
}).catch((err) => {
console.log(err);
});
const PORT = process.env.PORT || 5000; // port 預設為 5000 ,並可以在 .env 檔案中進行客製化 (如:PORT=5001)
app.listen(PORT, () => {
console.log(`Backend server is running on port ${PORT}`);
})
// 為 roles table 新增二筆初始資料
function initial() {
Role.create({
id: 1,
name: "user"
});
Role.create({
id: 2,
name: "admin"
});
}
執行程式産生資料表
程式執行成功後可以查看資料庫已順利産生四個資料表以及 roles table 中的二筆初始資料
了解 Node.js 路由
建立 user.routes.js router file
新增 routes 目錄,在此目錄下新增 user.routes.js 檔案
routes/user.routes.js
const router = require("express").Router();
router.get("/usertest", (req, res) => {
res.send("user test is successful");
});
module.exports = router;
在 server.js 程式中先匯入 “./app/routes/user.routes” 這個 router 設定檔,再透過 app.use 語法來使用這個 router(第9行)。
|
|
開啟瀏覧器,輸入 http:5000/api/users/usertest,瀏覧器將呈現成功訊息
user test is successfull!
為 routes/user.routes.js 再新增一個 post method
|
|
使用 postman 來測試 post,結果回傳的是 Server Error,原因是 express 預設是不接受 json 格式的資料。
在 server.js 程式中加入如第一行的設定
|
|
設定完成後就可正常了
建立 auth.routes.js router file
將使用者資料註冊和資用者帳號驗證的機制獨立在這個 route file 中,讓程式結構更清晰。內容如下:
|
|
在 auth.routes.js route 程式中基於`關注點分離原則`把`商業邏輯`的部份再分離至 services 中。
在 ./app 目錄下新增 services 子目錄,並新增一支 auth.service.js 程式,將使用者註冊及登入邏輯放在這支 service 程式中,內容如下:
在這支程式中會使用到額外的套件,必須先進行安裝。
$ yarn add crypto-js jsonwebtoken
。其中 crypto-js 用來進行使用者密碼加解密(程式第2、15、50、51行),而 jsonwebtoken 套件則是支援 Json Web Token 功能(程式第55行使用 jwt.sign 産生合法 Token)。
// auth.service.js
const db = require("../models");
const CryptoJS = require("crypto-js");
const jwt = require("jsonwebtoken");
const User = db.user;
const Role = db.role;
const Op = db.Sequelize.Op;
const signup = (req, res) => {
// Save User to Database
User.create({
username: req.body.username,
email: req.body.email,
password: CryptoJS.AES.encrypt(req.body.password, process.env.PASS_SEC).toString(),
}).then(user => {
if (req.body.roles) {
Role.findAll({
where: {
name: {
[Op.or]: req.body.roles
}
}
}).then(roles => {
user.setRoles(roles).then(() => {
res.send({ message: "User registered successfully!" });
});
});
} else {
// user role = 1
user.setRoles([1]).then(() => {
res.send({ message: "User registered successfully!" });
});
}
}).catch(err => {
res.status(500).send({ message: err.message });
});
};
const signin = (req, res) => {
User.findOne({
where: {
username: req.body.username
}
}).then(user => {
if (!user) {
return res.status(404).send({ message: "Wrong Credentials." });
}
const hashedPassword = CryptoJS.AES.decrypt(user.password, process.env.PASS_SEC);
const orginalPassword = hashedPassword.toString(CryptoJS.enc.Utf8);
orginalPassword !== req.body.password && res.status(401).json("Wrong Credentials");
const accessToken = jwt.sign(
{id: user.id},
process.env.JWT_SEC,
{ expiresIn: "3d" }
);
var authorities = [];
user.getRoles().then(roles => {
for (let i = 0; i < roles.length; i++) {
authorities.push("ROLE_" + roles[i].name.toUpperCase());
}
res.status(200).send({
id: user.id,
username: user.username,
email: user.email,
roles: authorities,
accessToken: accessToken
});
});
}).catch(err => {
res.status(500).send({ message: err.message });
});
};
module.exports = { signup, signin };
在 server.js 引用這個新的 router
|
|
測試使用者註冊及登入功能
使用 postman 進行使用者註冊功能測試
註冊成功後,在資料庫中已新增一筆使用者資料
使用 postman 進行使用者登入功能測試
登入成功後會回傳一個 Token
在 Node.js 中使用 JWT 來進行 Token-Based 的使用者授權驗證
使用 jsonwebtoken
套件可以實現 Token-base 的身份驗證與授權讓我們的 API 程式更安全
JWT 實作的過程大致可以分成三個部分:
- 在登入成功後産生合法的 JWT Token
- 每次收到 request 時驗證是否為合法有效的 JWT Token
- 在特定 API Endpoint 上驗證是否帶有 “合法有效的 JWT Token”,以達到權限管理的需求
産生合法 JWT
在登入證驗中加入産生 Token 的邏輯,在檢核使用者輸入的密碼正確後,將 User ID (_id這個內部 Key)這個屬性值透過 sign function 來産生 access token,並回傳給前端。
|
|
encrypt function 參數除了要加密的字串外,需要一個加密 Key,為彈性起見,把它寫在 .env 檔案
|
|
使用 JWT 來驗證 Token
在前端取得合法的 JWT Token後,來看看當使用者在呼叫其他 API 時一併回傳的 Token 如何在 server 端來進行驗證。
首先,我們要在 app/ 目錄下新增一個 middleware/ 的子目錄,並在其中新增一個名為 auth.jwt.js 的 express Middleware,程式內容如下
在程式第14行中使用 jsonwebtoken 套件的 verify function 就是用來驗證 request 中的 Token 是否為合法 Token。驗證時一樣需要`加密 Key`來當參數。
在這程式中除了驗證 request 中是否有合法的 Token 外,還有其他授權檢核的邏輯:驗證是否為管理者、驗證是否為版主、驗證是否為管理者或是版主等。
|
|
為簡化 middleware 使用時的匯入路徑,我們在 middleware/ 的子目錄,再新增一個名為 index.js 的程式,內容如下
const authJwt = require("./auth.jwt");
module.exports = {
authJwt
};
在特定 API Endpoint 上驗證是否帶有 “合法有效的 JWT Token”
在前面完成了驗證 Token 的 Middleware 後,我們來看看如何在 router 中套用這些 middleware, 打開 routers/userroutes.js 程式檔,並將內容修改如下:
在程式第九行 post 的第二個參數,加入呼叫 authJwt.verifyToken 這個 middleware 驗證 token 的 function。
|
|
再次執行前面已執行過的 Postman 的 postusertest 這個 request,結果這次會回傳 `未授權`的警告訊息。
將登入成功時回傳的 token,加入到 Header 中,再執行一次 postusertest 即可執行成功。
若送出 request 中未包含 Token 則會回傳“未被授權執行本功能”。
若送出 request 中包含的是不合法的 Token 則會回傳“Token不合法“。
完成 todo list 相關功能
定義 todo 資料模型
在 app/models/目錄中新增一支 todo.model.js 程式內容如下
module.exports = (sequelize, Sequelize) => {
const Todo = sequelize.define("todo", {
title: {
type: Sequelize.STRING
},
description: {
type: Sequelize.STRING
},
status: {
type: Sequelize.BOOLEAN
}
});
return Todo;
};
修改 app/models/index.js
app/models/index.js 中加入 db.todo = require("../models/todo.model")(sequelize, Sequelize);
const config = require("../config/db.config");
const Sequelize = require("sequelize");
const sequelize = new Sequelize(
config.DB,
config.USER,
config.PASSWORD,
{
host: config.HOST,
dialect: config.dialect,
operatorsAliases: 0,
pool: {
max: config.pool.max,
min: config.pool.min,
acquire: config.pool.acquire,
idle: config.pool.idle
}
}
);
const db = {};
db.Sequelize = Sequelize;
db.sequelize = sequelize;
db.user = require("../models/user.model")(sequelize, Sequelize);
db.role = require("../models/role.model")(sequelize, Sequelize);
db.todo = require("../models/todo.model")(sequelize, Sequelize);
db.role.belongsToMany(db.user, {
through: "user_roles",
foreignKey: "roleId",
otherKey: "userId"
});
db.user.belongsToMany(db.role, {
through: "user_roles",
foreignKey: "userId",
otherKey: "roleId"
});
db.ROLES = ["user", "admin"];
module.exports = db;
新增 todo.service.js
將實際讀寫資料的動作寫在 service 中。
//./app/services.todo.service.js
const db = require("../models");
const Todo = db.todo;
const Op = db.Sequelize.Op;
const create = (req, res) => {
if (!req.body.title) {
res.status(400).send({
message: "內容不得為空白!"
});
return;
}
// 新增一個 Todo
const todo = {
title: req.body.title,
description: req.body.description,
status: req.body.status ? req.body.status : false
}
// 將 todo 存入資料庫
Todo.create(todo)
.then(data => {
res.send(data);
}).catch(err => {
res.status(500).send({
message: err.message || "資料存檔時發生錯誤!"
});
});
}
const findAll = (req, res) => {
const title = req.query.title;
let condition = title ? { title: { [Op.like]: `%${title}%` } } : null;
Todo.findAll({ where: condition })
.then(data => {
res.send(data);
})
.catch(err => {
res.status(500).send({
message: err.message || "由資料庫讀取 Todo 資料時發生錯誤!"
});
});
};
const findOne = (req, res) => {
const id = req.params.id;
Todo.findByPk(id)
.then(data => {
if (data) {
res.send(data)
} else {
res.status(400).send({
message: `使用id=${id}搜尋時找到不 Todo 資料!`
})
}
})
.catch(err => {
res.status(500).send({
message: `使用id=${id}搜尋時找到不 Todo 資料!`
});
});
};
const update = (req, res) => {
const id = req.params.id;
Todo.update(req.body, {
where: { id: id }
})
.then(num => {
if (num == 1) {
res.send({
mdssage: "Todo 更新完成!"
});
} else {
res.status(500).send({
message: `使用 id= ${id} 更新資料時發生錯誤!`
});
};
})
};
const deleteOne = (req, res) => {
const id = req.params.id;
Todo.destroy({
where: { id: id }
})
.then(num => {
if (num === 1) {
res.send({
message: "Todo 刪除成功!"
})
} else {
res.send({
message: `使用 id= ${id} 刪除 Todo 時未找到任何資料!`
})
};
})
.catch(err => {
res.status(500).send({
message: `使用 id= ${id} 刪除 Todo 時發生錯誤!`
})
})
};
const deleteAll = (req, res) => {
Todo.destroy({
where: {},
truncat: false
})
.then(nums => {
res.send({
message: `${nums} Todo 資料被刪除成功!`
})
})
.catch(err => {
res.status(500).send({
message: err.message || "刪除所有資料時發生錯誤!"
})
})
};
const findAllDoneTodos = (req, res) => {
Todo.findAll({ where: { status: true } })
.then(data => {
res.send(data);
})
.catch(err => {
res.status(500).send({
message: err.message || "讀取資料時發生錯誤!"
})
})
};
module.exports = { create, findAll, findOne, update, deleteAll, deleteOne, findAllDoneTodos };
新增 todo.router.js
// ./app/routers/todo.routes.js
const router = require("express").Router();
const todo = require("../services/todo.service");
router.post("/", todo.create);
router.get("/", todo.findAll);
router.get("/done", todo.findAllDoneTodos);
router.get("/:id", todo.findOne);
router.put("/:id", todo.update);
router.delete("/:id", todo.deleteOne);
router.delete("/", todo.deleteAll);
module.exports = router;
將 todo router 加入到 service.js
//...
app.use(express.json());
const userRoute = require("./app/routes/user.routes");
app.use("/api/users", userRoute);
const todoRoute = require("./app/routes/todo.routes");
app.use("/api/todos", todoRoute);
const authRoute = require("./app/routes/auth.routes");
app.use("/api/auth", authRoute);
const PORT = process.env.PORT || 5000;
app.listen(PORT, () => {
console.log(`Backend server is running on port ${PORT}`);
});
//...
執行測試
新增一筆 Todo
找出所有的 Todos
找出 title 中有含某個 keyword 的所有 Todos
以 ID 找出 Todo
更新某一個 Todo 的 status 欄位
查詢所有已經完成的 Todos
刪除某一個 Todo
刪除所有的 Todos
使用 middleware 來進行檢核使用者角色
將 app/routes/todo.routes.js 內容修改成如下:
- 第五行:create 功能加入 authJwt.verifyToken middleware 檢核,必須是已登入的使用者才能執行
- 第六行:findall 功能加入 authJwt.verifyToken 以及 authJwt.isAdmin middleware 檢核,必須是已登入的使用者且具有管理者身份才能執行
|
|
再次執行 create Todo API endpoint,發現未登入的使用者已無法執行。
以一般使用者進行登入
打開 Postman 先使用已註冊的使用者帳號登入,取得 token
再將這個 token (使表使用者是 tom) 加入 request authorization header 中,再次造再次執行 create Todo API endpoint,結果可正常新建一個 Todo
再以相同 token (代表使用者是 tom) 造訪 get All Todos API endpoint ,結果回傳 “須為管理者角色者"才能造訪。
註冊一個新使用者且具備有 admin 角色
改用新註冊的使用者 Jeff 來重新登入,並取得回傳的 token
改採此 token (代表使用者是 jeff) 再次造訪 get All Todos API endpoint ,結果顯示可正常顯示所有的 Todos。
使用 middleware 來進行其他資料檢核
程式至此已經可以透過 JWT 相關功能查核使用者是否已正確登入系統、是以何種身份(角色)登入的。
最後要再呈現的是使用 middleware 功能來查核其他資料正確性,如:使用者註冊時是否使用了相同的使用者名稱?是否 email 已經被其他使用者使用過?
在 app/middleware 目錄下新增 verify.signup.js,內容如下:
|
|
修改 app/middleware/index.js 檔案如下:
|
|
將新的 middleware 功能放入到 SignUp 功能中。 打開 app/routes/auth.routes.js 程式檔,修改如下:
|
|
先使用已經被註冊過的 EMail來進行測試,結果回傳 EMail 已被使用。
再使用已被註冊過的 UserName 來進行註冊,結果回傳:
cors
這個後端專案預計要給前端 Angular 來使用,跨域存取問題就用 cors 設定來解決。
先安裝 cors 套件
$ yarn add cors
yarn add v1.22.19
[1/4] Resolving packages...
[2/4] Fetching packages...
[3/4] Linking dependencies...
[4/4] Building fresh packages...
success Saved lockfile.
success Saved 2 new dependencies.
info Direct dependencies
└─ cors@2.8.5
info All dependencies
├─ cors@2.8.5
└─ object-assign@4.1.1
Done in 0.86s.
打開 server.js,修改如下:
|
|