使用typescript&koa&typeorm开发后端api服务器

查看完整项目 请移步至 github

joi, swagger, koa 通过 decorator 的方式连接起来, 这是此项目的开发初衷.

[x] koa生态系统
[x] jwt登录验证
[x] 自动生成swagger接口文档
[x] nodemon 自动重启
[x] 使用 decorator 的方式完成接口参数验证
[x] controller 错误自动捕捉
[x] typescript 支持
[x] 基于 typeorm 的数据库 orm 支持
[x] mysql 事务支持(隔离支持)
[ ] 根据typeorm实体自动生成swagger的definition

开始使用

定义 controller

应用中所有请求均为一个 js class, 加上 @controller 即可

import {
  controller,
} from "../decorators";

@controller('/example')
export default class ExampleController {
   /* whatever you like */
}

上述例子即可实现一个基础 controller 的定义, @controller 中参数即为此控制器对应的响应路径前缀.

当然要完成一个简单请求, 这是不够的, 因为还没有定义与之相关的处理方法

定义处理函数

应用包装了 @get, @post, @put, @delete, @update 等常用http方法, 我们只需要为上一步中 controllermethod 上面加上对应方法的装饰器即可.

import {
  get,
  controller,
} from "../decorators";

@controller('/example')
export default class ExampleController {
  @get('/hello')
  async Hello() {
    return 'hello world'
  }
}

现在, 切换到项目根目录, 执行 npm start, 打开浏览器在地址栏输入 http://localhost:3002/example/hello 即可.

处理响应

通过包装 koa 的响应方法, 现在我们只需要在 controller 的方法下面直接 return 需要响应的值即可, 装饰器会接收响应参数并返回到浏览器.

需要注意: 这意味着我们必须显示return一个值给装饰器

如以下行为是不被建议的:

import {
  get,
  controller,
} from "../decorators";

export default class ExampleController {
  @get('/hello')
  async Hello() {
     // bad 没有显示的return
  }

  @get('/hello')
  async Hello() {
     // bad 没有return任何有价值的东西
     return ;
  }

  @get('/hello')
  async Hello() {
     // bad 同上, 没有return任何有价值的东西
     return undefined;
  }
  
  @get('/hello')
  async Hello(ctx) {
     // bad 请不要直接使用ctx.body = anything; 这会被覆盖
     ctx.body = 'hello';
     return 'world';
  }

  @get('/hello')
  async Hello(ctx) {
     // bad 这样实际是可以运行的, 但是仍然不推荐使用
     ctx.body = 'hello';
  }

  @get('/hello')
  async Hello() {
     // good! nice!
     return = 'hello';
  }
}

处理请求

api 系统中, 参数不可避免, 而且在处理方法内部对参数进行校验这实际会写上很多的样板代码, 也影响业务逻辑.
因此, 我们采用 joi 进行参数验证.

import {
  get,
  parameter,
  controller,
} from "../decorators";
import * as joi from 'joi';
import { IContext } from "../decorators/interface";

@controller('/example')
export default class ExampleController {
  @get('/hello')
  @parameter('params', joi.object().keys({
    a: joi.number().required(),
    b: joi.string()
  }), ENUM_PARAM_IN.body)
  @parameter('userId', joi.number().integer().description('用户id').required(), ENUM_PARAM_IN.path)
  @parameter('param1', joi.string().description('其他参数').required(), ENUM_PARAM_IN.query)
  async Hello(ctx: IContext) {
       // IContext 为定义的一个ts类型, 扩展了 koa 的 ctx
       // 为ctx加上了 $getParams 方法, 方便获取验证成功后的请求参数
       return ctx.$getParams();
  }
}

如上: 通过 @parameter 装饰器我们可以很方便定义接口参数, 并借助 joi 的魔力对其进行验证
系统可以对 querystring, path, body 进行参数验证
ENUM_PARAM_IN 为系统定义的 enum, 包含 'path' | 'body' | 'query', 默认 query

处理错误

每个控制器均支持错误 Error Catch 的能力

我们可以直接在应用中使用 throw 抛出错误, 应用在方法外层会自动捕捉并返回给前端, 参考以下示例

import {
  get,
  controller,
} from "../decorators";
import { IContext } from "../decorators/interface";

@controller('/example')
export default class ExampleController {
  @get('/hello')
  async Hello() {
     // 抛出一个默认的 500 错误, error message会默认发送给前端
     throw new Error('just an error');
  }
  @get('/hello')
  async Hello(ctx: IContext) {
     // 通过ctx.throw 我们可以抛出一个自定义状态码的错误
     ctx.throw(400, 'just an error');
  }
}

生成 swagger

应用提供了 @description, @tag, @summary, @response 等装饰器来处理swagger的情况

import {
  post,
  tag,
  summary,
  response,
  controller,
} from "../decorators";
import * as omit from 'omit.js';
import * as joi from 'joi';
import * as jwt from "jsonwebtoken";
import { User } from '../entity/User';
import { AppConfig } from '../utils/config';
import UserSchema from "../definitions/User";
import { Like } from "typeorm";
import { IContext } from "../decorators/interface";

@controller('/example')
export default class ExampleController {
  @post('/login')
  @parameter(
    'body', 
    joi.object().keys({
      name: joi.string().required().description('用户名'),
      password: joi.string().required().description('密码'),
    }), ENUM_PARAM_IN.body
  )
  @description('用户登录例子')
  @tag('用户管理')
  @summary('用户登录')
  @response(200, {
    user: { $ref: UserSchema, desc: '用户信息' },
    token: joi.string().description('token, 需要每次在请求头或者cookie中带上'),
  })
  async login(ctx: IContext) {
    const { name, password }: User = ctx.$getParams();
    const user: User = await User.findOne({ name });
    if (!user || user.password !== password) {
      throw new Error('用户名密码不匹配');
    }
    const token = jwt.sign({
      data: user.id
    }, AppConfig.appKey, { expiresIn: 60 * 60 });
    ctx.cookies.set('token', token);
    return {
      token,
      user: omit(user, ['password'])
    };
  }
}

现在, 打开浏览器, 在地址栏输入 http://localhost:3002/docs, 即可查看生成的接口swagger文档

user: { $ref: UserSchema, desc: '用户信息' }, 这里面的 $ref, 即转换后的 swagger definition, 在./src/definitions目录下即可定义

使用数据库

应用使用 typeorm 来作为数据库的 orm 工具.

  1. 安装并登录 mysql 创建一个数据库
sudo apt install mysql-server mysql-client

mysql -u root -p

> create database test default charset=utf8;
  1. 编辑 ormconfig.js 文件, 修改数据库相关配置

  2. ./src/entity 目录下面定义 typeorm 实体, 并定义实体的相关属性, 详细配置可参考 typeorm文档

  3. controller 中导入上一步中定义的模型, 使用方式参考 ./src/controlls/user.ts


使用事务

我们在 ctx 中内置了 typeormmanager, 在控制器开始前开启一个 typeorm 的事务, 检测到应用内抛出的异常之后, 则自动回滚事务, 若应用正常被处理, 则自动提交事务.

注意: 因为需要检测应用内异常, 所以只能通过throw 方式抛出的异常才能被正确处理, 而不能使用ctx.throw

一个栗子:

  /**
   * 新增用户
   */
  @post('/add')
  @tag('用户管理')
  @parameter(
    'body', 
    joi.object().keys({
      name: joi.string().required().description('用户名'),
      password: joi.string().required().description('密码'),
    }), ENUM_PARAM_IN.body
  )
  @summary('添加管理员')
  @login_required()
  @response(200, { $ref: UserSchema })
  async addUser(ctx: IContext) {
    const userInfo: User = ctx.$getParams();
    const lastUser = await User.findOne({ name: userInfo.name });
    if (lastUser) {
      throw new Error('用户已存在');
    }
    const user = new User(userInfo);
    await ctx.manager.save(user);
    return omit(user, ['password']);
  }

登录验证

系统提供了 login_required 装饰器, 使用时加上即可, 栗子见上面

目录说明

.
├── ./ormconfig.js  // typeorm配置文件
├── ./package.json 
├── ./package-lock.json
├── ./readme.md // readme
├── ./src // 源码目录
│   ├── ./upload // 文件上传目录
│   ├── ./src/controllers // 控制器相关
│   │   ├── ./src/controllers/example.ts 
│   ├── ./src/decorators // decorators相关
│   │   ├── ./src/decorators/controller.ts
│   │   ├── ./src/decorators/definition.ts
│   │   ├── ./src/decorators/description.ts
│   │   ├── ./src/decorators/index.ts
│   │   ├── ./src/decorators/interface.ts
│   │   ├── ./src/decorators/ischema.ts
│   │   ├── ./src/decorators/login_required.ts
│   │   ├── ./src/decorators/method.ts
│   │   ├── ./src/decorators/parameter.ts
│   │   ├── ./src/decorators/response.ts
│   │   ├── ./src/decorators/summary.ts
│   │   ├── ./src/decorators/tag.ts
│   │   └── ./src/decorators/utils.ts
│   ├── ./src/definitions // 主要用于swagger中模型
│   │   ├── ./src/definitions/BaseSchema.ts
│   │   └── ./src/definitions/User.ts
│   ├── ./src/entity // 数据库表(typeorm实体)
│   │   ├── ./src/entity/BaseEntity.ts
│   │   └── ./src/entity/User.ts
│   ├── ./src/main.ts // 应用入口
│   └── ./src/utils // 工具目录
│       ├── ./src/utils/config.ts // 应用配置
│       ├── ./src/utils/JoiToSwagger.ts 
│       └── ./src/utils/middlewares.ts // koa middleware相关
├── ./tsconfig.json // ts配置
├── ./tslint.json
├── ./typings.json
└── ./yarn.lock

持续更新中, 更多功能请关注此仓库...

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容