koa2+koa-generator+mysql快速搭建nodejs伺服器
阿新 • • 發佈:2021-02-02
# koa2+koa-generator+mysql快速搭建nodejs伺服器
> 用koa的腳手架koa-generator可以快速生成專案骨架,可以用於發開或者測試介面
> https://github.com/hellojinjin123/node-koa2-template
## 1. 全域性安裝koa-generator(不用全域性安裝koa)
專案名字fast-koa
```cmd
npm install koa-generator -g
koa2 fast-koa
cd fast-koa
npm install
```
> 目錄結構如下
-bin // www 專案啟動目錄 node ./www
-public // 靜態網站放置目錄 也就是vue dist程式碼放置的地 專案入口index.html
-routes // 路由
-views // 檢視 伺服器渲染使用的模板
-app.js // 專案入口
-packaga.json
## 2. 啟動專案
```json
// package.json
"scripts": {
"start": "node bin/www",
"dev": "./node_modules/.bin/nodemon bin/www",
"prd": "pm2 start bin/www",
"test": "echo \"Error: no test specified\" && exit 1"
}
```
執行npm run dev開啟伺服器
同時可以看到generator自帶了nodemon(Nodemon 是一款非常實用的工具,用來監控你 node.js 原始碼的任何變化和自動重啟你的伺服器)
如下圖:伺服器啟動了
![avatar1][base64str1]
## 3. 專案入口app.js
```js
// app.js
const Koa = require('koa')
const app = new Koa()
const views = require('koa-views')
const json = require('koa-json')
const onerror = require('koa-onerror')
const bodyparser = require('koa-bodyparser')
const logger = require('koa-logger')
const index = require('./routes/index')
const users = require('./routes/users')
// error handler
onerror(app)
// middlewares
app.use(bodyparser({
enableTypes:['json', 'form', 'text']
}))
app.use(json())
app.use(logger())
app.use(require('koa-static')(path.resolve(__dirname, config.publicPath))))
app.use(views(__dirname + '/views', {
extension: 'pug'
}))
// logger
app.use(async (ctx, next) => {
const start = new Date()
await next()
const ms = new Date() - start
console.log(`${ctx.method} ${ctx.url} - ${ms}ms`)
})
// routes
app.use(index.routes(), index.allowedMethods())
app.use(users.routes(), users.allowedMethods())
// error-handling
app.on('error', (err, ctx) => {
console.error('server error', err, ctx)
});
module.exports = app
```
可以在根目錄路新增config.js 把一些公共的配置放入 比如資料庫資訊,埠,靜態資源路徑等
```js
// config.js
const path = require('path');
const config = {
// 專案啟動監聽的埠
port: 3000,
publicPath: 'public',
logPath: 'logs/koa-template.log',
// 資料庫配置
database: {
HOST: 'xxx', // 資料庫地址
USERNAME: 'xxx', // 使用者名稱
PASSWORD: 'xxx', // 使用者密碼
DATABASE: 'xxx', // 資料庫名
PORT: 3306 // 資料庫埠(預設: 3306)
}
};
module.exports = config;
```
## 4. koa-static 靜態資源中介軟體
`app.use(require('koa-static')(path.resolve(__dirname, config.publicPath))))`
koa-generator已經配置了靜態資源中介軟體,只要放入public目錄,靜態網站就可以執行
瀏覽http://localhost:3000/,伺服器會優先讀取public下的index.html
如果沒有index.html,伺服器會根據路由,判斷'/'是否有內容返回,沒有對應路由則返回404 not found
因為koa-generator預設設定了路由,所以伺服器執行返回了`Hello Koa 2!`
如下:
```js
router.get('/', async (ctx, next) => {
await ctx.render('index', {
title: 'Hello Koa 2!'
})
})
```
## 5. 新增models目錄
相當於伺服器資料層,存放資料庫模型(相當於建表),使用Sequelize進行mysql操作
```js
const { sequelize, Sequelize } = require('../config/db')
const { DataTypes, Model } = Sequelize
class Admin extends Model {
/**
* @description: 新增管理員
* @param {*} username
* @param {*} password
* @return {*} 返回新增的資料
*/
static async createAdmin({ username, password }) {
return await this.create({
username,
password
})
}
/**
* @description: 根據id修改管理員密碼
* @param {*} id
* @return {*} 返回修改的資料
*/
static async updatepwdById({id, password}) {
return await this.update({ password }, {
where: {
id
}
})
}
/**
* @description: 根據id刪除管理員
* @param {*} id
* @return {*}
*/
static async deleteById(id){
return await this.destroy({
where: {
id
}
})
}
}
// 初始化表結構
Admin.init(
{
id: {
type: DataTypes.INTEGER,
allowNull: false, //非空
autoIncrement: true, //自動遞增
primaryKey: true //主鍵
},
username: {
type: DataTypes.STRING,
field: "username",
allowNull: false,
unique: true // 唯一約束 使用者名稱不能重複
},
password: {
type: DataTypes.STRING,
allowNull: false
},
active: {
type: DataTypes.BOOLEAN,
allowNull: false,
defaultValue: true
}
}, {
underscored: true, //額外欄位以下劃線來分割
timestamps: true, //取消預設生成的createdAt、updatedAt欄位
createdAt: "created_at",
updatedAt: "updated_at",
freezeTableName: true, // Model 對應的表名將與model名相同
comment: "管理員表類",
// paranoid: true //虛擬刪除
sequelize, // 我們需要傳遞連線例項
// modelName: 'Admin', // 我們需要選擇模型名稱
// tableName: 'Admin' // 表名
}
)
// 建立表格
; (async () => {
await Admin.sync();
console.log("Admin表剛剛(重新)建立!");
// 這裡是程式碼
})()
// 定義的模型是類本身
// console.log(User === sequelize.models.User); // true
module.exports = Admin
```
## 6. mysql資料庫的使用(Sequelize stars 23.6k in github )
Sequelize 是一個基於 promise 的 Node.js ORM, 目前支援 Postgres, MySQL, MariaDB, SQLite 以及 Microsoft SQL Server. 它具有強大的事務支援, 關聯關係, 預讀和延遲載入,讀取複製等功能。
Sequelize 遵從 語義版本控制。 支援 Node v10 及更高版本以便使用 ES6 功能。https://www.sequelize.com.cn/core-concepts/model-basics
安裝mysql&sequelize
```cmd
npm install --save mysql mysql2
npm install --save sequelize
```
安裝sequelize之後,在config目錄下建立db.js,進行資料庫連線設定
```js
const Sequelize = require('sequelize');
const db = require('./index').db
// 初始化資料庫
const sequelize = new Sequelize(db.database, db.username, db.password, {
host: db.host,
dialect: 'mysql',
pool: {
max: 5,
min: 0,
idle: 10000
}
})
//測試資料庫連結
sequelize.authenticate().then(function() {
console.log("資料庫連線成功");
}).catch(function(err) {
//資料庫連線失敗時列印輸出
console.error(err);
throw err;
});
module.exports = { sequelize, Sequelize }
```
## 7. 新增controllers目錄
有了模型層對操作資料庫的支援,就可以進行業務操作了,也就是控制器目錄(在這個層可以放心呼叫models層方法進行curd)
```js
// 匯入模型
const Admin = require('../models/admin')
module.exports = {
async getAllAdmins(ctx, next) {
try {
let data = await Admin.findAll()
ctx.body = { msg: 1001, data }
} catch (err) {
ctx.body = { code: -1, msg: 1000, }
}
await next();
},
async createAdmin(ctx, next) {
let req = ctx.request.body
if (!req.username || !req.password) {
ctx.body = { code: -1, msg: 1002 }
return await next();
}
try {
let data = await Admin.createAdmin(req)
ctx.body = { msg: 1003, data }
} catch (err) {
ctx.body = { code: -1, msg: 1000 }
}
await next();
},
async updatepwdById(ctx, next) {
let req = ctx.request.body
if (req.id && req.password) {
try {
await Admin.updatepwdById(req)
ctx.body = { msg: 1004 }
} catch (err) {
ctx.body = { code: -1, msg: 1000 }
}
} else {
ctx.body = { code: -1, msg: 1002 }
}
await next();
},
async deleteById(ctx, next) {
let query = ctx.request.query // 獲取get請求引數
if (query && query.id) {
try {
await Admin.deleteById(query.id)
ctx.body = { msg: 1005 }
} catch (err) {
ctx.body = { code: -1, msg: 1000 }
}
} else {
ctx.body = { code: -1, msg: 1002 }
}
await next();
}
}
```
## 8. 路由配置
```js
// app.js 中新增
// routes
const admin = require('./routes/admin')
app.use(admin.routes(), admin.allowedMethods())
```
到此為止,一個完整的請求(介面)就處理完成了
比如請求 http://localhost:3000/admin/getAllAdmins
koa經歷的簡單過程:
1. 瀏覽器發出請求 -> 中介軟體 ->路由中介軟體 -> 中介軟體 -> 中介軟體若干回撥 -> 瀏覽器收到響應
2. 路由:
- `router.get/post` -> `controllers.func` -> `models.func`-> mysql
- 請求行為 -> 業務邏輯 -> 模型支援 -> 入庫
## 9. 配置自定義的中介軟體處理日誌和response訊息
可以參考github程式碼
## 10. 附上一個很好理解的中介軟體原理的簡析
```js
// 根目錄下test.js
// koa2 中介軟體原理簡析
// 中介軟體的倉庫
const arr = [
async (next) => {
console.log(1);
await next();
console.log(2);
},
async (next) => {
console.log(3);
await new Promise((resolve, reject) => {
setTimeout(() => {
resolve(
console.log(4)
);
}, 3000);
}); // 非同步操作 await 會等待後面的promise resolve 後再向下執行
await next();
console.log(5);
},
async (next) => {
console.log(6);
},
async (next) => {
// 不會執行 因為上一個函式中沒有執行next
console.log(7);
await next();
console.log(8);
},
async (next) => {
// 不會執行 因為前面的函式中沒有執行next
console.log(9);
}
];
function fun(arr) {
function dispose(index) {
const currentFun = arr[index];
const next = dispose.bind(null, index + 1);
return currentFun(next); // 尾遞迴
}
dispose(0);
}
fun(arr); // 先列印 1 3 一秒後列印4 6 5 2
```
## 參考連結
- https://www.sequelize.com.cn/
- https://www.cnblogs.com/zjknb/p/12149420.html
[base64str1]: