UNPKG

tencentcloud-edgeone-migration-nodejs-v2

Version:

tencentcloud cdn config copy to edgeone

560 lines (441 loc) 19.8 kB
# Node SDK TAM 整体架构 for 前端监控设计,node-sdk 只做简单的上报,服务端数据上报建议使用 [CLS](https://console.cloud.tencent.com/cls) 或者 [智研日志汇](https://zhiyan.woa.com/)。 ## 安装 ### NPM 在项目支持 NPM 时推荐使用 NPM 安装 Aegis SDK。 ```bash $ tnpm install --save @tencent/aegis-node-sdk ``` ::: warning 注意 ⚠️ - 从 1.25.1 版本开始,SDK内部服务发现已经去掉了对L5的支持,默认使用北极星作为服务发现 - 如果用户使用electron进行的是客户端的数据上报或者本地开发的话,建议使用host模式,具体使用方式见[配置文档](#配置文档)里的`selector.type`配置项 - 如果用户是在devcloud机器上调用aegis接口上报的,建议先执行 ```bash unset http_proxy unset https_proxy ``` ::: ## 使用 ### 初始化 使用非常简单,只需要新建一个 Aegis 实例,传入相应的配置即可: ```javascript import Aegis from '@tencent/aegis-node-sdk'; const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', // 项目上报ID uin: 'xxx', // 用户唯一 ID(可选) }) ``` ## 日志 Aegis SDK 会主动收集用户的一些性能和错误日志,开发者可以根据不同的参数来配置哪些日志需要上报,以及上报的日志具体信息。 ### 日志类型 全部日志类型如下: ```javascript { logType: 'custom', name: '自定义测速' } { logType: 'event', name: '自定义事件' } { logType: 'log', name: '日志' } { logType: 'performance', name: '页面测速' } { logType: 'pv', name: '页面PV' } { logType: 'speed', name: '接口和静态资源测速' } { logType: 'vitals', name: 'web vitals' } ``` ### 日志等级 全部日志等级如下: ```javascript { level: 1, name: '白名单日志' }, { level: 2, name: '一般日志' }, { level: 4, name: '错误日志' }, { level: 8, name: 'Promise 错误' }, { level: 16, name: 'Ajax 请求异常' }, { level: 32, name: 'JS 加载异常' }, { level: 64, name: '图片加载异常' }, { level: 128, name: 'css 加载异常' }, { level: 256, name: 'console.error' }, { level: 512, name: '音视频资源异常' } { level: 1024, name: 'retcode 异常' } { level: 2048, name: 'aegis report' } { level: 4096, name: 'PV' } { level: 8192, name: '自定义事件' } { level: 16384, name: '小程序 页面不存在' } { level: 32768, name: 'websocket错误' } { level: 65536, name: 'js bridge错误' } ``` ### 日志上报 创建完 Aegis 实例之后,就可以开心的上报日志啦 🥰,日志上报同样简单 ```javascript // info 可以上报任意字符串,数字,数组,对象,但是只有打开页面的用户在名单中才会上报 aegis.info('test'); aegis.info('test', 123, ['a', 'b', 'c', 1], {a: '123'}); // 也可以上报特定的对象,支持用户传ext参数和trace参数 // 注意这种 case 一定要传 msg 字段 aegis.info({ msg: 'test', ext1: 'ext1', ext2: 'ext2', ext3: 'ext3', trace: 'trace', }); // 不同于 info,infoAll 表示全量上报 aegis.infoAll({ msg: 'test', ext1: 'ext1', ext2: 'ext2', ext3: 'ext3', trace: 'trace', }); // error 用来表示 JS 错误日志,也是全量上报,一般用于开发者主动获取JS异常,然后进行上报 aegis.error({ msg: 'test', ext1: 'ext1', ext2: 'ext2', ext3: 'ext3', trace: 'trace', }); aegis.error(new Error('主动上报一个错误')); // report 默认是 aegis.report 的日志类型,但是现在你可以传入任何日志类型了 aegis.report({ msg: '这是一个ajax错误日志', level: Aegis.logType.AJAX_ERROR, ext1: 'ext1', ext2: 'ext2', ext3: 'ext3', trace: 'trace', }); ``` ## 实例方法 Aegis 实例暴露接口简单实用,目前 Aegis 实例有以下方法供您使用: `setConfig` 、 `info` 、 `infoAll` 、 `report` 、 `error` 、 `reportEvent` 、 `reportTime` 、 `time` 、 `timeEnd`、`destroy` ### setConfig 该方法用来修改实例配置,比如下面场景: 在实例化 Aegis 时需要传入配置对象 ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', uin: '777' }) ``` 很多情况下,并不能一开始就获取到用户的 `uin`,而等获取到用户的 `uin` 才开始实例化 Aegis,如果这期间发生了错误 Aegis 将监听不到。`uin` 的设置可以在获取到用户的时候: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx' }) // 拿到uin之后... aegis.setConfig({ uin: '888' }) ``` ### info、infoAll、report 这三个方法是 Aegis 提供的主要上报手段。 ```javascript aegis.info('上报一条白名单日志,这两种情况这条日志才会报到后台:1、打开页面的用户在名单中;2、对应的页面发生了错误🤨'); aegis.infoAll('上报了一条日志,该上报与info唯一的不同就在于,所有用户都会上报'); aegis.report(new Error('上报一个错误')); ``` ### reportEvent 该方法可用来上报自定义事件,平台将会自动统计上报事件的各项指标,诸如:PV、UV、平台分布等... reportEvent 可以支持两种类型上报参数类型,一种是字符串类型 ```javascript aegis.reportEvent('XXX请求成功'); ``` 一种是对象类型,ext1 ext2 ext3 默认使用 new Aegis 的时候传入的参数,自定义事件上报的时候,可以覆盖默认值。 ```javascript aegis.reportEvent({ name: 'XXX请求成功', ext1: '额外参数1', ext2: '额外参数2', ext3: '额外参数3', }) ``` 注意,额外参数的三个 key 是固定的,目前只支持 ext1 ext2 ext3。 ### reportTime 该方法可用来上报自定义测速,例如: ```javascript // 假如‘onload’的时间是1s aegis.reportTime('onload', 1000); ``` 或者如果需要使用额外参数,可以传入对象类型参数,ext1,ext2,ext3 会覆盖默认值: ```javascript aegis.reportTime({ name: 'onload', // 自定义测速 name duration: 1000, // 自定义测速耗时(0 - 60000) ext1: 'test1', ext2: 'test2', ext3: 'test3', }); ``` > `onload` 可以修改为其他的命名。 ### time、timeEnd 该方法同样可用来上报自定义测速,适用于两个时间点之间时长的计算并上报,例如: ```javascript aegis.time('complexOperation'); /** * . * . * 做了很久的复杂操作之后。。。 * . * . */ aegis.timeEnd('complexOperation'); /** 此时日志已经报上去了😄**/ ``` > `complexOperation` 同样可以修改为其他的命名。 > 自定义测速是用户上报任意值,服务端对其进行统计和计算,因为服务端不能做脏数据处理,因此建议用户在上报端进行统计值限制,防止脏数据对整体产生影响。 > 目前 Aegis 只支持 0-60000 的数值计算,如果大于该值,建议进行合理改造。 > 高频率的自定义测速上报尽量使用 reportTime。time 和 timeEnd 上报会存在上报值覆盖的问题。比如 aegis.time(aaa), 在调用 aegis.timeEnd(aaa) 之前,又调用了一次 aegis.time(aaa), 则上报的时间为 timeEnd 时间 - 第二次 time 的时间。 ### reportSpeedLog 该方法用来上报cgi测速功能,上报完可到开发者平台的 “CGI测速” 栏目看到数据,如下: ![cgi测速](https://nowpic.gtimg.com/feeds_pic/PiajxSqBRaEILjSDiaDdp9RXb4hNJVcxegBAsoqwibkicrpx98RNG98a1Q/) 使用方法: ```javascript aegis.reportSpeedLog({ url: 'https://www.xxx.com/some/api/url', isHttps: true, method: 'get', duration: 900, // 耗时,单位:ms ret: 0, // retcode,默认为:unknown status: 200, // http status }) ``` ### destroy 该方法用于销毁 sdk 实例,销毁后,不再进行数据上报 ```javascript aegis.destroy(); ``` ## 白名单 白名单功能是适用于开发者希望对某些特定的用户上报更多的日志,但是又不希望太多上报来影响到全部日志数据,并且减少用户的接口请求次数,因为 TAM 设定了白名单的逻辑。 1. 白名单用户会上报全部的 API 请求信息,包括接口请求和请求结果。 2. 白名单用户可以使用 info 接口信息数据上报。 3. info vs infoAll:在开发者实际体验过程中,白名单用户可以添加更多的日志,并且使用 info 进行上报。infoAll 会对所有用户无差别进行上报,因此可能导致日志量上报巨大。 4. 通过接口 whitelist 来判断当前用户是否是白名单用户,白名单用户的返回结果会绑定在 aegis 实例上 (aegis.isWhiteList) 用来给开发者使用。 5. 用了减少开发者使用负担,白名单用户是团队有效,可以在 [组织管理-白名单管理](https://tam.woa.com/role/group-whitelist-manage) 内创建白名单,则团队下全部项目都生效。 ## 钩子函数 ### onBeforeRequest 该钩子函数会在所有请求发出前调用,参数中会传入请求的所有内容,必须返回待发送内容。 ```javascript function changeURLArg(url,arg,arg_val) { var pattern=arg+' = ([^&]*)'; var replaceText = arg+'='+arg_val; if (url.match(pattern)) { var tmp = '/('+ arg+'=)([^&]*)/gi'; tmp = url.replace(eval(tmp),replaceText); return tmp; } return url; } const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', onBeforeRequest(log) { if (log.type === 'performance') { // 页面测速,此时可以修改log内容,如修改页面测速platform log.url = changeURLArg(log.url, 'platform', type) } return log } }); // SEND_TYPE { // LOG = 'log', // 日志 // SPEED = 'speed', // 接口和静态资源测速 // PERFORMANCE = 'performance', // 页面测速 // OFFLINE = 'offline', // 离线日志上传 // WHITE_LIST = 'whiteList', // 白名单 // VITALS = 'vitals', // vitals // PV = 'pv', // 自定义pv // EVENT = 'event', // 自定义事件 // CUSTOM = 'custom', // 自定义测速 // SDK_ERROR = 'sdkError', // sdk报错 // } ``` ### beforeReport 1. 该钩子将会在日志上报(对应上报接口为 /collect?)前执行,例如 ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', beforeReport(log) { // 监听到下面抛出的错误 console.log(log); // {level: "4", msg: "发生错误啦!!!!"} return log; } }); throw new Error('发生错误啦!!!!'); ``` 2. 当该钩子返回 false 时,本条日志将不会进行上报,该功能可用来过滤某些不需要上报的错误,例如: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', beforeReport(log) { if (log.level === '4' && log.msg && log.msg.indexOf('碍眼的错误') !== -1) { return false } return log; } }); throw new Error('碍眼的错误'); // 该错误将不会被上报 ``` 上面例子中,当上报的错误内容包含 碍眼的错误 几个关键字时,将不会上报至前端监控后台中。 ### beforeReportSpeed 1. 该钩子将会在测速数据上报前(/speed?)被执行,例如: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', reportApiSpeed: true, reportAssetSpeed: true, beforeReportSpeed(msg) { console.log(msg); // {url: "https://localhost:3001/example.e31bb0bc.js", method: "get", duration: 7.4, status: 200, type: "static"} return msg } }); ``` 上述 msg 将会有以下几个字段: ```sh url: 该资源的请求地址; type: 该资源的类型,目前有 fetch 、 static 两种,当为 fetch 时,Aegis 将会把该资源当成 API 请求进行上报,static 时则视为静态资源; duration: 该资源请求耗时; method: 请求该资源时使用的 http method; status: 服务器返回状态码; payload: 提供给开发人员的完整资源请求信息(此数据不上报到 Aegis 后台,用户可自行操作) 完整的数据结构如下: payload.type - 表示该资源请求的类型,用于区分原始请求类型,可取值为:'fetch', 'xhr' payload.sourceURL - 表示完整的 URL 请求连接 payload.status - 表示请求状态码 payload.headers - 包含所有的请求头,且 value 值都为字符串 payload.data - 表示完整的请求资源,用户可自定义操作(当请求类型为 fetch 时,表示 response 对象;当请求类型为 XHR 时,表示 XMLHttpRequest 对象) ``` 上面的例子中,每当 Aegis 收集到一个资源的加载详情时,将会以该资源的加载情况(上面返回的 msg)作为参数调用 beforeReportSpeed 钩子。 2. 如果您配置了该钩子,Aegis 最终的上报内容将以钩子的执行结果为准。例如: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', reportApiSpeed: true, reportAssetSpeed: true, beforeReportSpeed(msg) { msg.type = 'static'; return msg; } }); ``` 上面的代码中,将所有的 msg.type 设置为 static,这意味着所有的资源都将被当成静态资源进行上报,API 请求也将被报至静态资源中。 3. 使用该钩子,您可以校准 Aegis 类型判断错误的请求。 假如您有一条接口 https://example.com/api,该接口的响应头 Content-Type 为 text/html。正常情况下,RUM 会将该资源当成静态资源进行上报。但在您的业务中,该接口就必须视为 API 请求进行上报,您可以给 Aegis 配置如下钩子进行校正: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', reportApiSpeed: true, reportAssetSpeed: true, beforeReportSpeed(msg) { if (msg.url === 'https://example.com/api') { msg.type = 'fetch'; } } }); ``` 4. 您还可以屏蔽某些资源的测速上报,例如: ```javascript const aegis = new Aegis({ id: 'pGUVFTCZyewxxxxx', reportApiSpeed: true, reportAssetSpeed: true, beforeReportSpeed(msg) { // 地址中包含‘https://example.com/api’的都不上报 if (msg.url.indexOf('https://example.com/api') !== -1) { // 返回 ‘false’ 将阻止本条测速日志的上报 return false } } }); ``` ### beforeRequest 该钩子将会在日志上报前执行,例如: ```javascript const aegis = new Aegis({ id: "pGUVFTCZyewxxxxx", beforeRequest: function(msg) { //{logs: {…}, logType: "log"} console.log(msg); } }); ``` 其中,`msg` 将会有以下几个字段: > 1.`logs`: 上报的日志内容; > 2.`logType`: 日志类型,有以下值 custom:自定义测速,event:自定义事件,log:日志,pv:页面PV,speed:接口和静态资源测速 logType等于`custom`时,logs的值如下: ```sh {name: "白屏时间", duration: 3015.7000000178814, ext1: '', ext2: '', ext3: '', from: ''} ``` logType等于`event`时,logs的值如下: ```sh {name: "ios", ext1: "", ext2: "", ext3: ""} ``` logType等于`log`时,logs的值如下: ```sh { level: '1', msg: '接口请求日志(白名单日志)' } // 具体level信息参考日志等级 ``` 当该钩子返回 `false` 时,本条日志将不会进行上报,该功能可用来过滤某些不需要上报的错误,例如: ```javascript const aegis = new Aegis({ id: "pGUVFTCZyewxxxxx", beforeRequest: function(data) { // 入参 data 的数据结构:{logs: {…}, logType: "log"} if (data.logType === 'log' && data.logs.msg.indexOf('otheve.beacon.qq.com') > -1) { // 拦截:日志类型为 log,且内容包含 otheve.beacon.qq.com 的请求 return false; } // 入参 data 数据结构:{logs: {}, logType: "speed"} if (data.logType === 'speed' && data.logs.url.indexOf('otheve.beacon.qq.com') > -1) { // 拦截:日志类型为 speed,并且接口 url 包含 otheve.beacon.qq.com 的请求 return false; } if (data.logType === 'performance') { // 修改:将性能数据的首屏渲染时间改为2s data.logs.firstScreenTiming = 2000; } return data; } } }); throw new Error('碍眼的错误'); // 该错误将不会被上报 ``` 上面例子中,当上报的错误内容包含 `碍眼的错误` 几个关键字时,将不会上报至 Aegis 后台中。 ### afterRequest 该勾子将会在测速数据上报后被执行,例如: ```javascript const aegis = new Aegis({ id: "pGUVFTCZyewxxxxx", reportApiSpeed: true, reportAssetSpeed: true, afterRequest: function(msg) { // {isErr: false, result: Array(1), logType: "log", logs: Array(4)} console.log(msg); } }); ``` 其中,`msg` 将会有以下几个字段: > 1.`isErr`: 请求上报接口是否错误; > 2.`result`: 上报接口的返回结果; > 3.`logs`: 上报的日志内容; > 4.`logType`: 日志类型,有以下值speed:接口和静态资源测速,performance:页面测速,vitals:web vitals,event:自定义事件,custom:自定义测速; ## 配置文档 | 配置 | 描述 | | -------- | -------- | | id | 必须,string,默认 无。<br>开发者平台分配的项目上报ID | | uin | 建议,string <br>业务定义的当前用户的唯一标识符,用来查找固定用户的日志。白名单上报时将根据该字段判定用户是否在白名单中,字段仅支持`字母数字@=._-`,正则表达式: `/^[@=.0-9a-zA-Z_-]{1,60}$/` | | version | 可选,string,默认 sdk 版本号。<br>当前上报版本,当页面使用了pwa或者存在离线包时,可用来判断当前的上报是来自哪一个版本的代码,仅支持`字母数字.,:_-`,长度在 60 位以内 `/^[0-9a-zA-Z.,:_-]{1,60}$/` | | delay | 可选,number,默认 1000 ms。<br>上报节流时间,在该时间段内的上报将会合并到一个上报请求中。 | | repeat | 可选,number,默认 5。<br>重复上报次数,对于同一个错误超过多少次不上报。 | | env | 可选,enum,默认 Aegis.environment.production。 <br> 当前项目运行所处的环境。| | hostUrl | 可选,默认是 `https://aegis.qq.com`。<br>影响全部上报数据的 host 地址,下面几个 url 地址设置后会覆盖对应的上报地址 | | url | 可选,string,默认 'https://aegis.qq.com/collect'。<br>日志上报地址 | | pvUrl | 可选,string, 默认 'https://aegis.qq.com/collect/pv' <br> pv 上报地址 | | whiteListUrl | 可选,string,默认 'https://aegis.qq.com/collect/whitelist'。<br>白名单确认接口 <br>如果想要关闭白名单接口请求,可以传空字符串| | offlineUrl | 可选,string,默认 'https://aegis.qq.com/collect/offline'。<br> 离线日志上报地址 | | eventUrl | 可选,string,默认 'https://aegis.qq.com/collect/events'。<br> 自定义事件上报地址 | | customTimeUrl | 可选,string,默认 'https://aegis.qq.com/speed/custom'。<br>自定义测速上报地址 | | speedUrl | 可选,string,默认 'https://aegis.qq.com/speed'。<br>测速日志上报地址 | | ext1 | 可选,string,自定义上报的额外维度,上报的时候可以被覆盖 | | ext2 | 可选,string,自定义上报的额外维度,上报的时候可以被覆盖 | | ext3 | 可选,string,自定义上报的额外维度,上报的时候可以被覆盖 | 在实例化 Aegis node sdk 时可传入额外配置: | 配置 | 描述 | | -------- | -------- | |selector|可选, object 寻址相关配置,不传默认会使用北极星进行寻址| |selector.type|必须, string 可选值:ip,host,polaris(北极星)| |protocol|可选,string, selector.type=host 时,默认为https, 其他类型默认为http,未来会支持trpc协议| ## 注意事项 - 由于aegis实例是node应用初始时构造的,因此建议使用者将 uin/sessionId等直接打到 `aegis.info`接口中,在平台查询时直接使用关键字查询。