50行代码串行Promise,koa洋葱模型原来是这么实现?
1. 前言
写相对很难的源码,耗费了自己的时间和精力,也没收获多少阅读点赞,其实是一件挺受打击的事情。从阅读量和读者受益方面来看,不能促进作者持续输出文章。
所以转变思路,写一些相对通俗易懂的文章。其实源码也不是想象的那么难,至少有很多看得懂。
之前写过 koa 源码文章学习 koa 源码的整体架构,浅析koa洋葱模型原理和co原理比较长,读者朋友大概率看不完,所以本文从koa-compose
50行源码讲述。
本文涉及到的 koa-compose 仓库 文件,整个index.js
文件代码行数虽然不到 50
行,而且测试用例test/test.js
文件 300
余行,但非常值得我们学习。
歌德曾说:读一本好书,就是在和高尚的人谈话。 同理可得:读源码,也算是和作者的一种学习交流的方式。
阅读本文,你将学到:
1. 熟悉 koa-compose 中间件源码、可以应对面试官相关问题
2. 学会使用测试用例调试源码
3. 学会 jest 部分用法
2. 环境准备
2.1 克隆 koa-compose 项目
本文仓库地址 koa-compose-analysis,求个star
~
# 可以直接克隆我的仓库,我的仓库保留的 compose 仓库的 git 记录
git clone https://github.com/lxchuan12/koa-compose-analysis.git
cd koa-compose/compose
npm i
顺带说下:我是怎么保留 compose
仓库的 git
记录的。
# 在 github 上新建一个仓库 `koa-compose-analysis` 克隆下来
git clone https://github.com/lxchuan12/koa-compose-analysis.git
cd koa-compose-analysis
git subtree add --prefix=compose https://github.com/koajs/compose.git main
# 这样就把 compose 文件夹克隆到自己的 git 仓库了。且保留的 git 记录
关于更多 git subtree
,可以看这篇文章用 Git Subtree 在多个 Git 项目间双向同步子项目,附简明使用手册
接着我们来看怎么根据开源项目中提供的测试用例调试源码。
2.2 根据测试用例调试 compose 源码
用VSCode
(我的版本是 1.60
)打开项目,找到 compose/package.json
,找到 scripts
和 test
命令。
// compose/package.json
{
"name": "koa-compose",
// debug (调试)
"scripts": {
"eslint": "standard --fix .",
"test": "jest"
},
}
在scripts
上方应该会有debug
或者调试
字样。点击debug
(调试),选择 test
。
接着会执行测试用例test/test.js
文件。终端输出如下图所示。
接着我们调试 compose/test/test.js
文件。
我们可以在 45行
打上断点,重新点击 package.json
=> srcipts
=> test
进入调试模式。
如下图所示。
接着按上方的按钮,继续调试。在compose/index.js
文件中关键的地方打上断点,调试学习源码事半功倍。
顺便提一下几个调试相关按钮。
- 继续(F5)
- 单步跳过(F10)
- 单步调试(F11)
- 单步跳出(Shift + F11)
- 重启(Ctrl + Shift + F5)
- 断开链接(Shift + F5)
接下来,我们跟着测试用例学源码。
3. 跟着测试用例学源码
分享一个测试用例小技巧:我们可以在测试用例处加上only
修饰。
// 例如
it.only('should work', async () => {})
这样我们就可以只执行当前的测试用例,不关心其他的,不会干扰调试。
3.1 正常流程
打开 compose/test/test.js
文件,看第一个测试用例。
// compose/test/test.js
'use strict'
/* eslint-env jest */
const compose = require('..')
const assert = require('assert')
function wait (ms) {
return new Promise((resolve) => setTimeout(resolve, ms || 1))
}
// 分组
describe('Koa Compose', function () {
it.only('should work', async () => {
const arr = []
const stack = []
stack.push(async (context, next) => {
arr.push(1)
await wait(1)
await next()
await wait(1)
arr.push(6)
})
stack.push(async (context, next) => {
arr.push(2)
await wait(1)
await next()
await wait(1)
arr.push(5)
})
stack.push(async (context, next) => {
arr.push(3)
await wait(1)
await next()
await wait(1)
arr.push(4)
})
await compose(stack)({})
// 最后输出数组是 [1,2,3,4,5,6]
expect(arr).toEqual(expect.arrayContaining([1, 2, 3, 4, 5, 6]))
})
}
大概看完这段测试用例,context
是什么,next
又是什么。
在koa
的文档上有个非常代表性的中间件 gif
图。
而compose
函数作用就是把添加进中间件数组的函数按照上面 gif
图的顺序执行。
3.1.1 compose 函数
简单来说,compose
函数主要做了两件事情。
- 接收一个参数,校验参数是数组,且校验数组中的每一项是函数。
- 返回一个函数,这个函数接收两个参数,分别是
context
和next
,这个函数最后返回Promise
。
/**
* Compose `middleware` returning
* a fully valid middleware comprised
* of all those which are passed.
*
* @param {Array} middleware
* @return {Function}
* @api public
*/
function compose (middleware) {
// 校验传入的参数是数组,校验数组中每一项是函数
if (!Array.isArray(middleware)) throw new TypeError('Middleware stack must be an array!')
for (const fn of middleware) {
if (typeof fn !== 'function') throw new TypeError('Middleware must be composed of functions!')
}
/**
* @param {Object} context
* @return {Promise}
* @api public
*/
return function (context, next) {
// last called middleware #
let index = -1
return dispatch(0)
function dispatch(i){
// 省略,下文讲述
}
}
}
接着我们来看 dispatch
函数。
3.1.2 dispatch 函数
function dispatch (i) {
// 一个函数中多次调用报错
// await next()
// await next()
if (i <= index) return Promise.reject(new Error('next() called multiple times'))
index = i
// 取出数组里的 fn1, fn2, fn3...
let fn = middleware[i]
// 最后 相等,next 为 undefined
if (i === middleware.length) fn = next
// 直接返回 Promise.resolve()
if (!fn) return Promise.resolve()
try {
return Promise.resolve(fn(context, dispatch.bind(null, i + 1)))
} catch (err) {
return Promise.reject(err)
}
}
值得一提的是:bind
函数是返回一个新的函数。第一个参数是函数里的this指向(如果函数不需要使用this
,一般会写成null
)。
这句fn(context, dispatch.bind(null, i + 1)
,i + 1
是为了 let fn = middleware[i]
取middleware
中的下一个函数。
也就是 next
是下一个中间件里的函数。也就能解释上文中的 gif
图函数执行顺序。
测试用例中数组的最终顺序是[1,2,3,4,5,6]
。
3.1.3 简化 compose 便于理解
自己动手调试之后,你会发现 compose
执行后就是类似这样的结构(省略 try catch
判断)。
// 这样就可能更好理解了。
// simpleKoaCompose
const [fn1, fn2, fn3] = stack;
const fnMiddleware = function(context){
return Promise.resolve(
fn1(context, function next(){
return Promise.resolve(
fn2(context, function next(){
return Promise.resolve(
fn3(context, function next(){
return Promise.resolve();
})
)
})
)
})
);
};
也就是说
koa-compose
返回的是一个Promise
,从中间件(传入的数组)
中取出第一个函数,传入context
和第一个next
函数来执行。
第一个next
函数里也是返回的是一个Promise
,从中间件(传入的数组)
中取出第二个函数,传入context
和第二个next
函数来执行。
第二个next
函数里也是返回的是一个Promise
,从中间件(传入的数组)
中取出第三个函数,传入context
和第三个next
函数来执行。
第三个...
以此类推。最后一个中间件中有调用next
函数,则返回Promise.resolve
。如果没有,则不执行next
函数。
这样就把所有中间件串联起来了。这也就是我们常说的洋葱模型。
不得不说非常惊艳,“玩还是大神会玩”。
3.2 错误捕获
it('should catch downstream errors', async () => {
const arr = []
const stack = []
stack.push(async (ctx, next) => {
arr.push(1)
try {
arr.push(6)
await next()
arr.push(7)
} catch (err) {
arr.push(2)
}
arr.push(3)
})
stack.push(async (ctx, next) => {
arr.push(4)
throw new Error()
})
await compose(stack)({})
// 输出顺序 是 [ 1, 6, 4, 2, 3 ]
expect(arr).toEqual([1, 6, 4, 2, 3])
})
相信理解了第一个测试用例和 compose
函数,也是比较好理解这个测试用例了。这一部分其实就是对应的代码在这里。
try {
return Promise.resolve(fn(context, dispatch.bind(null, i + 1)))
} catch (err) {
return Promise.reject(err)
}
3.3 next 函数不能调用多次
it('should throw if next() is called multiple times', () => {
return compose([
async (ctx, next) => {
await next()
await next()
}
])({}).then(() => {
throw new Error('boom')
}, (err) => {
assert(/multiple times/.test(err.message))
})
})
这一块对应的则是:
index = -1
dispatch(0)
function dispatch (i) {
if (i <= index) return Promise.reject(new Error('next() called multiple times'))
index = i
}
调用两次后 i
和 index
都为 1
,所以会报错。
compose/test/test.js
文件中总共 300余行,还有很多测试用例可以按照文中方法自行调试。
4. 总结
虽然koa-compose
源码 50行 不到,但如果是第一次看源码调试源码,还是会有难度的。其中混杂着高阶函数、闭包、Promise
、bind
等基础知识。
通过本文,我们熟悉了 koa-compose
中间件常说的洋葱模型,学会了部分 jest
用法,同时也学会了如何使用现成的测试用例去调试源码。
相信学会了通过测试用例调试源码后,会觉得源码也没有想象中的那么难。
开源项目,一般都会有很全面的测试用例。除了可以给我们学习源码调试源码带来方便的同时,也可以给我们带来的启发:自己工作中的项目,也可以逐步引入测试工具,比如 jest
。
此外,读开源项目源码是我们学习业界大牛设计思想和源码实现等比较好的方式。
链接:https://juejin.cn/post/7005375860509245471