亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

NodeJS仿WebApi路由示例

2019-11-19 17:23:12
字體:
來源:轉載
供稿:網友

用過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/name,api/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之類的,實現方法是給裝飾的方法加一個屬性Router,Router是個Symbol,確保唯一。 然后分析裝飾的功能存到這個屬性中,比如Method,Path等。

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

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
夜夜嗨av一区二区三区四区| 亚洲美女免费精品视频在线观看| 成人久久久久爱| 亚洲剧情一区二区| 亚洲成色777777在线观看影院| 在线视频一区二区| 欧美性xxxx极品hd欧美风情| 国产精品一区二区av影院萌芽| 97香蕉久久超级碰碰高清版| 亚洲最新视频在线| 久久久国产精品免费| 成人激情视频网| 国产精品久久久久久中文字| 国产视频亚洲精品| 国产精品国产三级国产专播精品人| 亚洲国产精品va在线| 久久久久久久97| 中文字幕日韩av综合精品| 久久精品国产欧美激情| 日韩精品一二三四区| 伊人成人开心激情综合网| 亚洲最大中文字幕| 久久综合久久88| 亚洲va久久久噜噜噜久久天堂| 欧美电影院免费观看| 欧美成人亚洲成人| 91香蕉亚洲精品| 久久精品国产69国产精品亚洲| 国产精品入口尤物| 亚洲久久久久久久久久| www日韩中文字幕在线看| 欧美性极品少妇精品网站| 欧美成人激情视频| 色悠久久久久综合先锋影音下载| 久久婷婷国产麻豆91天堂| 国产精品美女在线观看| 亚洲免费伊人电影在线观看av| 高清欧美电影在线| 国产欧美日韩精品专区| 自拍偷拍亚洲在线| 亚洲美女免费精品视频在线观看| 亚洲欧美中文日韩在线| 国产精品嫩草影院一区二区| 在线亚洲男人天堂| 国产精品久久99久久| 国产乱肥老妇国产一区二| 日韩精品中文字幕在线| 插插插亚洲综合网| 欧美激情综合色综合啪啪五月| 日韩中文在线中文网三级| 日韩视频亚洲视频| 日韩性生活视频| 亚洲精品wwww| 亚洲黄色www网站| 亚洲国产日韩欧美在线图片| 久久久国产精品免费| 欧美一区二区三区四区在线| 97视频免费看| 亚洲色图35p| 日韩成人在线视频| 欧美性猛交xxxx免费看漫画| 国产精品啪视频| 亚洲女人天堂成人av在线| 色青青草原桃花久久综合| 亚洲free嫩bbb| 亚洲a在线观看| 亚洲免费电影一区| 欧美成人激情视频| 亚洲欧美三级在线| 午夜精品久久17c| 欧美电影免费观看电视剧大全| 中文字幕av一区二区三区谷原希美| 2019av中文字幕| 亚洲一区二区久久| 亚洲精品日韩激情在线电影| 色狠狠久久aa北条麻妃| 欧美日韩亚洲一区二| www.99久久热国产日韩欧美.com| 国产精品最新在线观看| 日韩成人激情视频| 揄拍成人国产精品视频| 欧美在线观看网址综合| 少妇高潮久久久久久潘金莲| 欧美午夜性色大片在线观看| 欧美大片在线影院| 亚洲九九九在线观看| 亚洲欧美日韩中文在线制服| 精品成人在线视频| 91久久精品日日躁夜夜躁国产| 97视频在线观看视频免费视频| 欧美成年人在线观看| 精品国产1区2区| 国产精品免费福利| 亚洲国产高清高潮精品美女| 国产成人精品午夜| 北条麻妃在线一区二区| 欧美激情手机在线视频| 正在播放欧美视频| 久久噜噜噜精品国产亚洲综合| 精品久久久久久国产| 日韩电视剧在线观看免费网站| 欧美性jizz18性欧美| 亚洲а∨天堂久久精品喷水| 成人久久久久久久| 在线观看精品国产视频| 久久偷看各类女兵18女厕嘘嘘| 少妇精69xxtheporn| 九九热这里只有精品6| 成人黄在线观看| 日韩av影视在线| 欧美精品精品精品精品免费| 欧美激情精品久久久久| 伊人久久久久久久久久久久久| 欧美性生交大片免网| 久久久av亚洲男天堂| 成人福利网站在线观看11| 国产精品色视频| 青青草原一区二区| 国产精品18久久久久久麻辣| 精品调教chinesegay| 亚洲精品久久久久久下一站| 欧美日韩第一视频| 亚洲欧美三级在线| 亚洲欧洲国产伦综合| 精品伊人久久97| 久久久国产精品亚洲一区| 精品亚洲男同gayvideo网站| 在线视频欧美日韩精品| 欧美激情在线一区| 亚洲精品动漫100p| 亚洲国产欧美一区二区三区久久| 色哟哟入口国产精品| 久久久久久亚洲精品不卡| 色多多国产成人永久免费网站| 国产精品一久久香蕉国产线看观看| 亚洲va久久久噜噜噜| 亚洲欧美激情一区| 日韩在线视频线视频免费网站| 日韩中文字幕在线免费观看| 欧美性猛交xxxx| 欧美极度另类性三渗透| 亚洲欧洲一区二区三区在线观看| 中文字幕亚洲天堂| 久久免费观看视频| 久久精品亚洲94久久精品| 高潮白浆女日韩av免费看| 亚洲综合大片69999| 亚洲精品国产精品国自产观看浪潮| 欧美午夜激情视频| 亚洲美女av黄| 国产啪精品视频网站| 欧美激情在线视频二区| 精品国产一区二区三区久久狼5月| 国模吧一区二区| 日韩视频免费观看| 九九九热精品免费视频观看网站| 久久久免费在线观看| 国产精品高潮呻吟久久av无限| 亚洲iv一区二区三区| 欧美老女人bb| 国产精品日本精品| 91精品视频观看| 亚洲免费中文字幕| 欧美日韩在线视频一区|