istanbul-middleware(后面简称 middleware)本质上是一个基于 express 的网站。但其包含了数个针对 istanbul 覆盖率收集及报告生成的 http 接口,因此可用于作为单独的覆盖率报告生成网站。
URL | Description |
---|---|
GET / | 动态生成覆盖率 html 报告。和平时单测生成的静态版本一样,可以通过点击逐级深入,查看更细节的覆盖率数据。 |
POST /reset | 把覆盖率数据重置成基线(可以理解成清空当前覆盖率数据) |
GET /download | 下载一个包含 json 、lcov、html 三种格式覆盖率报告的压缩包 |
POST /client | 用于从浏览器主动发送覆盖率对象。覆盖率对象必须是 json 格式,且发送时 header 中必须有 Content-type: application/json 。这个对象需要和当前服务端已有的统计数据保持一致。补充:即不能把不同程序的覆盖率数据都一起发给同一个 middleware 服务端。 |
/coverage
路径的 handler ,处理上述的覆盖率接口请求。im.hookLoader(__dirname);
,app.use('/coverage', im.createHandler());
通过打断点的方式来查看server端自动插装过程
hookLoader
传入两个参数,第一个参数主要匹配除了node_modules外其他全部需要插装的问价路径,第二个参数主要是插装的一些配置选项hookLoader
中加入Instrumenter
对象,然后通过本地方法instrumenter.instrumentSync.bind(instrumenter)
获得一个transformer
(本地方法),把相关参数传入,postLoadHook(matcherFn, transformer, opts.verbose)
获得postLoadHookFn
hook.hookRequire
(本地方法)的方法对服务端代码进行插装,通过postLoadHook,postLoadHookFn
传入相关js文件,通过saveBaseline
函数对插装的函数进行保存,在saveBaseline
中baselineCoverage可以获得插装后的js文件信息function saveBaseline(file) {
var coverageObject = getCoverageObject(),
fileCoverage;
if (coverageObject && coverageObject[file]) {
fileCoverage = coverageObject[file];
if (!baselineCoverage[file]) {
baselineCoverage[file] = {
s: clone(fileCoverage.s),
f: clone(fileCoverage.f),
b: clone(fileCoverage.b)
};
}
}
}
app.use('/coverage', im.createHandler());
渲染相关接口页面,并提供相关接口。clientHandler
中可以通过res.send(instrumented);
把插装后的代码数据上传try {
instrumented = core.getInstrumenter().instrumentSync(contents, fullPath);
if (verbose) { console.log('Sending instrumented code for: ' + fullPath + ', url:' + req.url); }
res.setHeader('Content-type', 'application/javascript');
return res.send(instrumented);
} catch (ex) {
console.warn('Error instrumenting file:' + fullPath);
return next();
}
app.use(coverage.createClientHandler(publicDir, { matcher: matcher }));
,返回app