1. 程式人生 > >koa2+koa-generator+mysql快速搭建nodejs伺服器

koa2+koa-generator+mysql快速搭建nodejs伺服器

# 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]: