Skip to content

搭建 Koa 服务器

一、使用脚手架初始化项目

脚手架

koa-generator 是一个 Koa 的脚手架工具。

js
// 1. 安装脚手架
$ npm install -g koa-generator

// 2. 通过脚手架创建初始化项目
koa -e project          // 生成 koa 项目(koa版本为1.x)
// OR
koa2 -e koa2-project    // 生成 koa 项目(koa版本为2.x)

// 3. 进入项目
cd project  // OR: cd koa2-project
npm install

// 4. 启动项目
npm run dev

出现如图所示,则表示项目初始化 OK: An image

注意

本项目 DEMO 基于 koa2,也就是说运行的命令是「 koa2 -e koa2-project 」。

二、优化配置:

js
// ====== 项目初始化配置目录
bin
public
routes
index.js
user.js
views
error.ejs
index.ejs
app.js
package.json

// ====== 调整后的目录结构
bin
conf // add
db.js // 定义数据库连接配置信息
mysql.js // 定义数据库连接
constants.js // 定义全局常量
controller // add
user.js // 定义登录请求的查询语句信息
utils // add
cryp.js // 加密工具
model // add
resModel.js // 设置响应的通用model
public
routes
index.js
user.js
views
error.ejs
index.ejs
app.js
package.json

上述文件夹和文件定义好后,需要安装对应的依赖,并完善文件代码内容 第一步: 安装依赖

js
// 1. 安装依赖
$ npm i mysql xss jsonwebtoken

第二步:完善文件代码

2.1 conf/db.js

js
// 2.1 conf/db.js
const env = process.env.NODE_ENV
let MYSQL_CONF
if (env === 'dev') {
  MYSQL_CONF = {
    host: 'localhost',
    user: 'root',
    password: 'password',
    port: '3306',
    database: 'yuanbo_web'
  }
}
if (env === 'production') {
  MYSQL_CONF = {
    // ...
  }
}
module.exports = {
  MYSQL_CONF
}

2.2 conf/mysql.js

js
// 2.2 conf/mysql.js
const mysql = require('mysql')
const { MYSQL_CONF } = require('./db')
const con = mysql.createConnection(MYSQL_CONF)
// 开始连接数据库
con.connect()
function exec(sql) {
  const promise = new Promise((resolve, reject) => {
    con.query(sql, (err, result) => {
      if (err) {
        reject(err)
        return
      }
      resolve(result)
    })
  })
  return promise
}
module.exports = {
  exec,
  escape: mysql.escape
}

2.3 conf/constants.js

js
// 2.3 conf/constants.js
const SECRET_KEY = 'bobo_1022#'
module.exports = {
  SECRET_KEY
}

2.4 controller/user.js

js
// 2.4 controller/user.js
const { exec, escape } = require('../conf/mysql')
const { genPassword } = require('../utils/cryp')

// 登录
const logIn = async (loginData) => {
  const username = escape(loginData.username)
  // 生成加密密码
  let password = genPassword(loginData.password)
  password = escape(password)
  const sql = `SELECT username, realname FROM users WHERE username=${username} AND password=${password}`
  const rows = await exec(sql)
  return rows[0] || {}
}
// 退出登录
const logOut = async (logoutData = {}) => {
  return true
}
module.exports = {
  logIn,
  logOut
}

2.5 utils/cryp.js

js
// 2.5 utils/cryp.js
const crypto = require('crypto')
const { SECRET_KEY } = require('../conf/constants')

function md5(content) {
  let md5 = crypto.createHash('md5')
  return md5.update(content).digest('hex')
}

function genPassword(passsword) {
  const str = `password=${passsword}&key=${SECRET_KEY}`
  return md5(str)
}

module.exports = {
  genPassword
}

2.6 model/resModel.js

js
// 2.6 model/resModel.js
class BaseModel {
  constructor(data, message) {
    if (typeof data === 'string') {
      this.message = data
      data = null
      message = null
    }
    if (data) {
      this.data = data
    }
    if (message) {
      this.message = message
    }
  }
}

class SuccessModel extends BaseModel {
  constructor(data, message) {
    super(data, message)
    this.code = 0
  }
}

class ErrorModel extends BaseModel {
  constructor(data, message) {
    super(data, message)
    this.code = -1
  }
}

module.exports = {
  SuccessModel,
  ErrorModel
}

三、完善请求路由(以 /login 为例):

注意

/users/login 以 mock 模拟方式处理,可以通过 Postman 来验证模拟的请求结果。

js
// routes/user.js
const router = require('koa-router')()
const { logIn, logOut } = require('../controller/user')
const { SuccessModel, ErrorModel } = require('../model/resModel')
// const loginCheck = require('../middleware/loginCheck')
const jwt = require('jsonwebtoken')
const { SECRET_KEY } = require('../conf/constants')

router.prefix('/users')

// 登录
router.post('/login', async function (ctx, next) {
  const body = ctx.request.body
  if (!body.username || !body.password) {
    ctx.body = new ErrorModel('字段缺失')
  } else {
    // ====== mock 模拟方式
    let payload = { time: new Date().getTime(), timeout: 1000 * 60 * 60 * 2 }
    const token = jwt.sign(payload, SECRET_KEY, { expiresIn: '7d' })

    ctx.body = new SuccessModel({
      username: body.username,
      msg: '恭喜您,成功登录',
      token
    })

    // ====== 调用 mysql 方式
    // const data = await logIn(body)
    // if (data.username) {
    //   let payload = {time:new Date().getTime(), timeout: 1000 * 60 * 60 * 2}
    //   const token = jwt.sign(payload, SECRET_KEY, {expiresIn: '7d'})
    //   data.token = token
    //   // 设置session
    //   ctx.session.username = data.username
    //   ctx.session.realname = data.realname
    //   ctx.body = new SuccessModel(data)
    //   return
    // }
    // ctx.body = new ErrorModel('登录失败')
  }
})
// 退出登录
router.post('/logout', async function (ctx, next) {
  const body = ctx.request.body
  const data = await logOut(body)
  ctx.session = null
  ctx.body = new SuccessModel(data)
})

module.exports = router

An image

如果没有传递完整的字段值:

An image

DEMO 源码:
DEMO -- Koa2