全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

NodeJS仿WebApi路由示例

用过WebApi或Asp.net MVC的都知道微软的路由设计得非常好,十分方便,也十分灵活。虽然个人看来是有的太灵活了,team内的不同开发很容易使用不同的路由方式而显得有点混乱。 不过这不是重点,我在做Node项目的时候就觉得不停的用use(...)来指定路由路径很烦人,所以用Typescript写了这个基于KoaKoa-router的路由插件,可以简单实现一些类似WebApi的路由功能。

目标是和WebApi一样:

1.加入的controller会自动加入路由。

2.也可以通过path()手动指定路由。

3.可以定义http method, 如GETPOST等。

4.Api的参数可以指定url里的query param、path param以及body等。

包已经上传到npm中,npm install webapi-router 安装,可以先看看效果:

第一步,先设置controllers的目录和url的固定前缀

所有的controller都在这目录下,这样会根据物理路径自动算出路由。 url的固定前缀就是host和路由之间的,比如localhost/api/v2/user/nameapi/v2就是这个固定前缀。

import { WebApiRouter } from 'webapi-router';

app.use(new WebApiRouter().router('sample/controllers', 'api'));

第二步是controller都继承自BaseController

export class TestController extends BaseController
{

}

第三步给controller的方法加上装饰器

@POST('/user/:name')
postWithPathParam(@PathParam('name') name: string, @QueryParam('id') id: string, @BodyParam body: any) {
  console.info(`TestController - post with name: ${name}, body: ${JSON.stringify(body)}`);
  return 'ok';
}

@POST里的参数是可选的,空的话会用这个controller的物理路径做为路由地址。

:name是路径里的变量,比如 /user/brook, :name就是brook,可以在方法的参数里用@PathParam得到

@QueryParam可以得到url?后的参数

@BodyParam可以得到Post上来的body

是不是有点WebApi的意思了。

现在具体看看是怎么实现的

实现过程其实很简单,从上面的目标入手,首先得到controllers的物理路径,然后还要得到被装饰器装饰的方法以及它的参数。
装饰器的目的在于要得到是Get还是Post等,还有就是指定的Path,最后就是把node request里的数据赋值给方法的参数。

核心代码:

得到物理路径

initRouterForControllers() {
  //找出指定目录下的所有继承自BaseController的.js文件
  let files = FileUtil.getFiles(this.controllerFolder);

  files.forEach(file => {
    let exportClass = require(file).default;

    if(this.isAvalidController(exportClass)){
      this.setRouterForClass(exportClass, file);
    }
  });
}

从物理路径转成路由

private buildControllerRouter(file: string){

  let relativeFile = Path.relative(Path.join(FileUtil.getApiDir(), this.controllerFolder), file);
  let controllerPath = '/' + relativeFile.replace(/\\/g, '/').replace('.js','').toLowerCase();

  if(controllerPath.endsWith('controller'))
    controllerPath = controllerPath.substring(0, controllerPath.length - 10);

  return controllerPath;
}

装饰器的实现

装饰器需要引入reflect-metadata库

先看看方法的装饰器,@GET,@POST之类的,实现方法是给装饰的方法加一个属性RouterRouter是个Symbol,确保唯一。 然后分析装饰的功能存到这个属性中,比如MethodPath等。

export function GET(path?: string) {
  return (target: BaseController, name: string) => setMethodDecorator(target, name, 'GET', path);
} 

function setMethodDecorator(target: BaseController, name: string, method: string, path?: string){
  target[Router] = target[Router] || {};
  target[Router][name] = target[Router][name] || {};
  target[Router][name].method = method;
  target[Router][name].path = path;
}

另外还有参数装饰器,用来给参数赋上request里的值,如body,param等。

export function BodyParam(target: BaseController, name: string, index: number) {
  setParamDecorator(target, name, index, { name: "", type: ParamType.Body });
}

function setParamDecorator(target: BaseController, name: string, index: number, value: {name: string, type: ParamType}) {
  let paramTypes = Reflect.getMetadata("design:paramtypes", target, name);
  target[Router] = target[Router] || {};
  target[Router][name] = target[Router][name] || {};
  target[Router][name].params = target[Router][name].params || [];
  target[Router][name].params[index] = { type: paramTypes[index], name: value.name, paramType: value.type };
}

这样装饰的数据就存到对象的Router属性上,后面构建路由时就可以用了。

绑定路由到Koa-router

上面从物理路径得到了路由,但是是以装饰里的参数路径优先,所以先看看刚在存在原型里的Router属性里有没有Path,有的话就用这个作为路由,没有Path就用物理路由。

private setRouterForClass(exportClass: any, file: string) { 

  let controllerRouterPath = this.buildControllerRouter(file);
  let controller = new exportClass();

  for(let funcName in exportClass.prototype[Router]){
    let method = exportClass.prototype[Router][funcName].method.toLowerCase();
    let path = exportClass.prototype[Router][funcName].path;

    this.setRouterForFunction(method, controller, funcName, path ? `/${this.urlPrefix}${path}` : `/${this.urlPrefix}${controllerRouterPath}/${funcName}`);
  }
}

给controller里的方法参数赋上值并绑定路由到KoaRouter

private setRouterForFunction(method: string, controller: any, funcName: string, routerPath: string){
  this.koaRouter[method](routerPath, async (ctx, next) => { await this.execApi(ctx, next, controller, funcName) });
}

private async execApi(ctx: Koa.Context, next: Function, controller: any, funcName: string) : Promise<void> { //这里就是执行controller的api方法了
  try
  {
    ctx.body = await controller[funcName](...this.buildFuncParams(ctx, controller, controller[funcName]));
  }
  catch(err)
  {
    console.error(err);
    next(); 
  }
}

private buildFuncParams(ctx: any, controller: any, func: Function) { //把参数具体的值收集起来
  let paramsInfo = controller[Router][func.name].params;
  let params = [];
  if(paramsInfo)
  {
    for(let i = 0; i < paramsInfo.length; i++) {
      if(paramsInfo[i]){
        params.push(paramsInfo[i].type(this.getParam(ctx, paramsInfo[i].paramType, paramsInfo[i].name)));
      } else {
        params.push(ctx);
      }
    }
  }
  return params;
}

private getParam(ctx: any, paramType: ParamType, name: string){ // 从ctx里把需要的参数拿出来
  switch(paramType){
    case ParamType.Query:
      return ctx.query[name];
    case ParamType.Path:
      return ctx.params[name];
    case ParamType.Body:
      return ctx.request.body;
    default:
      console.error('does not support this param type');
  }
}

这样就完成了简单版的类似WebApi的路由.

源码下载webapi-router_jb51.rar

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# NodeJS仿WebApi  # nodejs  # webapi  # node.js中使用q.js实现api的promise化  # 实现一个完整的Node.js RESTful API的示例  # 详解本地Node.js服务器作为api服务器的解决办法  # node.js实现微信JS-API封装接口的示例代码  # 浅析Node.js 中 Stream API 的使用  # Node.js原生api搭建web服务器的方法步骤  # 30分钟用Node.js构建一个API服务器的步骤详解  # 零基础之Node.js搭建API服务器的详解  # NodeJS基础API搭建服务器详细过程记录  # 深入分析node.js的异步API和其局限性  # 浅析Node在构建超媒体API中的作用  # Node.js API详解之 tty功能与用法实例分析  # 就用  # 可以得到  # 绑定  # 是个  # 我在  # 目录下  # 在这  # 是怎么  # 很容易  # 微软  # 可以通过  # 用了  # 这不是  # 写了  # 非常好  # 可选  # 用过  # 时就  # 要得  # 拿出来 


相关文章: 建站主机类型有哪些?如何正确选型  5种Android数据存储方式汇总  建站之星Pro快速搭建教程:模板选择与功能配置指南  如何在阿里云通过域名搭建网站?  C#如何序列化对象为XML XmlSerializer用法  制作网站建设的公司有哪些,网站建设比较好的公司都有哪些?  大连 网站制作,大连天途有线官网?  行程制作网站有哪些,第三方机票电子行程单怎么开?  C++时间戳转换成日期时间的步骤和示例代码  威客平台建站流程解析:高效搭建教程与设计优化方案  文字头像制作网站推荐软件,醒图能自动配文字吗?  c# F# 的 MailboxProcessor 和 C# 的 Actor 模型  定制建站价位费用解析与套餐推荐全攻略  c# 服务器GC和工作站GC的区别和设置  家庭建站与云服务器建站,如何选择更优?  南平网站制作公司,2025年南平市事业单位报名时间?  建站之星安装模板失败:服务器环境不兼容?  北京网站制作的公司有哪些,北京白云观官方网站?  如何破解联通资金短缺导致的基站建设难题?  制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?  如何在阿里云部署织梦网站?  建站之星备案流程有哪些注意事项?  如何制作一个表白网站视频,关于勇敢表白的小标题?  如何在宝塔面板中修改默认建站目录?  深圳网站制作的公司有哪些,dido官方网站?  网页设计网站制作软件,microsoft office哪个可以创建网页?  临沂网站制作公司有哪些,临沂第四中学官网?  阿里云网站制作公司,阿里云快速搭建网站好用吗?  ,怎么用自己头像做动态表情包?  制作旅游网站html,怎样注册旅游网站?  免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?  怀化网站制作公司,怀化新生儿上户网上办理流程?  建站之星代理如何优化在线客服效率?  制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?  建站之星五站合一营销型网站搭建攻略,流量入口全覆盖优化指南  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法  制作网页的网站有哪些,电脑上怎么做网页?  ,制作一个手机app网站要多少钱?  ,sp开头的版面叫什么?  如何通过FTP服务器快速搭建网站?  宁波免费建站如何选择可靠模板与平台?  建站之星体验版:智能建站系统+响应式设计,多端适配快速建站  云南网站制作公司有哪些,云南最好的招聘网站是哪个?  商务网站制作工程师,从哪几个方面把握电子商务网站主页和页面的特色设计?  PHP正则匹配日期和时间(时间戳转换)的实例代码  C++如何将C风格字符串(char*)转换为std::string?(代码示例)  网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?  浅析上传头像示例及其注意事项  如何零成本快速生成个人自助网站? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。