在nodejs中,通过puppeteer来获取web页面中的window.performance对象,从而分析页面的性能。下面直接上代码。

const puppeteer = require('puppeteer');const path = require("path");const logger=require("./log");const log = logger.getPuppeteerRecordLogger() ;/* 启动浏览器*/ async function launchBrowser(){ //启动浏览器实例 [puppeteer.createBrowserFetcher([options])] let browser = await puppeteer.launch({ // 若是手动下载的chromium需要指定chromium地址, 默认引用地址为 /项目目录/node_modules/puppeteer/.local-chromium/ //executablePath: '/Users/huqiyang/Documents/project/z/chromium/Chromium.app/Contents/MacOS/Chromium', //如果是访问https页面 此属性会忽略https错误 ignoreHTTPSErrors: true, // 关闭headless模式, 不会打开浏览器 headless: true, //浏览器启动参数 https://peter.sh/experiments/chromium-command-line-switches/ --timeout args:['--disk-cache-size=0','--disable-cache','--disable-infobars','--window-size=800,600','--ignore-certificate-errors','--enable-feaures'], //是否为每个选项卡自动打开DevTools面板。 如果此选项为true,则headless选项将设置为false。 devtools: false, //Defaults to 30000 (30 seconds). Pass 0 to disable timeout. timeout: 0 //放慢puppeteer执行的动作,方便调试 //slowMo: 250 }); return browser ;}async function saveHarlog(url,dirPath,filename){ let homesite = url ; //保存的文件路径 let harFilePath = path.join(dirPath,filename) ; //处理URL if(!(url.startsWith('http://') || url.startsWith('https://'))){ url = "http://" + url ; } //打开浏览器 let browser = await launchBrowser() ; //创建一个新页面 //let page = await browser.newPage(); const page = (await browser.pages())[0]; try{ await page.goto(url,{ timeout:0 }); /* 在page上下文中得到window.performance.timing */ const timing = await page.evaluate( _ => { const {navigationStart,unloadEventStart,unloadEventEnd, redirectStart,redirectEnd,fetchStart,domainLookupStart, domainLookupEnd,connectStart,connectEnd,secureConnectionStart, requestStart,responseStart,responseEnd,domLoading,domInteractive, domContentLoadedEventStart,domContentLoadedEventEnd,domComplete, loadEventStart,loadEventEnd} = window.performance.timing; return ({navigationStart:navigationStart, unloadEventStart:unloadEventStart, unloadEventEnd:unloadEventEnd, redirectStart:redirectStart, redirectEnd:redirectEnd, fetchStart:fetchStart, domainLookupStart:domainLookupStart, domainLookupEnd:domainLookupEnd, connectStart:connectStart, connectEnd:connectEnd, secureConnectionStart:secureConnectionStart, requestStart:requestStart, responseStart:responseStart, responseEnd:responseEnd, domLoading:domLoading, domInteractive:domInteractive, domContentLoadedEventStart:domContentLoadedEventStart, domContentLoadedEventEnd:domContentLoadedEventEnd, domComplete:domComplete, loadEventStart:loadEventStart, loadEventEnd:loadEventEnd}) }) log.info('--->' + JSON.stringify(timing)) ; if(timing){ //long 型的毫秒数。上一个文档卸载(unload)结束时的UNIX时间戳。如果没有上一个文档,这个值会和fetchStart相同。 let navigationStart = timing.navigationStart ; //long 型的毫秒数.表征了unload事件抛出时的UNIX时间戳,如果没有上一个文档, 这个值会返回0 let unloadEventStart = timing.unloadEventStart ; //long 型的毫秒数,表征了unload事件处理完成时的UNIX时间戳。如果没有上一个文档, 这个值会返回0 let unloadEventEnd = timing.unloadEventEnd ; //long 型的毫秒数,表征了第一个HTTP重定向开始时的UNIX时间戳。如果没有重定向,或者重定向中的一个不同源,这个值会返回0. let redirectStart = timing.redirectStart ; //long 型的毫秒数,表征了最后一个HTTP重定向完成时(也就是说是HTTP响应的最后一个比特直接被收到的时间)的UNIX时间戳。如果没有重定向,或者重定向中的一个不同源,这个值会返回0. let redirectEnd = timing.redirectEnd ; //long 型的毫秒数,表征了浏览器准备好使用HTTP请求来获取(fetch)文档的UNIX时间戳。这个时间点会在检查任何应用缓存之前。 let fetchStart = timing.fetchStart ; //long 型的毫秒数,表征了域名查询开始的UNIX时间戳。如果使用了持续连接(persistent connection),或者这个信息存储到了缓存或者本地资源上,这个值将和fetchStart一致。 let domainLookupStart = timing.domainLookupStart ; //long 型的毫秒数,表征了域名查询结束的UNIX时间戳。如果使用了持续连接(persistent connection),或者这个信息存储到了缓存或者本地资源上,这个值将和 fetchStart一致。 let domainLookupEnd = timing.domainLookupEnd ; //long 型的毫秒数,返回HTTP请求开始向服务器发送时的Unix毫秒时间戳。如果使用持久连接(persistent connection),则返回值等同于fetchStart属性的值。 let connectStart = timing.connectStart ; //long 型的毫秒数,返回浏览器与服务器之间的连接建立时的Unix毫秒时间戳。如果建立的是持久连接,则返回值等同于fetchStart属性的值。连接建立指的是所有握手和认证过程全部结束。 let connectEnd = timing.connectEnd ; //long 型的毫秒数,返回浏览器与服务器开始安全链接的握手时的Unix毫秒时间戳。如果当前网页不要求安全连接,则返回0 let secureConnectionStart = timing.secureConnectionStart ; //long 型的毫秒数,返回浏览器向服务器发出HTTP请求时(或开始读取本地缓存时)的Unix毫秒时间戳 let requestStart = timing.requestStart ; //long 型的毫秒数,返回浏览器从服务器收到(或从本地缓存读取)第一个字节时的Unix毫秒时间戳。如果传输层在开始请求之后失败并且连接被重开,该属性将会被数制成新的请求的相对应的发起时间。 let responseStart = timing.responseStart ; //long 型的毫秒数,返回浏览器从服务器收到(或从本地缓存读取,或从本地资源读取)最后一个字节时(如果在此之前HTTP连接已经关闭,则返回关闭时)的Unix毫秒时间戳。 let responseEnd = timing.responseEnd ; //long 型的毫秒数,返回当前网页DOM结构开始解析时(即Document.readyState属性变为“loading”、相应的 readystatechange事件触发时)的Unix毫秒时间戳。 let domLoading = timing.domLoading ; //long 型的毫秒数,返回当前网页DOM结构结束解析、开始加载内嵌资源时(即Document.readyState属性变为“interactive”、相应的readystatechange事件触发时)的Unix毫秒时间戳 let domInteractive = timing.domInteractive ; //ong 型的毫秒数,返回当解析器发送DOMContentLoaded 事件,即所有需要被执行的脚本已经被解析时的Unix毫秒时间戳 let domContentLoadedEventStart = timing.domContentLoadedEventStart ; //long 型的毫秒数,返回当所有需要立即执行的脚本已经被执行(不论执行顺序)时的Unix毫秒时间戳 let domContentLoadedEventEnd = timing.domContentLoadedEventEnd ; //long 型的毫秒数,返回当前文档解析完成,即Document.readyState 变为 'complete'且相对应的readystatechange 被触发时的Unix毫秒时间戳 let domComplete = timing.domComplete ; //long 型的毫秒数,返回该文档下,load事件被发送时的Unix毫秒时间戳。如果这个事件还未被发送,它的值将会是0 let loadEventStart = timing.loadEventStart ; //long 型的毫秒数,返回当load事件结束,即加载事件完成时的Unix毫秒时间戳。如果这个事件还未被发送,或者尚未完成,它的值将会是0 let loadEventEnd = timing.loadEventEnd ; //呈现了如何导航到当前文档的信息 //let navigation = performance.navigation ; } }catch(error){ log.info('resovle error :' + url + "; error message:" + error) ; }finally{ if(browser){ await browser.close(); } } }exports.launchBrowser = launchBrowser;exports.saveHarlog = saveHarlog;指标说明

//@param t -> timingasync function getPerformanceTiming (t) { if (!t) { log.info('not allow null'); return; } var times = {}; //【重要】页面加载完成的时间 //【原因】这几乎代表了用户等待页面可用的时间 times.loadPage = t.loadEventEnd - t.navigationStart; //【重要】解析 DOM 树结构的时间 //【原因】反省下你的 DOM 树嵌套是不是太多了! times.domReady = t.domComplete - t.responseEnd; //【重要】重定向的时间 //【原因】拒绝重定向!比如,http://example.com/ 就不该写成 http://example.com times.redirect = t.redirectEnd - t.redirectStart; //【重要】DNS 查询时间 //【原因】DNS 预加载做了么?页面内是不是使用了太多不同的域名导致域名查询的时间太长? // 可使用 HTML5 Prefetch 预查询 DNS ,见:[HTML5 prefetch](http://segmentfault.com/a/1190000000633364) times.lookupDomain = t.domainLookupEnd - t.domainLookupStart; //【重要】读取页面第一个字节的时间 //【原因】这可以理解为用户拿到你的资源占用的时间,加异地机房了么,加CDN 处理了么?加带宽了么?加 CPU 运算速度了么? // TTFB 即 Time To First Byte 的意思 // 维基百科:https://en.wikipedia.org/wiki/Time_To_First_Byte times.ttfb = t.responseStart - t.navigationStart; //【重要】内容加载完成的时间 //【原因】页面内容经过 gzip 压缩了么,静态资源 css/js 等压缩了么? times.request = t.responseEnd - t.requestStart; //【重要】执行 onload 回调函数的时间 //【原因】是否太多不必要的操作都放到 onload 回调函数里执行了,考虑过延迟加载、按需加载的策略么? times.loadEvent = t.loadEventEnd - t.loadEventStart; // DNS 缓存时间 times.appcache = t.domainLookupStart - t.fetchStart; // 卸载页面的时间 times.unloadEvent = t.unloadEventEnd - t.unloadEventStart; // TCP 建立连接完成握手的时间 times.connect = t.connectEnd - t.connectStart; return times;}