koa-grace:一个基于koa的node多应用MVC框架

news/2024/9/17 17:37:52

春节期间没回家留在北京写了一个基于koa的node MVC框架:koa-grace ,大家有兴趣可以star & fork下,谢谢支持啦!!

img

项目地址:

https://github.com/xiongwilee/koa-grace

详细文档:

1. 简介

koa-grace是基于koa 1.x的Nodejs多站点MVC框架。

1.1 特征

为什么koa-grace是新一代Nodejs MVC框架:

  • 一个node服务,多个站点应用;

  • yield 异步语法支持,忘掉回调噩梦;

  • 继承koa中间件,扩展性更强;

  • 支持路径即路由,更优雅的路由方式;

  • RESTful数据代理支持,前后端完全解耦;
    ……

1.2 目录结构

├── app                        // 站点总目录
│   ├── blog                     // 站点:blog目录
│   │   ├── controller                 // 站点:blog的路由(控制器)目录
│   │   ├── model                     // 站点:blog的模型目录,包括公共控制器、mongo等
│   │   ├── static                     // 站点:blog的静态文件目录
│   │   └── views                     // 站点:blog的html模板目录
│   ├── reactjs-boilerplate     // 站点:reactjs-boilerplate目录
│   └── shop                     // 站点:shop目录
├── bin                        // server启动器目录
│   ├── koa-grace                 // TODO:命令行工具
│   └── server.js                  // server启动器
├── config                     // 配置文件目录
│   └── main.js                    // 主配置文件
├── package.json
└── src                        // 核心文件└── app.js                    // 主文件

2. 快速开始

在开始使用koa-grace之前请确保您已经安装并运行了下面的工具:

  • Nodejs (v4+)

  • MongoDB (DEMO 演示使用,正式环境可以配置不用)

2.1 安装

$ git clone https://github.com/xiongwilee/koa-grace.git 
$ cd koa-grace && npm install

2.2 配置

在koa-grace目录下,打开配置文件:

$ vi config/main.js

修改配置项:config.mongo.blog 为您的本地mongoDB的路径:

// mongo configuration
mongo: {'blog': 'mongodb://localhost:27017/blog'
}

**更多关于配置项的文档可以看下面的使用文档。

2.3 运行

在koa-grace目录下执行,可能需要root权限:

$ npm run dev

然后访问 http://127.0.0.1:3000 ,就可以看到koa-grace其中的一个案例站点了!

您也可以访问koa-grace的一个线上应用: http://mlsfe.biz 。

3. 详细使用文档

虽说koa-grace是一个完整的MVC框架 , 但其本质是基于一种多站点解决方案的koa中间件的集合。其核心中间件包括但不仅限于: koa-router , koa-views , koa-mount , koa-static , koa-grace-vhost , koa-grace-router , koa-grace-proxy , koa-grace-model , koa-grace-mongo , ...

3.1 多站点配置 - Vhost

koa-grace是基于 koa-grace-vhost 进行vhost管理,基本原理是:

一个域名对应一个应用,一个应用对应一个目录

如此一来,配置多站点就很简单了,在配置文件config/main.js中:

// vhost配置
vhost: {'test.mlsfe.biz':'blog','127.0.0.1':'blog','localhost':'shop','0.0.0.0':'reactjs-boilerplate'
}

其中,vhost配置中127.0.0.1是URI的hostname, blog是站点总目录app下的一个目录app/blog。如果当前请求URI为:http://127.0.0.1/home 则koa-grace会自动访问app/blog目录里的路由、模型、静态文件。

需要说明的是,多站点配置仅以URI的hostname为主键 ;也就是说,访问带端口号的http://127.0.0.1:3000/home 也会定位到app/blog目录。

3.2 路由及控制器 - Router&Controller

很好,如果你配置好了一个vhost为'127.0.0.1':'blog' , koa-grace就会自动生成一个vhost到app/blog目录了!接下来,进入app/blog/controller目录进行路由配置。

koa-grace基于 koa-grace-router 进行路由管理的,koa-grace-router又是依赖于:koa-router 。

3.2.1 文件路径即路由

blog站点为例,koa-grace-router会找到 app/blog/* 目录下的所有.js后缀的文件,并以文件路径生成路由。我们再看一下案例中blog的路由文件:

├── api
│   └── post.js
├── dashboard
│   ├── post.js
│   ├── site.js
│   ├── user.js
│   └── userAuthor.js
├── error.js
├── home.js
├── post.js
└── user.js

如果当前请求URI为:http://127.0.0.1/dashboard/post/* 则路由将自动落在dashboard/post.js文件中。

那么,如果请求路径如果是http://127.0.0.1/dashboard/post/list ,这个dashboard/post.js文件是如何控制的呢?

3.2.2 路由文件详细说明

打开app/blog/controller/dashboard/post.js文件:

/*...*/
exports.list = function* () {// 绑定默认控制器yield this.bindDefault();// 独立权限控制if (!userAuthor.checkAuth(this, this.userInfo)) {return};// 获取请求query参数let pageNum = this.query.page;// 获取数据let PostModel = this.mongo('Post');let posts = yield PostModel.page(pageNum,20);let page = yield PostModel.count(pageNum,20);// 渲染模板yield this.render('dashboard/post_list',{breads : ['文章管理','文章列表'],posts:posts,page:page,userInfo: this.userInfo,siteInfo: this.siteInfo})
}
exports.list.__method__ = 'get';
exports.list.__regular__ = null;
/*...*/

对,就是你猜的那样:koa-grace-router是通过post.js的module.exports进行下一步的路由控制。

另外,需要说明以下几点:

  • 如果需要配置dashboard/post/list请求为POST方法,则post.js中声明 exports.list.__method__ = 'post'即可(不声明默认为get请求),更多方法类型请参看:koa-router#routergetputpostpatchdelete--router;

  • 如果要进一步配置dashboard/post/list/id路由,则在post.js中声明exports.list.__regular__ = '/:id';即可,更多相关配置请参看:koa-router#named-routes

  • 如果当前文件路由就是一个独立的控制器,则module.exports返回一个yield方法即可,可以参考案例blog中的controll/home.js

  • 如果当前文件仅仅是一个依赖,仅仅被其他文件引用;则在文件中配置exports.__controller__ = false,该文件就不会生成路由了

当然,如果一个路由文件中的控制器方法都是post方法,您可以在控制器文件最底部加入:module.exports.__method__ = 'post'即可。__regular__的配置同理。

3.2.3 控制器

刚刚我们看到了post.js中的exports.list方法,事实上它就是一个控制器(controller)了。

您可以新建一个app/blog/controller/hello.js文件

exports.koagrace = funtion* (){this.body = 'hello koa-grace!';
}

访问 http://127.0.0.1/hello/koagrace ,就可以看到“hello koa-grace!”输出。它是典型的一个基于上下文(context)的yield方法。几个关键方法/参数使用如下:

context属性类型说明
this.queryobjectget参数
this.request.bodyobjectpost参数,由于koa-grace默认引入了koa-bodypaser,您可以直接在this.request.body中获取到post参数
this.bindDefaultfunction公共控制器,相当于require('app/blog/model/defaultCtrl.js')
this.renderfunction模板引擎渲染方法,请参看:3.5 模板引擎- Template engine
this.mongofunction数据库操作方法,请参看:3.3 数据库 - Database
this.mongoMapfunction并行数据库多操作方法,请参看:3.3 数据库 - Database
this.proxyfunctionRESTful数据请求方法,请参看:3.4.1 数据代理
this.downloadfunction文件请求代理方法,请参看:3.4.2 请求代理
this.uploadfunction文件上传方法,请参看: 3.4.3 文件上传

更多context文档请参看koa官网,或http://koajs.in/doc/

3.3 数据库 - Database

koa-grace引入基于mongoose的koa-grace-mongo ,可以非常便捷地使用mongoDB。

3.3.1 连接数据库

在配置文件config/main.js中进行配置:

  // mongo配置mongo: {options:{// mongoose 配置},api:{'blog': 'mongodb://localhost:27017/blog'}},

其中,mongo.options配置mongo连接池等信息,mongo.api配置站点对应的数据库连接路径。

值得注意的是,配置好数据库之后,一旦koa-grace server启动mongoose就启动连接,直到koa-grace server关闭

3.3.2 mongoose的schema配置

依旧以案例blog为例,参看app/blog/model/mongo目录:

└── mongo├── Category.js├── Link.js├── Post.js└── User.js

一个js文件即一个数据库表即相关配置,以app/blog/model/mongo/Category.js

'use strict';// model名称,即表名
let model = 'Category';// 表结构
let schema = [{id: {type: String,unique: true,required: true},name: {type: String,required: true},numb: {type: Number,'default':0}
}, {autoIndex: true,versionKey: false
}];// 静态方法:http://mongoosejs.com/docs/guide.html#statics
let statics = {}// 方法扩展 http://mongoosejs.com/docs/guide.html#methods
let methods = {/*** 获取博客分类列表*/list: function* () {return this.model('Category').find();}
}module.exports.model = model;
module.exports.schema = schema;
module.exports.statics = statics;
module.exports.methods = methods;

主要有四个参数:

  • model , 即表名,最好与当前文件同名

  • schema , 即mongoose schema

  • methods , 即schema扩展方法,推荐把数据库元操作都定义在这个对象中

  • statics , 即静态操作方法

3.3.3 在控制器中调用数据库

在控制器中使用非常简单,主要通过this.mongo,this.mongoMap两个方法。

1) this.mongo(name)

调用mongoose Entity对象进行数据库CURD操作

参数说明:

@param [string] name : 在app/blog/model/mongo中配置Schema名,

返回:

@return [object] 一个实例化Schema之后的Mongoose Entity对象,可以通过调用该对象的methods进行数据库操作

案例

参考上文中的Category.js的配置,以app/blog/controller/dashboard/post.js为例,如果要在博客列表页中获取博客分类数据:

// http://127.0.0.1/dashboard/post/list
exports.list = function* (){let cates = yield this.mongo('Category').list();this.body = cates;
}

2)this.mongoMap(option)

并行多个数据库操作

参数说明

@param [array] option
@param [Object] option[].model mongoose Entity对象,通过this.mongo(model)获取
@param [function] option[].fun mongoose Entity对象方法
@param [array] option[].arg mongoose Entity对象方法参数

返回

@return [array] 数据库操作结果,以对应数组的形式返回

案例

  let PostModel = this.mongo('Post');let mongoResult = yield this.mongoMap([{model: PostModel,fun: PostModel.page,arg: [pageNum]},{model: PostModel,fun:PostModel.count,arg: [pageNum]}]);let posts = mongoResult[0];// 获取第一个查询PostModel.page的结果let page = mongoResult[1]; // 获取第二个查询PostModel.count的结果,两者并发执行

3.4 代理 - Proxy

除了在控制器中直接进行数据库操作,Web应用还有可能由其他服务进行后端部署。针对这种场景,koa-grace引入了基于 Request 的 koa-grace-proxy。

3.4.1 数据代理

在koa-grace的控制器中使用this.proxy方法进行数据代理非常简单:

exports.list = function* (){yield this.proxy({userInfo:'github:post:user/login/oauth/access_token?client_id=****',otherInfo:'github:other/info?test=test',});console.log(this.backData);/***    {*        userInfo : {...},*        otherInfo : {...}*    }*/
}

你也可以不传this.backData参数,默认注入到上下文的this.backData对象中:

exports.list = function* (){yield this.proxy({userInfo:'github:post:user/login/oauth/access_token?client_id=****',otherInfo:'github:other/info?test=test',});console.log(this.backData);/***    {*        userInfo : {...},*        otherInfo : {...}*    }*/
}

另外,github:post:user/login/oauth/access_token?client_id=****说明如下:

  • github: 为在config.main.jsapi 对象中进行配置;

  • post : 为数据代理请求的请求方法,该参数可以不传,默认为get

  • path: 后面请求路径中的query参数会覆盖当前页面的请求参数(this.query),将query一同传到请求的接口

  • 你也可以写完整的路径:{userInfo:'https://api.github.com/user/login?test=test'}

3.4.2 文件代理

文件请求代理也很简单,比如如果需要从github代理一个图片请求返回到浏览器中,参考:http://mlsfe.biz/user/avatar?img=https://avatars.githubusercontent.com/u/1962352?v=3

exports.avatar = function* (){let imgUrl = query.img;yield this.download(imgUrl);
}

3.4.3 文件上传

TODO: 文件上传并代理到其他服务或者存储到本地

3.5 模板引擎- Template engine

koa-grace引入koa-views , 进行模板引擎管理。默认的模板引擎为swig, 您可以在config/main.js中配置template属性您想要模板引擎:

  // 模板引擎配置template: 'swig'

目前支持的模板引擎列表在这里:consolidate.js#supported-template-engines

在控制器中调用this.render方法渲染模板引擎:

exports.home = function* () {yield this.bindDefault();yield this.render('dashboard/site_home',{breads : ['站点管理','通用'],userInfo: this.userInfo,siteInfo: this.siteInfo})
}

模板文件在app/blog/views目录中。

3.6 静态文件服务 - Static server

koa-grace引入koa-mount 及 koa-static,将静态文件代理到/static

  // 配置静态文件路由vapp.use(mount('/static',koastatic(appPath + '/static')));

以案例中blog的静态文件为例,静态文件在blog项目下的路径为:app/blog/static/image/bg.jpg,则访问路径为http://127.0.0.1/static/image/bg.jpg。

3.7 启动服务及调试 - Process & DEBUG

3.7.1 开发环境

在开发环境可以使用npm命令完成。

1) 普通启动:

$ npm run start

2) watch启动:

$ npm run dev
# 在80端口启动
$ npm PORT=80 run dev
# DEBUG模式启动,默认为DEBUG=koa-grace*
$ npm DEBUG=* run dev 

3.7.2 生产环境

在生产环境推荐使用 pm2 进行进程管理:

$ npm install pm2 -g
$ pm2 start node ./bin/server.js

更多使用方法,请参看 pm2。

5. 贡献

  • 提ISSUE: https://github.com/xiongwilee/koa-grace/issues

  • 欢迎提PR

  • 给作者提问&提建议:xiongwilee@foxmail.com


http://lihuaxi.xjx100.cn/news/285931.html

相关文章

07-图6 旅游规划 (25分)(以此感谢zyx佬)

这个题的话算是模板题改编了一点吧,不过个人感觉这个改编很有助于你理解迪杰斯特拉这个算法的真谛。 题解:新开一个cost数组来记录花费,仍然是用了优先队列优化的一个思想,与模板题不同的是只需要加一句话(感谢zyx佬发…

百度Apollo 3.5是如何设计Cyber RT计算框架的?

自百度Apollo自动驾驶平台开源以来,已快速迭代至 3.5 版本,代码行数超过 39 万行,合作伙伴超过 130 家,吸引了来自 97 个国家的超 15000 名开发者。无疑,Apollo 是目前世界范围内最活跃的自动驾驶开放平台之一。最新发…

有没有必要把机器学习算法自己实现一遍?

编辑:机器学习算法与自然语言处理-忆臻,Charlotte数据挖掘-小杜https://www.zhihu.com/question/36768514作者:微调https://www.zhihu.com/question/36768514/answer/376510114不少自学的朋友很容易陷入到焦虑当中,尤其是在学习理…

Chapter 3、Java语法基础(二)----Java基本数据类型、变量与常量 (20th,Feb)

基本数据类型 1、整数类型 整数类型用来储存整数数值,即没有小数部分的数值,可以使正数、负数,也可以是零。根据所占内存的大小不同,分为byte、short、int、long 4种类型。 Byte型 整型中所分配内存空间最少的,只分配1…

后缀数组 + Hash + 二分 or Hash + 二分 + 双指针 求 LCP ---- 2017icpc 青岛 J Suffix (假题!!)

题目链接 题目大意: 就是给你n个串每个串取一个后缀,要求把串拼起来要求字典序最小!! sum_length_of_n≤5e5sum\_length\_of\_n\leq 5e5sum_length_of_n≤5e5 MY Slove : 首先我们知道对于最后一个串肯定是取最小后缀的 那么我们可以把最后…

第2关:实现一个链接存储的栈

#if !defined(LINKED_STACK_H_985552) #define LINKED_STACK_H_985552 typedef int T; //数据元素类型 struct LNode {T data;LNode* next; };struct LinkStack {LNode* top; // 栈顶指针int len; // 栈的长度 };LinkStack* LS_Create(); void LS_Free(LinkStack* ls); void LS…

最常用 150 个Linux命令汇总(建议收藏)

点击上方[视学算法]→右上角[...]→[设为星标⭐]来源:banana 童www.cnblogs.com/bananaaa/p/7774467.htmllinux 命令是对 Linux 系统进行管理的命令。对于 Linux 系统来说,无论是中央处理器、内存、磁盘驱动器、键盘、鼠标,还是用户等都是文件…

牛客小白月赛25 补题+题解[A-J]

加油加油加油! 文章目录A.AOE还是单体?B.k-size字符串C.白魔法师D.抽卡E.点击消除F.疯狂的自我检索者G.解方程H.神奇的字母(二)I.十字爆破J.异或和之和A.AOE还是单体? 思路:这题数据范围2e5,如…