路由和控制器

midway 使用 koa-router 作为路由的承载者,同时在 ts 的语法上做了一些简化,我们将路由和控制器放在了一起,使用装饰器来标注路由。

路由装饰器

在新的 ts 体系中,我们的控制器目录为 app/controller ,我们在其中编写 *.ts 文件。例如下面的 userController.ts ,我们提供了一个获取用户的接口。

  1. @provide()
  2. @controller('/user')
  3. export class UserController {
  4. @inject('userService')
  5. service: IUserService;
  6. @get('/:id')
  7. async getUser(ctx): Promise<void> {
  8. const id: number = ctx.params.id;
  9. const user: IUserResult = await this.service.getUser({id});
  10. ctx.body = {success: true, message: 'OK', data: user};
  11. }
  12. }

我们创建了 @controller 装饰器用来定义这个类为 Controller,同时,提供了方法装饰器用于标注请求的类型。

小贴士

便于大家理解,@controller 的参数为字符串 pattern,我们会将这个值传入 router.prefix(prefix) 的参数中。

midway 针对 web 请求,提供了和 koa-router 对应的方法装饰器,列表如下。

  • @get
  • @post
  • @del
  • @put
  • @patch
  • @options
  • @head
  • @all
    这几个装饰器用于修饰不同的异步方法,同时对应到了 koa-router 的相应的方法。和原有提供的控制器一样,每个控制器都为异步方法,参数为 koa 上下文。
  1. @get('/:id')
  2. async getUser(ctx, next): Promise<void> {
  3. // TODO ctx...
  4. }

路由绑定

在以往框架的写法中,提供的 app/router 文件,虽然可以直接使用,但是由于控制器被 IoC 管控的关系,会有一些区别。

和以往的写法不同的是,我们需要从容器中拿到对应的控制器实例,并绑定到路由的 pattern 上

假如我们有一个控制器,同时没有提供 @controller 装饰器,表明他不是一个控制器,在应用初始化时不会自动绑定到某个路由上,但是由于有 @provide 装饰器,他会被 IoC 容器所加载。

  1. // app/controller/api.ts
  2. @provide()
  3. export class BaseApi {
  4. async index(ctx) {
  5. ctx.body = 'index';
  6. }
  7. }

假如我们希望这个控制器可以被外部的路由使用。

  1. // app/router.ts
  2. module.exports = function(app) {
  3. app.get('/api/index', app.generateController('baseApi.index'));
  4. };

midway 扩展了一个 app.generateController 的方法来简化绑定的这个步骤,参数为 ClassName.methodName 的字符串形式。

路由优先级

在单页应用下,经常会出现 /* 这种路由,在原本的路由文件中,我们可以通过调整代码的顺序,来使的路由的匹配顺序发生变化。而由于使用了装饰器的关系,在新题的体系无法控制文件扫描和加载顺序,这就使得路由匹配的顺序不可控。

midway 提供了 @priority(priority: number) 装饰器,用于修饰 class,定义路由的优先级,默认的路由优先级为 0,可以设置负数让优先级降低。

  1. @provide()
  2. @priority(-1)
  3. @controller('/')
  4. export class HomeController {
  5. @get('/hello')
  6. async index(ctx) {
  7. ctx.body = 'hello';
  8. }
  9. @get('/*')
  10. async all(ctx) {
  11. ctx.body = 'world';
  12. }
  13. }