koa2核心功能实现
前言
koa和express都是nodejs的web框架,但它们的设定不同。express自身集成很多东西,比较重,适合企业级的应用开发。koa功能纯粹,扩展功能需高度依赖生态。这种可插拔的形式使得其更为灵活,轻量。从npm周下载量来看,express千万级,koa十万级,差距还是很明显。
本篇为源码系列核心实现第一篇,对应下图koa2部分。
src
核心功能概览
koa2即使是全部代码,也没多少东西。从整体上看,主要分为五大方向。
namedescapplication创建上下文,合并中间件,启动服务request对原生http模块req的扩展response对原生http模块res的扩展context对request和response的合并与代理compose合并中间件(洋葱模型的体现)
koa是如何启动一个服务的?
基本使用
在详解上述功能模块前,先来看看koa最基础的使用。
const Koa=require('koa')const app=new Koa()app.listen(3000,()=>{ console.log('run server__')})核心实现
简单的几行代码就实现了koa最核心的功能——提供基础的HTTP应用服务。它是如何实现的呢?其实也没什么玄机,直接用了内置的http模块。注意一个小细节:Application是一个类,这也就解释了在使用koa时为什么要new一下。
const http=require('http')class Application{ listen(...args){ const server= http.createServer() server.listen(...args) }}
koa是如何处理一个请求的?
基础使用
既然应用启动是直接使用了http模块,那对于请求处理是不是也和http模块的处理相似呢?是的,确实如此,但koa做的更多。先来看一个例子, 下面代码背后包含了koa对于一个请求到响应的完整处理过程。
app.use(async ctx=>{ ctx.body='hello world'})一个请求--响应带来的思考
就这?一个use,一个ctx,一个body,啥也没看出来啊?还有,为啥要加个async ,去掉不行?
从视觉入手,这几个最直观的点也最好解释。带着问题去思考问题,是我最喜欢的学习方式。ok,接下来,我们就看一下koa内部是如何处理请求的。
class Application{ use(fn){ this.fn=fn } handleRequest(req,res){ const ctx=this.createContext(req,res) this.fn(ctx) } listen(...args){ const server= http.createServer(this.handleRequest.bind(this)) server.listen(...args) }} 这几行代码没什么玄机,所谓的use方法,不过是将接收的函数暂存,后续在handleRequest中执行。
handleRequest函数见名知意,用于处理请求。其实,也就是把原生http模块 createServer的事件处理函数做了一个提取,并将上下文this指向我们自己 写的Application类。其中,this.fn(ctx)这行代码,解释了为什么每一个请求处理函数都会接收ctx参数。
createContext是什么?
看名字是要创建上下文。是的,这就是koa独特的地方。
koa将原生http模块事件处理函数的req和res参数合二为一后又做了一层增强。最终的结果就是:req,res有的,ctx皆有。req,res没有的,ctx还有。对开发者而言,合二为一后完全无需关心某个方法是req的还是res的,直接一个ctx完事。
说了这么多,我们就来看一下createContext的庐山真面目。
class Application{ constructor(){ //这三个是外部引入的 this.context=context; this.request=request; this.response=response; } createContext(req,res){ //使用Object.create是为了在不对原模块进行干扰的情况下进行扩展,也是一层继承 const context=Object.create(this.context) const request=Object.create(this.request) const response=Object.create(this.response) //上下文关联与合并 context.req=context.request.req=req context.res=context.response.res=res //返回一个合并后的context; return context }} 为了便于理解,我在必要部分加了一些注释。暂且不管构造器里的context,request,response是什么,先只看createContext函数做了什么。里边有两行看起来很绕的连续赋值代码,其实就是往ctx上挂载东西。
比如你用原生http模块结束响应是res.end(),那现在可以用ctx.res.end(),也可以用ctx.response.res.end()。req和res是一样的道理,这里就不再赘述了。
request和response
到这里,我们之前疑惑的use,ctx已经解释完。在进一步解释async和body前,我们暂且引出一个新的问题。request和response与http模块事件处理函数的req,res的关系是什么?
首先可以肯定的是,绝不是同一个东西。但是,request和response是对req和res的一个增强。这两个文件,是koa单独搞出来的,其内部使用了getter,setter。
先来看一个request的简单示例:
const request={ get url() { return this.req.url; }}module.exports=request 这里的this指向request对象,但仅这样还看不出玄机,别忘了在createContext方法中的request身上恰好挂了一个req。这意味着什么?意味着访问ctx.request.req 就是访问原生req。
举个例子:上边request中的url访问方式看似是ctx.request.url,实际上是ctx.request.req.url。
app.use(ctx => { ctx.body=ctx.request.req.url===ctx.request.url //true}) 也许你会觉得,这个例子看起来好像是代理啊,如何起到增强作用呢?
const request = { get headers() { return this.req.headers; }, get header() { return this.req.headers; },} 个人习惯问题,在写代码时会纠结是headers还是header。koa考虑的很人性化,不管你用哪种,都对,最终都是访问的headers。从这点考虑,岂不就是容错性的增强?当然,实际的增强并不仅限于此,甚至可以自定义你想要的业务逻辑。
response和request同理,这里不再赘述。只提最关键的一点,body。
const response = { _body: '', get body() { return this._body }, set body(newBody) { this._body = newBody }}module.exports = response 这样一看是不是body也没那么神奇了呢?不过就是一个变量而已。若是访问,直接返回变量_body;若是设置,接收新值完成更新。
也许你会好奇,ctx.body和这个body是一个东西吗?是的,当然是。那它们是如何关联上的?看起来是代理?是的,就是代理,通过context。
神奇的context代理
先来看看context内部实现吧。
const context = {}function delegateGet(prop, key) { //__defineGetter__这个方法是当访问对象的某个key时,执行回调 context.__defineGetter__(key, function () { return this })}function delegateSet(prop, key) { context.__defineSetter__(key, function (newValue) { this = newValue })}delegateGet('response', 'body')//访问ctx.bodydelegateSet('response', 'body')//设置ctx.body='xxx'delegateGet('request', 'url')//ctx.url<=>ctx.request.urlmodule.exports = context 看完上述代码来个小总结吧。ctx本质是代理,并非增强; ctx做的响应相关的,一定是交给response; ctx做的请求相关的,一定是交给request。
关于__defineGetter__,可参考文末MDN相关链接。如果你去看koa源码,你会发现它使用了一个第三方包:delegates,其实这东西实现也是用的__defineGetter__
神奇的组合能力:compose
到这里我们之前提到的疑惑只剩下async尚未解决,接下来就深入展开一波。在探究compose函数实现前,先来想一下为什么需要组合?解决了什么问题?
app.use(bodyParser())app.use(koaStatic())... koa中,use函数可以多次调用,但是默认情况下只会执行第一个。后边的如果想执行,需要上游调用next函数,也就是use函数的第二个参数。那如何涉及异步怎么搞?这就是async的意义所在。多个中间件如何执行呢?这就是compose函数的意义。(洋葱模型)
依次输出123456
app.use((ctx, next) => { console.log(1) next() console.log(6)})app.use((ctx, next) => { console.log(2) next() console.log(5)})app.use((ctx, next) => { console.log(3) next() console.log(4)}) 接下来我们来研究一波它的实现。核心三要素,按存储顺序依次执行,支持异步,洋葱模型。
//做一个小小的改造,支持多个use调用 use(fn){ this.middlewares.push(fn) } compose(ctx) { //这里dispatch(也就是next)使用箭头函数 //内部的this就指向了自定义的Application const dispatch = (index) => { //越界处理 handleRequest还有then 不能直接return ,要返回promise if (index === this.middlewares.length) return Promise.resolve() //获取当前的中间件 最开始是第一个 const middleware = this.middlewares // 中间件执行需要两个参数 const exec = middleware(ctx, () => dispatch(++index)) //有可能这个方法没有加async,包装一层 //保证返回的是一个promise //这样handleRequest的then函数就不会报错(下文解释) return Promise.resolve(exec) } return dispatch(0)} 到这里,compose实现基本就完活了,边界细节可以去看koa-compose。
上边有提到最后要返回promise,看起来有些突兀。莫慌,补上最后一波代码就可以理解了。
handleRequest
handleRequest(req, res) { const ctx = this.createContext(req, res) //组合中间件 并执行返回后的promise,获取到_body 响应出去 //注意看这里,是有一个then的 //这就意味着最后不管你写的函数加不加async,进了compose,都是异步 this.compose(ctx).then(() => { //默认只能处理buffer 和string let _body = ctx.body; if (_body === '') { //如果没设置body 就给个默认值,状态码设置为404 res.statusCode = 404 _body = 'not found' return res.end(_body) } else if (_body instanceof Stream) { //koa也支持直接返回一个文件流,通过pipe就可以做到 return _body.pipe(res) } else if (typeof _body !== 'null' && typeof _body === 'object') { //对对象的处理 return res.end(JSON.stringify(_body)) } else if (_body == null) { //null 和undefined 无法直接调用toString 可以拼接一下 return res.end(_body + '') } else { //其他类型的直接toString return res.end(_body.toString()) } }).catch(err => { //这里为app添加了错误监听 //application继承events模块即可 this.emit('error', err) }) } 到这里核心部分就都解释完了,源码我会以压缩包形式给出。
最后来一波加餐,中间件的实现,本质就是函数返回一个接收ctx,next为参数的异步函数。
koa-static
function koaStatic(dirname) { return async function (ctx, next) { try { let filepath = path.join(dirname, ctx.path) const stat = fs.statSync(filepath); if (stat.isDirectory()) { filepath = path.join(filepath, './index.html') if (fs.existsSync(filepath)) { ctx.set('Content-Type', `text/html;charset=utf-8`) ctx.body = fs.createReadStream(filepath) } else { await next() } } else { if (fs.existsSync(filePath)) { //来自第三方包mime const mimeType = mime.getType(filepath) ctx.set('Content-Type', `${mimeType};charset=utf-8`) ctx.body = fs.createReadStream(filepath) } else { await next() } } } catch (error) { await next() } }}//根据目录查找对应文件 找到返回 找不到nextapp.use(koaStatic(__dirname))app.use(koaStatic(path.resolve(__dirname, 'public')))
相关链接
defineGetter: https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Object/defineGetter
源码压缩包
[*]https://github.com/lengyuexin/code
文档来源:51CTO技术博客https://blog.51cto.com/u_14219805/3008747
页:
[1]