monitor-track 1.10.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +31 -0
- package/README.md +101 -0
- package/dist/CHANGELOG.md +21 -0
- package/dist/README.md +101 -0
- package/dist/cjs/config/global.d.ts +17 -0
- package/dist/cjs/config/index.d.ts +33 -0
- package/dist/cjs/constant.d.ts +2 -0
- package/dist/cjs/handlers/error.d.ts +23 -0
- package/dist/cjs/handlers/index.d.ts +3 -0
- package/dist/cjs/handlers/pv.d.ts +24 -0
- package/dist/cjs/handlers/user-activity.d.ts +9 -0
- package/dist/cjs/index.d.ts +30 -0
- package/dist/cjs/index.js +1485 -0
- package/dist/cjs/main.d.ts +1 -0
- package/dist/cjs/reporter.d.ts +198 -0
- package/dist/cjs/types/config.d.ts +59 -0
- package/dist/cjs/types/global.d.ts +70 -0
- package/dist/cjs/types/index.d.ts +189 -0
- package/dist/cjs/utils/index.d.ts +45 -0
- package/dist/esm/_virtual/_tslib.js +26 -0
- package/dist/esm/config/global.d.ts +17 -0
- package/dist/esm/config/global.js +271 -0
- package/dist/esm/config/index.d.ts +33 -0
- package/dist/esm/config/index.js +32 -0
- package/dist/esm/constant.d.ts +2 -0
- package/dist/esm/constant.js +4 -0
- package/dist/esm/handlers/error.d.ts +23 -0
- package/dist/esm/handlers/error.js +181 -0
- package/dist/esm/handlers/index.d.ts +3 -0
- package/dist/esm/handlers/pv.d.ts +24 -0
- package/dist/esm/handlers/pv.js +99 -0
- package/dist/esm/handlers/user-activity.d.ts +9 -0
- package/dist/esm/handlers/user-activity.js +66 -0
- package/dist/esm/index.d.ts +30 -0
- package/dist/esm/index.js +115 -0
- package/dist/esm/main.d.ts +1 -0
- package/dist/esm/package.json.js +3 -0
- package/dist/esm/reporter.d.ts +198 -0
- package/dist/esm/reporter.js +359 -0
- package/dist/esm/types/config.d.ts +59 -0
- package/dist/esm/types/global.d.ts +70 -0
- package/dist/esm/types/index.d.ts +189 -0
- package/dist/esm/utils/index.d.ts +45 -0
- package/dist/esm/utils/index.js +354 -0
- package/dist/index.js +1534 -0
- package/dist/package.json +41 -0
- package/package.json +42 -0
- package/rollup.config.js +66 -0
- package/scripts/cp.js +11 -0
- package/src/config/global.ts +309 -0
- package/src/config/index.ts +53 -0
- package/src/constant.ts +2 -0
- package/src/handlers/error.ts +187 -0
- package/src/handlers/index.ts +3 -0
- package/src/handlers/pv.ts +124 -0
- package/src/handlers/user-activity.ts +69 -0
- package/src/index.ts +127 -0
- package/src/main.ts +8 -0
- package/src/reporter.ts +372 -0
- package/src/types/config.ts +58 -0
- package/src/types/global.ts +73 -0
- package/src/types/index.ts +227 -0
- package/src/typing.d.ts +15 -0
- package/src/utils/index.ts +371 -0
- package/tsconfig.esm.json +14 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,1534 @@
|
|
|
1
|
+
var ShuyunTrack = (function (axios, uuid, uaParserJs, rrweb, ErrorStackParser, html2canvas) {
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
|
|
5
|
+
|
|
6
|
+
function _interopNamespace(e) {
|
|
7
|
+
if (e && e.__esModule) return e;
|
|
8
|
+
var n = Object.create(null);
|
|
9
|
+
if (e) {
|
|
10
|
+
Object.keys(e).forEach(function (k) {
|
|
11
|
+
if (k !== 'default') {
|
|
12
|
+
var d = Object.getOwnPropertyDescriptor(e, k);
|
|
13
|
+
Object.defineProperty(n, k, d.get ? d : {
|
|
14
|
+
enumerable: true,
|
|
15
|
+
get: function () { return e[k]; }
|
|
16
|
+
});
|
|
17
|
+
}
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
n["default"] = e;
|
|
21
|
+
return Object.freeze(n);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
var axios__default = /*#__PURE__*/_interopDefaultLegacy(axios);
|
|
25
|
+
var rrweb__namespace = /*#__PURE__*/_interopNamespace(rrweb);
|
|
26
|
+
var ErrorStackParser__default = /*#__PURE__*/_interopDefaultLegacy(ErrorStackParser);
|
|
27
|
+
var html2canvas__default = /*#__PURE__*/_interopDefaultLegacy(html2canvas);
|
|
28
|
+
|
|
29
|
+
/******************************************************************************
|
|
30
|
+
Copyright (c) Microsoft Corporation.
|
|
31
|
+
|
|
32
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
|
33
|
+
purpose with or without fee is hereby granted.
|
|
34
|
+
|
|
35
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
36
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
37
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
38
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
39
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
40
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
41
|
+
PERFORMANCE OF THIS SOFTWARE.
|
|
42
|
+
***************************************************************************** */
|
|
43
|
+
|
|
44
|
+
var __assign = function() {
|
|
45
|
+
__assign = Object.assign || function __assign(t) {
|
|
46
|
+
for (var s, i = 1, n = arguments.length; i < n; i++) {
|
|
47
|
+
s = arguments[i];
|
|
48
|
+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
|
|
49
|
+
}
|
|
50
|
+
return t;
|
|
51
|
+
};
|
|
52
|
+
return __assign.apply(this, arguments);
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
function __awaiter(thisArg, _arguments, P, generator) {
|
|
56
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
57
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
58
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
59
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
60
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
61
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function __generator(thisArg, body) {
|
|
66
|
+
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
|
|
67
|
+
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
|
|
68
|
+
function verb(n) { return function (v) { return step([n, v]); }; }
|
|
69
|
+
function step(op) {
|
|
70
|
+
if (f) throw new TypeError("Generator is already executing.");
|
|
71
|
+
while (g && (g = 0, op[0] && (_ = 0)), _) try {
|
|
72
|
+
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
|
|
73
|
+
if (y = 0, t) op = [op[0] & 2, t.value];
|
|
74
|
+
switch (op[0]) {
|
|
75
|
+
case 0: case 1: t = op; break;
|
|
76
|
+
case 4: _.label++; return { value: op[1], done: false };
|
|
77
|
+
case 5: _.label++; y = op[1]; op = [0]; continue;
|
|
78
|
+
case 7: op = _.ops.pop(); _.trys.pop(); continue;
|
|
79
|
+
default:
|
|
80
|
+
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
|
|
81
|
+
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
|
|
82
|
+
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
|
|
83
|
+
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
|
|
84
|
+
if (t[2]) _.ops.pop();
|
|
85
|
+
_.trys.pop(); continue;
|
|
86
|
+
}
|
|
87
|
+
op = body.call(thisArg, _);
|
|
88
|
+
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
|
|
89
|
+
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
typeof SuppressedError === "function" ? SuppressedError : function (error, suppressed, message) {
|
|
94
|
+
var e = new Error(message);
|
|
95
|
+
return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
|
|
96
|
+
};
|
|
97
|
+
|
|
98
|
+
var shuyunTrackId = 'shuyun-track-id';
|
|
99
|
+
var shuyunTrackSessionId = 'shuyun-track-session-id';
|
|
100
|
+
|
|
101
|
+
/*
|
|
102
|
+
* @Author: Mark.Zhang
|
|
103
|
+
* @Description 配置项相关参数及方法
|
|
104
|
+
*/
|
|
105
|
+
/** */
|
|
106
|
+
/**
|
|
107
|
+
* @description config配置项默认值
|
|
108
|
+
*/
|
|
109
|
+
var Config = {
|
|
110
|
+
reportUrl: '',
|
|
111
|
+
projectID: '',
|
|
112
|
+
maxLength: 1000,
|
|
113
|
+
spa: false,
|
|
114
|
+
hash: false,
|
|
115
|
+
enableBehavior: true,
|
|
116
|
+
enableError: true,
|
|
117
|
+
enableVisualTrack: false,
|
|
118
|
+
ignore: {
|
|
119
|
+
urls: [],
|
|
120
|
+
errors: [],
|
|
121
|
+
apis: []
|
|
122
|
+
}
|
|
123
|
+
};
|
|
124
|
+
/**
|
|
125
|
+
* @description 设置config配置项
|
|
126
|
+
* @param config 配置项
|
|
127
|
+
*/
|
|
128
|
+
function setConfig(config) {
|
|
129
|
+
Object.assign(Config, config);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/*
|
|
133
|
+
* @Author: Mark.Zhang
|
|
134
|
+
* @Date: 2020-10-26 11:10:04
|
|
135
|
+
* @Description 资源加载错误及js错误相关的处理方法
|
|
136
|
+
*/
|
|
137
|
+
var eventsMatrix = [[]];
|
|
138
|
+
rrweb__namespace.record({
|
|
139
|
+
emit: function (event, isCheckout) {
|
|
140
|
+
// isCheckout 是一个标识,告诉你重新制作了快照
|
|
141
|
+
if (isCheckout) {
|
|
142
|
+
eventsMatrix.push([]);
|
|
143
|
+
}
|
|
144
|
+
if (eventsMatrix.length > 2) {
|
|
145
|
+
eventsMatrix.shift();
|
|
146
|
+
}
|
|
147
|
+
var lastEvents = eventsMatrix[eventsMatrix.length - 1];
|
|
148
|
+
lastEvents.push(event);
|
|
149
|
+
},
|
|
150
|
+
// 每30秒重新制作快照
|
|
151
|
+
checkoutEveryNms: 30 * 1000
|
|
152
|
+
});
|
|
153
|
+
/**
|
|
154
|
+
* @description 错误事件触发后的操作
|
|
155
|
+
*/
|
|
156
|
+
function handleError(event) {
|
|
157
|
+
setReportValue('type', 'error');
|
|
158
|
+
if (event.type === 'error') {
|
|
159
|
+
if (event instanceof ErrorEvent) {
|
|
160
|
+
setCaughtError(event);
|
|
161
|
+
}
|
|
162
|
+
else {
|
|
163
|
+
setResourceError(event);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
else if (event.type === 'unhandledrejection') {
|
|
167
|
+
setPromiseError(event);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
/**
|
|
171
|
+
* @description 设置错误信息
|
|
172
|
+
*/
|
|
173
|
+
function setCaughtError(event) {
|
|
174
|
+
var _a, _b;
|
|
175
|
+
if (ignoreError(event.message)) {
|
|
176
|
+
return;
|
|
177
|
+
}
|
|
178
|
+
var filename = event.filename, line = event.lineno, column = event.colno, functionName = '', stackFrame = JSON.stringify([]);
|
|
179
|
+
if (event.error && event.error.stack) {
|
|
180
|
+
var error = ErrorStackParser__default["default"].parse(event.error);
|
|
181
|
+
stackFrame = JSON.stringify(error);
|
|
182
|
+
var info = error.find(function (item) { return item.fileName && !/node_modules|vendor|bundle/.test(item.fileName); });
|
|
183
|
+
if (info) {
|
|
184
|
+
filename = info.fileName;
|
|
185
|
+
line = Number(info.lineNumber);
|
|
186
|
+
column = Number(info.columnNumber);
|
|
187
|
+
functionName = info.functionName || '';
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
setReportValue('error', {
|
|
191
|
+
subType: 'js',
|
|
192
|
+
line: line,
|
|
193
|
+
column: column,
|
|
194
|
+
message: event.message,
|
|
195
|
+
filename: filename,
|
|
196
|
+
functionName: functionName,
|
|
197
|
+
stack: ((_b = (_a = event === null || event === void 0 ? void 0 : event.error) === null || _a === void 0 ? void 0 : _a.stack) === null || _b === void 0 ? void 0 : _b.substring(0, Config.maxLength)) || '',
|
|
198
|
+
stackFrame: stackFrame,
|
|
199
|
+
events: getUserEvents(false)
|
|
200
|
+
});
|
|
201
|
+
report(getReport());
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* @description 设置异步错误信息
|
|
205
|
+
*/
|
|
206
|
+
function setPromiseError(event) {
|
|
207
|
+
var message = event.reason;
|
|
208
|
+
if (event.reason && typeof event.reason === 'object') {
|
|
209
|
+
message = JSON.stringify(event.reason);
|
|
210
|
+
if (message === '{}') {
|
|
211
|
+
message = event.reason.stack;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
if (ignoreError(message)) {
|
|
215
|
+
return;
|
|
216
|
+
}
|
|
217
|
+
setReportValue('error', {
|
|
218
|
+
subType: 'async',
|
|
219
|
+
message: message,
|
|
220
|
+
events: getUserEvents(false)
|
|
221
|
+
});
|
|
222
|
+
report(getReport());
|
|
223
|
+
}
|
|
224
|
+
/**
|
|
225
|
+
* @description 设置资源错误信息
|
|
226
|
+
*/
|
|
227
|
+
function setResourceError(event) {
|
|
228
|
+
var target = event.target;
|
|
229
|
+
if (ignoreError(target.outerHTML)) {
|
|
230
|
+
return;
|
|
231
|
+
}
|
|
232
|
+
setReportValue('error', {
|
|
233
|
+
subType: 'resource',
|
|
234
|
+
message: target.outerHTML,
|
|
235
|
+
filename: target.src || target.currentSrc,
|
|
236
|
+
stack: target.localName.toUpperCase().substring(0, Config.maxLength),
|
|
237
|
+
events: getUserEvents(false)
|
|
238
|
+
});
|
|
239
|
+
report(getReport());
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* 忽略的error
|
|
243
|
+
* @param errorMsg 错误信息
|
|
244
|
+
*/
|
|
245
|
+
function ignoreError(errorMsg) {
|
|
246
|
+
var _a;
|
|
247
|
+
var errors = ((_a = Config === null || Config === void 0 ? void 0 : Config.ignore) === null || _a === void 0 ? void 0 : _a.errors) || [];
|
|
248
|
+
var existIgnoreError = errors.findIndex(function (item) {
|
|
249
|
+
if (typeof item === 'string') {
|
|
250
|
+
return item === errorMsg;
|
|
251
|
+
}
|
|
252
|
+
else if (Object.prototype.toString.call(item) === '[object Object]') {
|
|
253
|
+
if (item.regExp && typeof item.input === 'string') {
|
|
254
|
+
var regex = new RegExp(item.input, 'g');
|
|
255
|
+
return regex.test(errorMsg);
|
|
256
|
+
}
|
|
257
|
+
return false;
|
|
258
|
+
}
|
|
259
|
+
else {
|
|
260
|
+
return false;
|
|
261
|
+
}
|
|
262
|
+
});
|
|
263
|
+
return existIgnoreError > -1;
|
|
264
|
+
}
|
|
265
|
+
/**
|
|
266
|
+
* 获取用户最近得操作,视频录像时间应当在30秒到60秒之间
|
|
267
|
+
*/
|
|
268
|
+
var lastGetTime = Date.now(); //缓存操作录像,防止段时间内大量获取录像数据导滞内存溢出
|
|
269
|
+
function getUserEvents(notUseCache) {
|
|
270
|
+
var events;
|
|
271
|
+
if (Date.now() - lastGetTime > 2000 || notUseCache) {
|
|
272
|
+
lastGetTime = Date.now();
|
|
273
|
+
if (eventsMatrix.length >= 2) {
|
|
274
|
+
var finalVideoData = eventsMatrix[eventsMatrix.length - 1];
|
|
275
|
+
events = JSON.stringify(eventsMatrix[eventsMatrix.length - 2].concat(finalVideoData));
|
|
276
|
+
if (events.length > 1024 * 1024 * 15 && JSON.stringify(finalVideoData).length > 1024 * 1024) {
|
|
277
|
+
//如果录像数据量太大且最后那个分片的数据量不算小,那么缩短上报的录像时长
|
|
278
|
+
events = JSON.stringify(eventsMatrix[eventsMatrix.length - 1]);
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
else {
|
|
282
|
+
events = JSON.stringify(eventsMatrix);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return events;
|
|
286
|
+
}
|
|
287
|
+
function getFullScreenShoot(filename) {
|
|
288
|
+
var name = filename || uuid.v4();
|
|
289
|
+
return html2canvas__default["default"](document.body).then(function (canvas) {
|
|
290
|
+
var _a;
|
|
291
|
+
var urlData = canvas.toDataURL('image/png', 1);
|
|
292
|
+
var bytes = window.atob(urlData.split(',')[1]);
|
|
293
|
+
var mime = (_a = urlData.split(',')[0].match(/:(.*?);/)) === null || _a === void 0 ? void 0 : _a[1];
|
|
294
|
+
var ab = new ArrayBuffer(bytes.length);
|
|
295
|
+
var ia = new Uint8Array(ab);
|
|
296
|
+
for (var i = 0; i < bytes.length; i++) {
|
|
297
|
+
ia[i] = bytes.charCodeAt(i);
|
|
298
|
+
}
|
|
299
|
+
var file = new File([ab], "".concat(name, ".png"), { type: mime });
|
|
300
|
+
return file;
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
/** 上报数据映射 */
|
|
305
|
+
var reportMap = {
|
|
306
|
+
/** SDK 版本 */
|
|
307
|
+
version: '_v',
|
|
308
|
+
/** 项目ID */
|
|
309
|
+
projectID: 'pid',
|
|
310
|
+
/** location.host */
|
|
311
|
+
host: 'host',
|
|
312
|
+
/** 用户uuid */
|
|
313
|
+
uuid: '_id',
|
|
314
|
+
/** 上报日志的时间戳 */
|
|
315
|
+
time: 'time',
|
|
316
|
+
/** 源页面 */
|
|
317
|
+
originPage: 'o',
|
|
318
|
+
/** 当前页面 */
|
|
319
|
+
page: 'p',
|
|
320
|
+
/** 自定义payload */
|
|
321
|
+
customPayload: 'cP',
|
|
322
|
+
/** 信息类型 */
|
|
323
|
+
type: 't',
|
|
324
|
+
/** viewport的宽高 */
|
|
325
|
+
viewport: 'vp',
|
|
326
|
+
/** 屏幕宽高 */
|
|
327
|
+
screen: 'sc',
|
|
328
|
+
/** 页面title */
|
|
329
|
+
pageTitle: 'pt',
|
|
330
|
+
/** 页面referrer */
|
|
331
|
+
referrer: 'ref',
|
|
332
|
+
/** 页面编码 */
|
|
333
|
+
charset: 'char',
|
|
334
|
+
/** language */
|
|
335
|
+
language: 'lan',
|
|
336
|
+
/** vendor, 浏览器厂商 */
|
|
337
|
+
navigatorVendor: 'nV',
|
|
338
|
+
/** 网络连接类型 */
|
|
339
|
+
connectionType: 'cT',
|
|
340
|
+
/** 浏览器名称 */
|
|
341
|
+
browserName: 'bN',
|
|
342
|
+
/** 浏览器版本 */
|
|
343
|
+
browserVersion: 'bV',
|
|
344
|
+
/** 浏览器引擎名称 */
|
|
345
|
+
engineName: 'enN',
|
|
346
|
+
/** 浏览器引擎版本 */
|
|
347
|
+
engineVersion: 'enV',
|
|
348
|
+
/** 操作系统名称 */
|
|
349
|
+
osName: 'osN',
|
|
350
|
+
/** 操作系统版本 */
|
|
351
|
+
osVersion: 'osV',
|
|
352
|
+
pv: {
|
|
353
|
+
/** 子类型 */
|
|
354
|
+
subType: 'suT',
|
|
355
|
+
/** 停留时间 */
|
|
356
|
+
stayTime: 'sT'
|
|
357
|
+
},
|
|
358
|
+
error: {
|
|
359
|
+
/** 子类型 */
|
|
360
|
+
subType: 'suT',
|
|
361
|
+
/** 错误行 */
|
|
362
|
+
line: 'line',
|
|
363
|
+
/** 错误列 */
|
|
364
|
+
column: 'col',
|
|
365
|
+
/** 错误信息 */
|
|
366
|
+
message: 'msg',
|
|
367
|
+
/** 错误文件 */
|
|
368
|
+
filename: 'fil',
|
|
369
|
+
/** 错误函数名 */
|
|
370
|
+
functionName: 'fn',
|
|
371
|
+
/** 错误栈信息 */
|
|
372
|
+
stack: '_st',
|
|
373
|
+
/** 错误帧 */
|
|
374
|
+
stackFrame: '_sf',
|
|
375
|
+
/** 视频录制数据 */
|
|
376
|
+
events: 'evt',
|
|
377
|
+
/** 页面截图 */
|
|
378
|
+
picture: 'pic'
|
|
379
|
+
},
|
|
380
|
+
ua: {
|
|
381
|
+
/** 子类型 */
|
|
382
|
+
subType: 'suT',
|
|
383
|
+
/** x轴坐标 */
|
|
384
|
+
x: 'x',
|
|
385
|
+
/** y轴坐标 */
|
|
386
|
+
y: 'y',
|
|
387
|
+
/** 元素路径 */
|
|
388
|
+
path: 'path'
|
|
389
|
+
},
|
|
390
|
+
/** 性能数据 */
|
|
391
|
+
perf: {
|
|
392
|
+
navigation: {
|
|
393
|
+
/**
|
|
394
|
+
*一个无符号短整型,表示是如何导航到这个页面的。可能的值如下:
|
|
395
|
+
TYPE_NAVIGATE (0)
|
|
396
|
+
当前页面是通过点击链接,书签和表单提交,或者脚本操作,或者在url中直接输入地址,type值为0
|
|
397
|
+
TYPE_RELOAD (1)
|
|
398
|
+
点击刷新页面按钮或者通过Location.reload()方法显示的页面,type值为1
|
|
399
|
+
The page was accessed by clicking the Reload button or via the Location.reload() method.
|
|
400
|
+
TYPE_BACK_FORWARD (2)
|
|
401
|
+
页面通过历史记录和前进后退访问时。type值为2
|
|
402
|
+
TYPE_RESERVED (255)
|
|
403
|
+
任何其他方式,type值为255
|
|
404
|
+
*/
|
|
405
|
+
type: 't',
|
|
406
|
+
/** 无符号短整型,表示在到达这个页面之前重定向了多少次。 */
|
|
407
|
+
redirectCount: 'rc'
|
|
408
|
+
},
|
|
409
|
+
memory: {
|
|
410
|
+
/** 上下文内可用堆的最大体积,以字节计算。 */
|
|
411
|
+
jsHeapSizeLimit: 'hsl',
|
|
412
|
+
/** 已分配的堆体积,以字节计算。 */
|
|
413
|
+
totalJSHeapSize: 'ths',
|
|
414
|
+
/** 当前 JS 堆活跃段(segment)的体积,以字节计算。 */
|
|
415
|
+
usedJSHeapSize: 'uhs'
|
|
416
|
+
},
|
|
417
|
+
/** 返回性能测量开始时的时间的高精度时间戳。 */
|
|
418
|
+
timeOrigin: 'to',
|
|
419
|
+
timing: {
|
|
420
|
+
/** 是一个无符号long long 型的毫秒数,返回浏览器与服务器之间的连接建立时的Unix毫秒时间戳。如果建立的是持久连接,则返回值等同于fetchStart属性的值。连接建立指的是所有握手和认证过程全部结束。 */
|
|
421
|
+
connectEnd: 'ce',
|
|
422
|
+
/** 是一个无符号long long 型的毫秒数,返回HTTP请求开始向服务器发送时的Unix毫秒时间戳。如果使用持久连接(persistent connection),则返回值等同于fetchStart属性的值 */
|
|
423
|
+
connectStart: 'cs',
|
|
424
|
+
/** 是一个无符号long long 型的毫秒数,返回当前文档解析完成,即Document.readyState 变为 'complete'且相对应的readystatechange (en-US) 被触发时的Unix毫秒时间戳。 */
|
|
425
|
+
domComplete: 'dc',
|
|
426
|
+
/** 是一个无符号long long 型的毫秒数,返回当所有需要立即执行的脚本已经被执行(不论执行顺序)时的Unix毫秒时间戳。 */
|
|
427
|
+
domContentLoadedEventEnd: 'scLee',
|
|
428
|
+
/** 是一个无符号long long 型的毫秒数,返回当解析器发送DOMContentLoaded (en-US) 事件,即所有需要被执行的脚本已经被解析时的Unix毫秒时间戳。 */
|
|
429
|
+
domContentLoadedEventStart: 'dcLes',
|
|
430
|
+
/** 是一个无符号long long 型的毫秒数,返回当前网页DOM结构结束解析、开始加载内嵌资源时(即Document.readyState属性变为“interactive”、相应的readystatechange (en-US)事件触发时)的Unix毫秒时间戳。 */
|
|
431
|
+
domInteractive: 'di',
|
|
432
|
+
/** 是一个无符号long long 型的毫秒数,返回当前网页DOM结构开始解析时(即Document.readyState属性变为“loading”、相应的 readystatechange (en-US)事件触发时)的Unix毫秒时间戳。 */
|
|
433
|
+
domLoading: 'dl',
|
|
434
|
+
/** 是一个无符号long long 型的毫秒数,表征了域名查询结束的UNIX时间戳。如果使用了持续连接(persistent connection),或者这个信息存储到了缓存或者本地资源上,这个值将和 PerformanceTiming.fetchStart一致。 */
|
|
435
|
+
domainLookupEnd: 'dle',
|
|
436
|
+
/** 是一个无符号long long 型的毫秒数,表征了域名查询开始的UNIX时间戳。如果使用了持续连接(persistent connection),或者这个信息存储到了缓存或者本地资源上,这个值将和 PerformanceTiming.fetchStart一致。 */
|
|
437
|
+
domainLookupStart: 'dls',
|
|
438
|
+
/** 是一个无符号long long 型的毫秒数,表征了浏览器准备好使用HTTP请求来获取(fetch)文档的UNIX时间戳。这个时间点会在检查任何应用缓存之前。 */
|
|
439
|
+
fetchStart: 'fs',
|
|
440
|
+
/** 是一个无符号long long 型的毫秒数,返回当load (en-US)事件结束,即加载事件完成时的Unix毫秒时间戳。如果这个事件还未被发送,或者尚未完成,它的值将会是0. */
|
|
441
|
+
loadEventEnd: 'lee',
|
|
442
|
+
/** 是一个无符号long long 型的毫秒数,返回该文档下,load (en-US)事件被发送时的Unix毫秒时间戳。如果这个事件还未被发送,它的值将会是0。 */
|
|
443
|
+
loadEventStart: 'les',
|
|
444
|
+
/** 是一个无符号long long 型的毫秒数,表征了从同一个浏览器上下文的上一个文档卸载(unload)结束时的UNIX时间戳。如果没有上一个文档,这个值会和PerformanceTiming.fetchStart相同 */
|
|
445
|
+
navigationStart: 'ns',
|
|
446
|
+
/** 是一个无符号long long 型的毫秒数,表征了最后一个HTTP重定向完成时(也就是说是HTTP响应的最后一个比特直接被收到的时间)的UNIX时间戳。如果没有重定向,或者重定向中的一个不同源,这个值会返回0. */
|
|
447
|
+
redirectEnd: 're',
|
|
448
|
+
/** 是一个无符号long long 型的毫秒数,表征了第一个HTTP重定向开始时的UNIX时间戳。如果没有重定向,或者重定向中的一个不同源,这个值会返回0. */
|
|
449
|
+
redirectStart: 'redS',
|
|
450
|
+
/** 是一个无符号long long 型的毫秒数,返回浏览器向服务器发出HTTP请求时(或开始读取本地缓存时)的Unix毫秒时间戳。 */
|
|
451
|
+
requestStart: 'reqS',
|
|
452
|
+
/** 是一个无符号long long 型的毫秒数,返回浏览器从服务器收到(或从本地缓存读取,或从本地资源读取)最后一个字节时(如果在此之前HTTP连接已经关闭,则返回关闭时)的Unix毫秒时间戳。 */
|
|
453
|
+
responseEnd: 'resE',
|
|
454
|
+
/** 是一个无符号long long 型的毫秒数,返回浏览器从服务器收到(或从本地缓存读取)第一个字节时的Unix毫秒时间戳。如果传输层在开始请求之后失败并且连接被重开,该属性将会被数制成新的请求的相对应的发起时间。 */
|
|
455
|
+
responseStart: 'resS',
|
|
456
|
+
/** 是一个无符号long long 型的毫秒数,返回浏览器与服务器开始安全链接的握手时的Unix毫秒时间戳。如果当前网页不要求安全连接,则返回0。 */
|
|
457
|
+
secureConnectionStart: 'scs',
|
|
458
|
+
/** 是一个无符号long long 型的毫秒数,表征了unload (en-US)事件处理完成时的UNIX时间戳。如果没有上一个文档,or if the previous document, or one of the needed redirects, is not of the same origin, 这个值会返回0. */
|
|
459
|
+
unloadEventEnd: 'uee',
|
|
460
|
+
/** 是一个无符号long long 型的毫秒数,表征了unload (en-US)事件抛出时的UNIX时间戳。如果没有上一个文档,or if the previous document, or one of the needed redirects, is not of the same origin, 这个值会返回0. */
|
|
461
|
+
unloadEventStart: 'ees'
|
|
462
|
+
}
|
|
463
|
+
},
|
|
464
|
+
/** api请求相关 */
|
|
465
|
+
req: {
|
|
466
|
+
/** 请求类型,xhr或fetch */
|
|
467
|
+
requestType: 'rT',
|
|
468
|
+
/** 请求地址 */
|
|
469
|
+
responseURL: 'url',
|
|
470
|
+
/** 请求结果 */
|
|
471
|
+
status: 'stat',
|
|
472
|
+
/** 请求耗时 */
|
|
473
|
+
loadTime: 'lT',
|
|
474
|
+
/** 请求状态 */
|
|
475
|
+
statusText: 'sT',
|
|
476
|
+
/** 上报原因, slow或failed */
|
|
477
|
+
reason: 're',
|
|
478
|
+
/** 上报时带上的详细信息 */
|
|
479
|
+
detail: 'de'
|
|
480
|
+
},
|
|
481
|
+
vD: {
|
|
482
|
+
/** 埋点名称 */
|
|
483
|
+
trackName: 'tN',
|
|
484
|
+
/** 埋点参数 */
|
|
485
|
+
params: 'params'
|
|
486
|
+
},
|
|
487
|
+
manualReport: 'mR',
|
|
488
|
+
lifeCycleId: 'li',
|
|
489
|
+
sessionId: 'sI'
|
|
490
|
+
};
|
|
491
|
+
var xhrFunc = function (filename, userId, asyncUpdateId, file, res, callback, rest) {
|
|
492
|
+
var formData = new FormData();
|
|
493
|
+
for (var i in rest) {
|
|
494
|
+
if (Object.prototype.hasOwnProperty.call(rest, i)) {
|
|
495
|
+
formData.append(i, rest[i]);
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
formData.append('filename', filename);
|
|
499
|
+
formData.append('asyncUpdateId', asyncUpdateId); //用于上传到oss后异步更新到db的id
|
|
500
|
+
formData.append('userId', userId);
|
|
501
|
+
formData.append('files', file);
|
|
502
|
+
var xhr = new XMLHttpRequest();
|
|
503
|
+
xhr.addEventListener('error', function () { return res(); }, false);
|
|
504
|
+
xhr.open('POST', Config.reportUrl);
|
|
505
|
+
xhr.onreadystatechange = function () {
|
|
506
|
+
var _a;
|
|
507
|
+
if (xhr.readyState === 4) {
|
|
508
|
+
if (xhr.status === 200) {
|
|
509
|
+
try {
|
|
510
|
+
var response = xhr.responseText;
|
|
511
|
+
var resp = JSON.parse(response);
|
|
512
|
+
callback((_a = resp.result) === null || _a === void 0 ? void 0 : _a.response);
|
|
513
|
+
}
|
|
514
|
+
catch (_err) {
|
|
515
|
+
//
|
|
516
|
+
}
|
|
517
|
+
res();
|
|
518
|
+
}
|
|
519
|
+
else {
|
|
520
|
+
res();
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
};
|
|
524
|
+
xhr.send(formData);
|
|
525
|
+
};
|
|
526
|
+
var ERROR_MESSAGE_MAP = {};
|
|
527
|
+
function reportFunc(data) {
|
|
528
|
+
var _a, _b, _c;
|
|
529
|
+
var payload = convertToSchema(data, reportMap);
|
|
530
|
+
payload.time = new Date().getTime(); // 设置上报时间
|
|
531
|
+
var picturePromise = Promise.resolve();
|
|
532
|
+
var hasErrorEvent = false;
|
|
533
|
+
var filename = uuid.v4();
|
|
534
|
+
if (data.type === 'error') {
|
|
535
|
+
var message_1 = ((_a = data.error) === null || _a === void 0 ? void 0 : _a.message) || 'default';
|
|
536
|
+
if (((_b = data.error) === null || _b === void 0 ? void 0 : _b.subType) === 'resource') {
|
|
537
|
+
message_1 = 'resource';
|
|
538
|
+
}
|
|
539
|
+
if (ERROR_MESSAGE_MAP[message_1]) ;
|
|
540
|
+
else {
|
|
541
|
+
ERROR_MESSAGE_MAP[message_1] = true;
|
|
542
|
+
setTimeout(function () {
|
|
543
|
+
//节流效果,每隔3秒上报一次重复的错误录像和错误截图
|
|
544
|
+
delete ERROR_MESSAGE_MAP[message_1];
|
|
545
|
+
}, 3000);
|
|
546
|
+
var p_1 = payload.p, pid_1 = payload.pid, host_1 = payload.host, bN_1 = payload.bN, pt_1 = payload.pt;
|
|
547
|
+
if ((_c = data.error) === null || _c === void 0 ? void 0 : _c.events) {
|
|
548
|
+
hasErrorEvent = true;
|
|
549
|
+
var errorEvent_1 = data.error.events;
|
|
550
|
+
var delayTime = Math.ceil(Math.random() * 10000);
|
|
551
|
+
setTimeout(function () {
|
|
552
|
+
var blob = new Blob([errorEvent_1], { type: 'text/json' });
|
|
553
|
+
return new Promise(function (res) {
|
|
554
|
+
xhrFunc("".concat(filename, ".json"), data.uuid, filename, blob, res, function () { }, { p: p_1, pid: pid_1, host: host_1, bN: bN_1, pt: pt_1 });
|
|
555
|
+
});
|
|
556
|
+
}, delayTime < 3500 ? 3500 : delayTime);
|
|
557
|
+
}
|
|
558
|
+
picturePromise = getFullScreenShoot(filename)
|
|
559
|
+
.then(function (file) {
|
|
560
|
+
return new Promise(function (res) {
|
|
561
|
+
xhrFunc(file.name, data.uuid, filename, file, res, function (result) {
|
|
562
|
+
payload.error.pic = result;
|
|
563
|
+
}, { p: p_1, pid: pid_1, host: host_1, bN: bN_1, pt: pt_1 });
|
|
564
|
+
});
|
|
565
|
+
})["catch"](function (err) {
|
|
566
|
+
payload.error.picError = (err === null || err === void 0 ? void 0 : err.stack) || (err === null || err === void 0 ? void 0 : err.toString());
|
|
567
|
+
});
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
return picturePromise
|
|
571
|
+
.then(function () {
|
|
572
|
+
var _a;
|
|
573
|
+
if ((_a = payload.error) === null || _a === void 0 ? void 0 : _a.evt)
|
|
574
|
+
delete payload.error.evt;
|
|
575
|
+
if (window && window.navigator && typeof window.navigator.sendBeacon === 'function') {
|
|
576
|
+
var formData_1 = new FormData();
|
|
577
|
+
Object.keys(payload).forEach(function (key) {
|
|
578
|
+
var value = payload[key];
|
|
579
|
+
if (value !== null && value !== undefined) {
|
|
580
|
+
if (typeof value === 'object') {
|
|
581
|
+
value = JSON.stringify(value);
|
|
582
|
+
}
|
|
583
|
+
formData_1.append(key, value);
|
|
584
|
+
}
|
|
585
|
+
});
|
|
586
|
+
//eventErrorFilename,将错误录像的文件名附在错误上报的埋点里,便于将错误录像上报回调地址异步的更新到db
|
|
587
|
+
if (hasErrorEvent)
|
|
588
|
+
formData_1.append('eef', filename);
|
|
589
|
+
window.navigator.sendBeacon(Config.reportUrl, formData_1);
|
|
590
|
+
}
|
|
591
|
+
else {
|
|
592
|
+
if (hasErrorEvent)
|
|
593
|
+
payload.eef = filename;
|
|
594
|
+
new Image().src = "".concat(Config.reportUrl, "?").concat(serialize(payload));
|
|
595
|
+
}
|
|
596
|
+
})["catch"](function (_err) {
|
|
597
|
+
//
|
|
598
|
+
});
|
|
599
|
+
}
|
|
600
|
+
// 上报
|
|
601
|
+
function report(data) {
|
|
602
|
+
return new Promise(function (res) {
|
|
603
|
+
//在帧的空闲时间上报
|
|
604
|
+
var dataCopy = JSON.parse(JSON.stringify(data));
|
|
605
|
+
if (typeof window.requestIdleCallback === 'function') {
|
|
606
|
+
window.requestIdleCallback(function () { return reportFunc(dataCopy).then(res); }, { timeout: 2000 });
|
|
607
|
+
}
|
|
608
|
+
else {
|
|
609
|
+
setTimeout(function () { return reportFunc(dataCopy).then(res); }, 0);
|
|
610
|
+
}
|
|
611
|
+
});
|
|
612
|
+
}
|
|
613
|
+
/**
|
|
614
|
+
* @description 将数据转换成最终提交的数据,主要目的是简化数据的key的长度,从而降低荷载大小
|
|
615
|
+
* @param data 要转换的数据
|
|
616
|
+
* @param map 要转换的数据的字段映射
|
|
617
|
+
* @param redundancyData 冗余数据
|
|
618
|
+
*/
|
|
619
|
+
function convertToSchema(data, map, redundancyData) {
|
|
620
|
+
var reportData = {};
|
|
621
|
+
if (redundancyData) {
|
|
622
|
+
map = __assign(__assign({}, map), redundancyData);
|
|
623
|
+
}
|
|
624
|
+
for (var _i = 0, _a = Object.entries(map); _i < _a.length; _i++) {
|
|
625
|
+
var result = _a[_i];
|
|
626
|
+
var _b = result, key = _b[0], mapKey = _b[1];
|
|
627
|
+
if (key === 'manualReport') {
|
|
628
|
+
reportData[mapKey] = data[key];
|
|
629
|
+
}
|
|
630
|
+
else {
|
|
631
|
+
//字段值为数组的场景,直接转换
|
|
632
|
+
if (typeof data[key] === 'object' && !Array.isArray(data[key])) {
|
|
633
|
+
if (data[key]) {
|
|
634
|
+
if (redundancyData && Object.prototype.hasOwnProperty.call(redundancyData, key)) {
|
|
635
|
+
reportData[key] = data[key];
|
|
636
|
+
}
|
|
637
|
+
else {
|
|
638
|
+
reportData[key] = convertToSchema(data[key], map[key]);
|
|
639
|
+
}
|
|
640
|
+
}
|
|
641
|
+
else {
|
|
642
|
+
reportData[key] = data[key];
|
|
643
|
+
}
|
|
644
|
+
}
|
|
645
|
+
else {
|
|
646
|
+
// 排除undefined
|
|
647
|
+
if (data[key] !== undefined) {
|
|
648
|
+
reportData[mapKey] = data[key];
|
|
649
|
+
}
|
|
650
|
+
}
|
|
651
|
+
}
|
|
652
|
+
}
|
|
653
|
+
return reportData;
|
|
654
|
+
}
|
|
655
|
+
|
|
656
|
+
/* eslint-disable prefer-rest-params */
|
|
657
|
+
/** 路由栈数组,存储路由变化信息 */
|
|
658
|
+
var routerStack = [];
|
|
659
|
+
/** 路由触发的时间 */
|
|
660
|
+
var time = 0;
|
|
661
|
+
/**
|
|
662
|
+
* @description 派发pushState, replaceState 的监听
|
|
663
|
+
* @param type
|
|
664
|
+
*/
|
|
665
|
+
function _history(type) {
|
|
666
|
+
var origin = history[type];
|
|
667
|
+
return function () {
|
|
668
|
+
// @ts-ignore
|
|
669
|
+
var r = origin.apply(this, arguments);
|
|
670
|
+
var e = new Event(type);
|
|
671
|
+
// @ts-ignore
|
|
672
|
+
e.arguments = arguments;
|
|
673
|
+
window.dispatchEvent(e);
|
|
674
|
+
return r;
|
|
675
|
+
};
|
|
676
|
+
}
|
|
677
|
+
/**
|
|
678
|
+
* @description 处理history变化
|
|
679
|
+
* @param e history事件
|
|
680
|
+
*/
|
|
681
|
+
function handleHistoryChange(__e) {
|
|
682
|
+
visualTrackFunc();
|
|
683
|
+
setReport();
|
|
684
|
+
}
|
|
685
|
+
/**
|
|
686
|
+
* 设置信息并触发上报
|
|
687
|
+
*/
|
|
688
|
+
function setReport() {
|
|
689
|
+
var pageUrl = location.href;
|
|
690
|
+
/**
|
|
691
|
+
* 此判断的逻辑是因为spa且hash模式下,路由搜索参数更改也会触发handleHistoryChange事件
|
|
692
|
+
* 所以路由栈中的信息只保存无搜索参数的地址
|
|
693
|
+
*/
|
|
694
|
+
if (Config.hash && Config.spa) {
|
|
695
|
+
var currentUrlSplit = location.href.split('#');
|
|
696
|
+
var currentUrlPostfix = '';
|
|
697
|
+
if (currentUrlSplit[1]) {
|
|
698
|
+
var index = currentUrlSplit[1].indexOf('?');
|
|
699
|
+
if (index > -1) {
|
|
700
|
+
currentUrlPostfix = currentUrlSplit[1].substring(0, index);
|
|
701
|
+
}
|
|
702
|
+
else {
|
|
703
|
+
currentUrlPostfix = currentUrlSplit[1];
|
|
704
|
+
}
|
|
705
|
+
}
|
|
706
|
+
var currentUrlResult = "".concat(currentUrlSplit[0], "#").concat(currentUrlPostfix);
|
|
707
|
+
pageUrl = currentUrlResult;
|
|
708
|
+
}
|
|
709
|
+
var _a = setRouteStack(pageUrl), originPage = _a.originPage, page = _a.page;
|
|
710
|
+
// 愿页面等于当前页面,说明无路由变更。只有spa且hash模式下才会出现
|
|
711
|
+
if (originPage === page) {
|
|
712
|
+
return;
|
|
713
|
+
}
|
|
714
|
+
var currentTime = new Date().getTime();
|
|
715
|
+
// 页面停留时间
|
|
716
|
+
var stayTime = currentTime - time;
|
|
717
|
+
// 设置触发路由变化的时间
|
|
718
|
+
time = currentTime;
|
|
719
|
+
// 设置信息
|
|
720
|
+
setReportValue('originPage', originPage);
|
|
721
|
+
setReportValue('page', page);
|
|
722
|
+
setReportValue('type', 'pv');
|
|
723
|
+
setReportValue('pv', { stayTime: stayTime });
|
|
724
|
+
// 上报数据
|
|
725
|
+
report(getReport());
|
|
726
|
+
}
|
|
727
|
+
function setPVTime() {
|
|
728
|
+
time = new Date().getTime();
|
|
729
|
+
}
|
|
730
|
+
/**
|
|
731
|
+
* @description 添加路由栈信息
|
|
732
|
+
* @param page 页面信息
|
|
733
|
+
*/
|
|
734
|
+
function setRouteStack(page) {
|
|
735
|
+
if (typeof page === 'string') {
|
|
736
|
+
routerStack.push(page);
|
|
737
|
+
}
|
|
738
|
+
else {
|
|
739
|
+
routerStack = routerStack.concat(page);
|
|
740
|
+
}
|
|
741
|
+
// 路由栈只需保留两个来作为源页面和当前页面
|
|
742
|
+
routerStack = routerStack.slice(-2);
|
|
743
|
+
return {
|
|
744
|
+
originPage: routerStack[0],
|
|
745
|
+
page: routerStack[1] || routerStack[0]
|
|
746
|
+
};
|
|
747
|
+
}
|
|
748
|
+
|
|
749
|
+
var version = "1.10.0";
|
|
750
|
+
|
|
751
|
+
// 上报数据
|
|
752
|
+
var Report = {
|
|
753
|
+
uuid: '',
|
|
754
|
+
projectID: '',
|
|
755
|
+
host: '',
|
|
756
|
+
originPage: '',
|
|
757
|
+
page: '',
|
|
758
|
+
time: 0,
|
|
759
|
+
browserName: '',
|
|
760
|
+
browserVersion: '',
|
|
761
|
+
engineName: '',
|
|
762
|
+
engineVersion: '',
|
|
763
|
+
language: '',
|
|
764
|
+
navigatorVendor: '',
|
|
765
|
+
connectionType: '2g',
|
|
766
|
+
osName: '',
|
|
767
|
+
osVersion: '',
|
|
768
|
+
type: 'init',
|
|
769
|
+
viewport: '',
|
|
770
|
+
screen: '',
|
|
771
|
+
version: '',
|
|
772
|
+
charset: '',
|
|
773
|
+
pageTitle: '',
|
|
774
|
+
referrer: '',
|
|
775
|
+
pv: null,
|
|
776
|
+
ua: null,
|
|
777
|
+
error: null,
|
|
778
|
+
dpr: 1,
|
|
779
|
+
perf: null,
|
|
780
|
+
vD: undefined,
|
|
781
|
+
manualReport: undefined,
|
|
782
|
+
lifeCycleId: uuid.v4(),
|
|
783
|
+
sessionId: getSessionId()
|
|
784
|
+
};
|
|
785
|
+
/**
|
|
786
|
+
* @description 初始化上报数据并上报
|
|
787
|
+
*/
|
|
788
|
+
function initReport() {
|
|
789
|
+
if (document.readyState === 'complete') {
|
|
790
|
+
initReportFunc();
|
|
791
|
+
}
|
|
792
|
+
else {
|
|
793
|
+
// 注意:这里不要使用window.onload,因为一个项目window.onload只能用一次,如果这里用了就会影响宿主项目的功能
|
|
794
|
+
window.addEventListener('load', initReportFunc);
|
|
795
|
+
}
|
|
796
|
+
}
|
|
797
|
+
function initReportFunc() {
|
|
798
|
+
setRouteStack([location.href, location.href]);
|
|
799
|
+
setPVTime();
|
|
800
|
+
Report = __assign(__assign({}, getReport()), {
|
|
801
|
+
page: location.href,
|
|
802
|
+
originPage: location.href,
|
|
803
|
+
type: 'init',
|
|
804
|
+
perf: performance,
|
|
805
|
+
sessionId: getSessionId(),
|
|
806
|
+
lifeCycleId: uuid.v4()
|
|
807
|
+
});
|
|
808
|
+
report(Report);
|
|
809
|
+
}
|
|
810
|
+
/**
|
|
811
|
+
* @description 设置Report的值
|
|
812
|
+
* @param key Report key
|
|
813
|
+
* @param value Report 值
|
|
814
|
+
*/
|
|
815
|
+
function setReportValue(key, value) {
|
|
816
|
+
if (Object.prototype.hasOwnProperty.call(Report, key)) {
|
|
817
|
+
if (['pv', 'ua', 'error', 'request'].includes(key)) {
|
|
818
|
+
Report = __assign(__assign({}, Report), {
|
|
819
|
+
pv: null,
|
|
820
|
+
ua: null,
|
|
821
|
+
error: null,
|
|
822
|
+
req: null,
|
|
823
|
+
manualReport: undefined,
|
|
824
|
+
perf: null
|
|
825
|
+
});
|
|
826
|
+
}
|
|
827
|
+
Report[key] = value;
|
|
828
|
+
}
|
|
829
|
+
}
|
|
830
|
+
/**
|
|
831
|
+
* @description 获取上报数据
|
|
832
|
+
*/
|
|
833
|
+
function getReport() {
|
|
834
|
+
var nav = getNavigator();
|
|
835
|
+
var viewport = getViewport();
|
|
836
|
+
return __assign(__assign(__assign({}, Report), nav), {
|
|
837
|
+
version: version,
|
|
838
|
+
projectID: Config.projectID,
|
|
839
|
+
host: location.host,
|
|
840
|
+
uuid: getUid(),
|
|
841
|
+
viewport: viewport,
|
|
842
|
+
screen: "".concat(screen.width, " x ").concat(screen.height),
|
|
843
|
+
pageTitle: document.title,
|
|
844
|
+
referrer: document.referrer,
|
|
845
|
+
charset: document.charset,
|
|
846
|
+
customPayload: Config.customPayload,
|
|
847
|
+
dpr: window.devicePixelRatio
|
|
848
|
+
});
|
|
849
|
+
}
|
|
850
|
+
function ajaxEventTrigger(event) {
|
|
851
|
+
var ajaxEvent = new CustomEvent(event, {
|
|
852
|
+
detail: this
|
|
853
|
+
});
|
|
854
|
+
window.dispatchEvent(ajaxEvent);
|
|
855
|
+
}
|
|
856
|
+
var OldXHR = window.XMLHttpRequest;
|
|
857
|
+
function newXHR() {
|
|
858
|
+
var realXHR = new OldXHR();
|
|
859
|
+
realXHR.addEventListener('loadstart', function () {
|
|
860
|
+
ajaxEventTrigger.call(this, 'ajaxLoadStart');
|
|
861
|
+
}, false);
|
|
862
|
+
realXHR.addEventListener('loadend', function () {
|
|
863
|
+
ajaxEventTrigger.call(this, 'ajaxLoadEnd');
|
|
864
|
+
}, false);
|
|
865
|
+
// 此处的捕获的异常会连日志接口也一起捕获,如果日志上报接口异常了,就会导致死循环了。
|
|
866
|
+
realXHR.onerror = function (e) {
|
|
867
|
+
// eslint-disable-next-line no-console
|
|
868
|
+
console.warn('realXHR.onerror, e', e);
|
|
869
|
+
};
|
|
870
|
+
return realXHR;
|
|
871
|
+
}
|
|
872
|
+
/**
|
|
873
|
+
* 页面接口请求监控
|
|
874
|
+
*/
|
|
875
|
+
var tempUrlInfo = {};
|
|
876
|
+
var recordXMLHttpRequestLog = function (XMLHttpRequestTimeout) {
|
|
877
|
+
XMLHttpRequestTimeout = typeof XMLHttpRequestTimeout === 'number' ? XMLHttpRequestTimeout : 1000;
|
|
878
|
+
var timeRecordArray = [];
|
|
879
|
+
window.__XMLHttpRequest__ = window.XMLHttpRequest;
|
|
880
|
+
window.XMLHttpRequest = newXHR;
|
|
881
|
+
window.addEventListener('ajaxLoadStart', function (e) {
|
|
882
|
+
var tempObj = {
|
|
883
|
+
timeStamp: new Date().getTime(),
|
|
884
|
+
event: e
|
|
885
|
+
};
|
|
886
|
+
timeRecordArray.push(tempObj);
|
|
887
|
+
});
|
|
888
|
+
window.addEventListener('ajaxLoadEnd', function () {
|
|
889
|
+
var timeRecordArrayCopy = [].concat(timeRecordArray);
|
|
890
|
+
var _loop_1 = function (i) {
|
|
891
|
+
if (timeRecordArrayCopy[i].event.detail && timeRecordArrayCopy[i].event.detail.status > 0) {
|
|
892
|
+
var currentTime = new Date().getTime();
|
|
893
|
+
var _a = timeRecordArrayCopy[i].event.detail, responseURL_1 = _a.responseURL, status_1 = _a.status, statusText = _a.statusText, timeStamp = _a.timeStamp;
|
|
894
|
+
var previousTime = timeStamp || timeRecordArrayCopy[i].timeStamp;
|
|
895
|
+
var loadTime = currentTime - previousTime;
|
|
896
|
+
var request = {
|
|
897
|
+
requestType: 'xhr',
|
|
898
|
+
responseURL: responseURL_1,
|
|
899
|
+
status: status_1,
|
|
900
|
+
loadTime: loadTime,
|
|
901
|
+
statusText: statusText,
|
|
902
|
+
reason: '',
|
|
903
|
+
detail: ''
|
|
904
|
+
};
|
|
905
|
+
if (loadTime && loadTime > XMLHttpRequestTimeout) {
|
|
906
|
+
request.reason = 'slow';
|
|
907
|
+
request.detail = "request is too slow, XMLHttpRequestTimeout: ".concat(XMLHttpRequestTimeout);
|
|
908
|
+
}
|
|
909
|
+
else if (status_1 && status_1 >= 300 && statusText && statusText.toLowerCase() !== 'ok') {
|
|
910
|
+
request.reason = 'failed';
|
|
911
|
+
request.detail = "request is failed, status: ".concat(status_1, ", statusText: ").concat(statusText);
|
|
912
|
+
}
|
|
913
|
+
else ;
|
|
914
|
+
if (request.reason) {
|
|
915
|
+
if (!tempUrlInfo[responseURL_1]) {
|
|
916
|
+
tempUrlInfo[responseURL_1] = true;
|
|
917
|
+
setTimeout(function () {
|
|
918
|
+
delete tempUrlInfo[responseURL_1];
|
|
919
|
+
}, 10);
|
|
920
|
+
setReportValue('error', null);
|
|
921
|
+
report(__assign(__assign({}, getReport()), {
|
|
922
|
+
page: location.href,
|
|
923
|
+
originPage: location.href,
|
|
924
|
+
type: 'request',
|
|
925
|
+
req: request
|
|
926
|
+
}));
|
|
927
|
+
}
|
|
928
|
+
}
|
|
929
|
+
}
|
|
930
|
+
// 当前请求成功后就在数组中移除掉
|
|
931
|
+
timeRecordArray.splice(i, 1);
|
|
932
|
+
};
|
|
933
|
+
for (var i = 0; i < timeRecordArrayCopy.length; i++) {
|
|
934
|
+
_loop_1(i);
|
|
935
|
+
}
|
|
936
|
+
});
|
|
937
|
+
};
|
|
938
|
+
var hackFetch = function (XMLHttpRequestTimeout) {
|
|
939
|
+
XMLHttpRequestTimeout = typeof XMLHttpRequestTimeout === 'number' ? XMLHttpRequestTimeout : 1000;
|
|
940
|
+
if (typeof window.fetch === 'function') {
|
|
941
|
+
var __fetch__1 = window.fetch;
|
|
942
|
+
window.__fetch__ = __fetch__1;
|
|
943
|
+
window.fetch = function (t) {
|
|
944
|
+
var args = [];
|
|
945
|
+
for (var _i = 1; _i < arguments.length; _i++) {
|
|
946
|
+
args[_i - 1] = arguments[_i];
|
|
947
|
+
}
|
|
948
|
+
var begin = Date.now();
|
|
949
|
+
//禁用数组的扩展运算符,否则portal的生产环境会报Uncaught TypeError: Object(...) is not a function,
|
|
950
|
+
//编译后的__spreadArray函数有问题,而这个函数来自于tslib.具体报错原因不知.
|
|
951
|
+
//其他平台使用数组的扩展运算符没有问题,比如前端监控平台的管理界面
|
|
952
|
+
var params = [].concat(t).concat(args);
|
|
953
|
+
return __fetch__1
|
|
954
|
+
.apply(window, params)
|
|
955
|
+
.then(function (res) {
|
|
956
|
+
var response = res.clone();
|
|
957
|
+
var headers = response.headers;
|
|
958
|
+
if (headers && typeof headers.get === 'function') {
|
|
959
|
+
var ct = headers.get('content-type');
|
|
960
|
+
if (ct && !/(text)|(json)/.test(ct)) {
|
|
961
|
+
return res;
|
|
962
|
+
}
|
|
963
|
+
}
|
|
964
|
+
var loadTime = Date.now() - begin;
|
|
965
|
+
response
|
|
966
|
+
.text()
|
|
967
|
+
.then(function (result) {
|
|
968
|
+
var url = response.url, status = response.status, statusText = response.statusText, ok = response.ok;
|
|
969
|
+
var request = {
|
|
970
|
+
requestType: 'fetch',
|
|
971
|
+
responseURL: url,
|
|
972
|
+
status: status,
|
|
973
|
+
loadTime: loadTime,
|
|
974
|
+
statusText: statusText,
|
|
975
|
+
reason: '',
|
|
976
|
+
detail: ''
|
|
977
|
+
};
|
|
978
|
+
if (!ok || status >= 300) {
|
|
979
|
+
request.reason = 'failed';
|
|
980
|
+
request.detail = "request is failed, status: ".concat(status, ", statusText: ").concat(statusText);
|
|
981
|
+
}
|
|
982
|
+
else if (loadTime > XMLHttpRequestTimeout) {
|
|
983
|
+
request.reason = 'slow';
|
|
984
|
+
request.detail = "request is too slow, XMLHttpRequestTimeout: ".concat(XMLHttpRequestTimeout, ", result: ").concat(result === null || result === void 0 ? void 0 : result.slice(0, 500));
|
|
985
|
+
}
|
|
986
|
+
if (request.reason) {
|
|
987
|
+
if (!tempUrlInfo[url]) {
|
|
988
|
+
tempUrlInfo[url] = true;
|
|
989
|
+
setTimeout(function () {
|
|
990
|
+
delete tempUrlInfo[url];
|
|
991
|
+
}, 10);
|
|
992
|
+
setReportValue('error', null);
|
|
993
|
+
report(__assign(__assign({}, getReport()), {
|
|
994
|
+
page: location.href,
|
|
995
|
+
originPage: location.href,
|
|
996
|
+
type: 'request',
|
|
997
|
+
req: request
|
|
998
|
+
}));
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
1001
|
+
})["catch"](function (err) {
|
|
1002
|
+
// eslint-disable-next-line no-console
|
|
1003
|
+
console.log('hackFetch response.text() err', err);
|
|
1004
|
+
});
|
|
1005
|
+
return res;
|
|
1006
|
+
})["catch"](function (err) {
|
|
1007
|
+
// eslint-disable-next-line no-console
|
|
1008
|
+
console.log('hackFetch err', err);
|
|
1009
|
+
});
|
|
1010
|
+
};
|
|
1011
|
+
}
|
|
1012
|
+
};
|
|
1013
|
+
|
|
1014
|
+
/*
|
|
1015
|
+
* @Author: Mark.Zhang
|
|
1016
|
+
* @Date: 2020-10-26 11:10:04
|
|
1017
|
+
* @Description 用户行为处理方法
|
|
1018
|
+
*/
|
|
1019
|
+
/**
|
|
1020
|
+
* @description 点击事件触发后的操作
|
|
1021
|
+
*/
|
|
1022
|
+
function handleClick(event) {
|
|
1023
|
+
var target = event.target;
|
|
1024
|
+
if (target.nodeName === 'INPUT' || target.nodeName === 'TEXTAREA') {
|
|
1025
|
+
return;
|
|
1026
|
+
}
|
|
1027
|
+
var path = getElmPath(target);
|
|
1028
|
+
if (path) {
|
|
1029
|
+
setReportValue('type', 'ua');
|
|
1030
|
+
setReportValue('ua', {
|
|
1031
|
+
subType: 'ui.click',
|
|
1032
|
+
x: event.x,
|
|
1033
|
+
y: event.y,
|
|
1034
|
+
path: path
|
|
1035
|
+
});
|
|
1036
|
+
report(getReport());
|
|
1037
|
+
}
|
|
1038
|
+
}
|
|
1039
|
+
/**
|
|
1040
|
+
* @description 点击触发失焦后的操作
|
|
1041
|
+
*/
|
|
1042
|
+
function handleBlur(event) {
|
|
1043
|
+
var target = event.target;
|
|
1044
|
+
if (target.nodeName !== 'INPUT' && target.nodeName !== 'TEXTAREA') {
|
|
1045
|
+
return;
|
|
1046
|
+
}
|
|
1047
|
+
var path = getElmPath(target);
|
|
1048
|
+
if (path) {
|
|
1049
|
+
setReportValue('type', 'ua');
|
|
1050
|
+
setReportValue('ua', {
|
|
1051
|
+
subType: 'ui.blur',
|
|
1052
|
+
x: event.x,
|
|
1053
|
+
y: event.y,
|
|
1054
|
+
path: path,
|
|
1055
|
+
value: target.value
|
|
1056
|
+
});
|
|
1057
|
+
report(getReport());
|
|
1058
|
+
}
|
|
1059
|
+
}
|
|
1060
|
+
/**
|
|
1061
|
+
* @description 滚动的操作
|
|
1062
|
+
*/
|
|
1063
|
+
var timeout$1;
|
|
1064
|
+
function handleScroll(_event) {
|
|
1065
|
+
clearTimeout(timeout$1);
|
|
1066
|
+
timeout$1 = setTimeout(function () {
|
|
1067
|
+
setReportValue('type', 'ua');
|
|
1068
|
+
setReportValue('ua', {
|
|
1069
|
+
subType: 'ui.scroll'
|
|
1070
|
+
});
|
|
1071
|
+
report(getReport());
|
|
1072
|
+
}, 1000);
|
|
1073
|
+
}
|
|
1074
|
+
|
|
1075
|
+
var parser = new uaParserJs.UAParser();
|
|
1076
|
+
/**
|
|
1077
|
+
* @description 获取uuid
|
|
1078
|
+
*/
|
|
1079
|
+
function getUid() {
|
|
1080
|
+
var uid = localStorage.getItem(shuyunTrackId) || '';
|
|
1081
|
+
if (!uid) {
|
|
1082
|
+
uid = uuid.v4();
|
|
1083
|
+
localStorage.setItem(shuyunTrackId, uid);
|
|
1084
|
+
}
|
|
1085
|
+
return uid;
|
|
1086
|
+
}
|
|
1087
|
+
/**
|
|
1088
|
+
* @description 获取session id
|
|
1089
|
+
*/
|
|
1090
|
+
function getSessionId() {
|
|
1091
|
+
var sessionId = sessionStorage.getItem(shuyunTrackSessionId);
|
|
1092
|
+
if (sessionId) {
|
|
1093
|
+
return sessionId;
|
|
1094
|
+
}
|
|
1095
|
+
else {
|
|
1096
|
+
var id = uuid.v4();
|
|
1097
|
+
sessionStorage.setItem(shuyunTrackSessionId, id);
|
|
1098
|
+
return id;
|
|
1099
|
+
}
|
|
1100
|
+
}
|
|
1101
|
+
/**
|
|
1102
|
+
* @description 获取浏览器信息
|
|
1103
|
+
*/
|
|
1104
|
+
function getNavigator() {
|
|
1105
|
+
var _a;
|
|
1106
|
+
var uaResult = parser.getResult();
|
|
1107
|
+
return {
|
|
1108
|
+
language: navigator.language,
|
|
1109
|
+
navigatorVendor: navigator.vendor,
|
|
1110
|
+
connectionType: ((_a = navigator === null || navigator === void 0 ? void 0 : navigator.connection) === null || _a === void 0 ? void 0 : _a.effectiveType) || '2g',
|
|
1111
|
+
browserName: uaResult.browser.name || '',
|
|
1112
|
+
browserVersion: uaResult.browser.version || '',
|
|
1113
|
+
engineName: uaResult.engine.name || '',
|
|
1114
|
+
engineVersion: uaResult.engine.version || '',
|
|
1115
|
+
osName: uaResult.os.name || '',
|
|
1116
|
+
osVersion: uaResult.os.version || ''
|
|
1117
|
+
};
|
|
1118
|
+
}
|
|
1119
|
+
/**
|
|
1120
|
+
* @description 获取viewport的宽高
|
|
1121
|
+
*/
|
|
1122
|
+
function getViewport() {
|
|
1123
|
+
var w = document.documentElement.clientWidth || document.body.clientWidth;
|
|
1124
|
+
var h = document.documentElement.clientHeight || document.body.clientHeight;
|
|
1125
|
+
return "".concat(w, " x ").concat(h);
|
|
1126
|
+
}
|
|
1127
|
+
/**
|
|
1128
|
+
* @description 获取元素路径,最多保留5层
|
|
1129
|
+
* @param e
|
|
1130
|
+
*/
|
|
1131
|
+
var getElmPath = function (target) {
|
|
1132
|
+
if (!target || target.nodeType !== 1) {
|
|
1133
|
+
return '';
|
|
1134
|
+
}
|
|
1135
|
+
var ret = [];
|
|
1136
|
+
// 层数,最多5层
|
|
1137
|
+
var deepLength = 0;
|
|
1138
|
+
// 元素
|
|
1139
|
+
var elm = '';
|
|
1140
|
+
if (typeof target.innerText === 'string') {
|
|
1141
|
+
ret.push("(".concat(target.innerText.substr(0, 50), ")"));
|
|
1142
|
+
}
|
|
1143
|
+
for (var t = target || null; t && deepLength++ < 5 && !((elm = normalTarget(t)) === 'html');) {
|
|
1144
|
+
// eslint-disable-next-line no-sequences
|
|
1145
|
+
ret.push(elm), (t = t.parentNode);
|
|
1146
|
+
}
|
|
1147
|
+
return ret.reverse().join(' > ');
|
|
1148
|
+
};
|
|
1149
|
+
/**
|
|
1150
|
+
* @description 处理html node
|
|
1151
|
+
* @param e
|
|
1152
|
+
*/
|
|
1153
|
+
var normalTarget = function (target) {
|
|
1154
|
+
var t, n, r, a, i;
|
|
1155
|
+
var o = [];
|
|
1156
|
+
if (!target || !target.tagName) {
|
|
1157
|
+
return '';
|
|
1158
|
+
}
|
|
1159
|
+
o.push(target.tagName.toLowerCase());
|
|
1160
|
+
if (target.id) {
|
|
1161
|
+
o.push('#'.concat(target.id));
|
|
1162
|
+
}
|
|
1163
|
+
if ((t = target.className) && Object.prototype.toString.call(t) === '[object String]') {
|
|
1164
|
+
for (n = t.split(/\s+/), i = 0; i < n.length; i++) {
|
|
1165
|
+
// className包含active的不加入路径
|
|
1166
|
+
if (n[i].indexOf('active') < 0)
|
|
1167
|
+
o.push('.'.concat(n[i]));
|
|
1168
|
+
}
|
|
1169
|
+
}
|
|
1170
|
+
var s = ['type', 'name', 'title', 'alt'];
|
|
1171
|
+
for (i = 0; i < s.length; i++) {
|
|
1172
|
+
r = s[i];
|
|
1173
|
+
if ((a = target.getAttribute(r))) {
|
|
1174
|
+
o.push('['.concat(r, '="').concat(a, '"]'));
|
|
1175
|
+
}
|
|
1176
|
+
}
|
|
1177
|
+
return o.join('');
|
|
1178
|
+
};
|
|
1179
|
+
/**
|
|
1180
|
+
* @description 序列化对象
|
|
1181
|
+
* @param obj
|
|
1182
|
+
*/
|
|
1183
|
+
function serialize(obj) {
|
|
1184
|
+
var str = [];
|
|
1185
|
+
for (var p in obj) {
|
|
1186
|
+
if (Object.prototype.hasOwnProperty.call(obj, p) && typeof obj[p] !== 'undefined') {
|
|
1187
|
+
var value = typeof obj[p] === 'object' ? JSON.stringify(obj[p]) : obj[p];
|
|
1188
|
+
str.push(encodeURIComponent(p) + '=' + encodeURIComponent(value));
|
|
1189
|
+
}
|
|
1190
|
+
}
|
|
1191
|
+
return str.join('&');
|
|
1192
|
+
}
|
|
1193
|
+
/**
|
|
1194
|
+
* @description 注册事件监听
|
|
1195
|
+
* @param event 事件
|
|
1196
|
+
* @param fn 回调方法
|
|
1197
|
+
*/
|
|
1198
|
+
var on = function (event, fn) {
|
|
1199
|
+
if (window.addEventListener) {
|
|
1200
|
+
window.addEventListener(event, fn, true);
|
|
1201
|
+
}
|
|
1202
|
+
else {
|
|
1203
|
+
window.attachEvent("on".concat(event), fn);
|
|
1204
|
+
}
|
|
1205
|
+
};
|
|
1206
|
+
/**
|
|
1207
|
+
* @description 移除事件监听
|
|
1208
|
+
* @param event 事件
|
|
1209
|
+
* @param fn 回调方法
|
|
1210
|
+
*/
|
|
1211
|
+
var off = function (event, fn) {
|
|
1212
|
+
if (window.removeEventListener) {
|
|
1213
|
+
return window.removeEventListener(event, fn);
|
|
1214
|
+
}
|
|
1215
|
+
else {
|
|
1216
|
+
return window.detachEvent("on".concat(event), fn);
|
|
1217
|
+
}
|
|
1218
|
+
};
|
|
1219
|
+
var formatTrackElementXPath = function (xPath) {
|
|
1220
|
+
var result = [];
|
|
1221
|
+
var array = xPath.split('/');
|
|
1222
|
+
array.forEach(function (item) {
|
|
1223
|
+
if (!['svg', 'path', 'g', 'image', 'text', 'line', 'rect', 'polygon', 'circle', 'ellipse'].some(function (child) {
|
|
1224
|
+
if (item.toLowerCase().split('[')[0] === child) {
|
|
1225
|
+
result.push("/*[name()=\"".concat(child, "\"]"));
|
|
1226
|
+
return true;
|
|
1227
|
+
}
|
|
1228
|
+
else {
|
|
1229
|
+
return false;
|
|
1230
|
+
}
|
|
1231
|
+
})) {
|
|
1232
|
+
result.push(item);
|
|
1233
|
+
}
|
|
1234
|
+
});
|
|
1235
|
+
return result.join('/');
|
|
1236
|
+
};
|
|
1237
|
+
var formatElementXPath = function (info) {
|
|
1238
|
+
var _a;
|
|
1239
|
+
var elementXPathValue = '';
|
|
1240
|
+
var elementXPath = info.elementXPath;
|
|
1241
|
+
if (elementXPath) {
|
|
1242
|
+
var xpathElement = document.evaluate(formatTrackElementXPath(elementXPath), document).iterateNext();
|
|
1243
|
+
elementXPathValue = xpathElement ? (_a = (xpathElement.textContent || xpathElement.value)) === null || _a === void 0 ? void 0 : _a.trim() : '';
|
|
1244
|
+
}
|
|
1245
|
+
return {
|
|
1246
|
+
elementXPath: elementXPath,
|
|
1247
|
+
elementXPathValue: elementXPathValue
|
|
1248
|
+
};
|
|
1249
|
+
};
|
|
1250
|
+
var existTrackListenEvent = {};
|
|
1251
|
+
var elements = [];
|
|
1252
|
+
var timeout;
|
|
1253
|
+
var visualTrackFunc = function () {
|
|
1254
|
+
if (Config.enableVisualTrack) {
|
|
1255
|
+
if (timeout)
|
|
1256
|
+
clearTimeout(timeout);
|
|
1257
|
+
timeout = setTimeout(function () {
|
|
1258
|
+
timeout = null;
|
|
1259
|
+
var url = location.href.split('?')[0];
|
|
1260
|
+
axios__default["default"]
|
|
1261
|
+
.get("".concat(Config.reportUrl.replace('/s/r', ''), "/visual/get_data_list?url=").concat(encodeURIComponent(url)))
|
|
1262
|
+
.then(function (res) {
|
|
1263
|
+
var _a, _b;
|
|
1264
|
+
var listData = (_b = (_a = res.data.result) === null || _a === void 0 ? void 0 : _a.result) === null || _b === void 0 ? void 0 : _b.dataList;
|
|
1265
|
+
if (Array.isArray(listData)) {
|
|
1266
|
+
try {
|
|
1267
|
+
listData.forEach(function (item) {
|
|
1268
|
+
if (item.trackType === 'mount') {
|
|
1269
|
+
visualReportEvent(item);
|
|
1270
|
+
}
|
|
1271
|
+
else {
|
|
1272
|
+
var element = document
|
|
1273
|
+
.evaluate(formatTrackElementXPath(item.trackElementXPath), document)
|
|
1274
|
+
.iterateNext();
|
|
1275
|
+
var key = url + item.trackElementXPath;
|
|
1276
|
+
if (element) {
|
|
1277
|
+
if (!existTrackListenEvent[key]) {
|
|
1278
|
+
existTrackListenEvent[key] = true;
|
|
1279
|
+
elements.push(element);
|
|
1280
|
+
//@ts-ignore
|
|
1281
|
+
element.setAttribute('visual-track-data-attr', JSON.stringify(item));
|
|
1282
|
+
element.addEventListener('click', clickEventFunc);
|
|
1283
|
+
}
|
|
1284
|
+
}
|
|
1285
|
+
else {
|
|
1286
|
+
var func_1 = function () {
|
|
1287
|
+
var _a;
|
|
1288
|
+
var _b = getNavigator(), browserName = _b.browserName, browserVersion = _b.browserVersion;
|
|
1289
|
+
var payload = {
|
|
1290
|
+
projectID: Config.projectID,
|
|
1291
|
+
version: version,
|
|
1292
|
+
url: url,
|
|
1293
|
+
trackElementXPath: item.trackElementXPath,
|
|
1294
|
+
trackNameInfo: item.trackNameInfo,
|
|
1295
|
+
params: item.params,
|
|
1296
|
+
customPayload: Config.customPayload,
|
|
1297
|
+
type: 'visual_track_xpath_not_found',
|
|
1298
|
+
browser: "".concat(browserName, " - ").concat(browserVersion)
|
|
1299
|
+
};
|
|
1300
|
+
if (typeof ((_a = window.navigator) === null || _a === void 0 ? void 0 : _a.sendBeacon) === 'function') {
|
|
1301
|
+
var formData = new FormData();
|
|
1302
|
+
for (var i in payload) {
|
|
1303
|
+
if (Object.prototype.hasOwnProperty.call(payload, i)) {
|
|
1304
|
+
var value = payload[i];
|
|
1305
|
+
formData.append(i, value && typeof value === 'object' ? JSON.stringify(value) : value);
|
|
1306
|
+
}
|
|
1307
|
+
}
|
|
1308
|
+
window.navigator.sendBeacon(Config.reportUrl, formData);
|
|
1309
|
+
}
|
|
1310
|
+
else {
|
|
1311
|
+
new Image().src = "".concat(Config.reportUrl, "?").concat(serialize(payload));
|
|
1312
|
+
}
|
|
1313
|
+
};
|
|
1314
|
+
if (typeof window.requestIdleCallback === 'function') {
|
|
1315
|
+
window.requestIdleCallback(function () { return func_1(); }, { timeout: 2000 });
|
|
1316
|
+
}
|
|
1317
|
+
else {
|
|
1318
|
+
setTimeout(function () { return func_1(); }, 0);
|
|
1319
|
+
}
|
|
1320
|
+
}
|
|
1321
|
+
}
|
|
1322
|
+
});
|
|
1323
|
+
}
|
|
1324
|
+
catch (err) {
|
|
1325
|
+
// eslint-disable-next-line no-console
|
|
1326
|
+
console.log('visual track evaluate err', err);
|
|
1327
|
+
}
|
|
1328
|
+
}
|
|
1329
|
+
});
|
|
1330
|
+
}, 1000);
|
|
1331
|
+
}
|
|
1332
|
+
};
|
|
1333
|
+
var visualReportEvent = function (item) {
|
|
1334
|
+
var _a;
|
|
1335
|
+
var _b = formatElementXPath(item.trackNameInfo), elementXPath = _b.elementXPath, elementXPathValue = _b.elementXPathValue;
|
|
1336
|
+
var visualTrackData = {
|
|
1337
|
+
trackType: item.trackType,
|
|
1338
|
+
trackName: elementXPath ? elementXPathValue : item.trackNameInfo.name,
|
|
1339
|
+
params: (_a = item.params) === null || _a === void 0 ? void 0 : _a.map(function (child) {
|
|
1340
|
+
var _a = formatElementXPath(child), elementXPath = _a.elementXPath, elementXPathValue = _a.elementXPathValue;
|
|
1341
|
+
return {
|
|
1342
|
+
name: child.name,
|
|
1343
|
+
value: elementXPath ? elementXPathValue : child.value
|
|
1344
|
+
};
|
|
1345
|
+
})
|
|
1346
|
+
};
|
|
1347
|
+
setReportValue('type', 'visual');
|
|
1348
|
+
setReportValue('vD', visualTrackData);
|
|
1349
|
+
report(getReport());
|
|
1350
|
+
setReportValue('vD', undefined);
|
|
1351
|
+
};
|
|
1352
|
+
var clickEventFunc = function (e) {
|
|
1353
|
+
//@ts-ignore
|
|
1354
|
+
var item = JSON.parse(e.target.getAttribute('visual-track-data-attr'));
|
|
1355
|
+
visualReportEvent(item);
|
|
1356
|
+
};
|
|
1357
|
+
var initWindowObjectFunction = function () {
|
|
1358
|
+
if (typeof window.manualReportTrackFunc === 'undefined') {
|
|
1359
|
+
window.manualReportTrackFunc = function (data) {
|
|
1360
|
+
if (Object.prototype.toString.call(data) !== '[object Object]') {
|
|
1361
|
+
// eslint-disable-next-line no-console
|
|
1362
|
+
console.warn('manualReportTrackFunc参数必须是一个对象!');
|
|
1363
|
+
}
|
|
1364
|
+
else {
|
|
1365
|
+
setReportValue('type', 'manual');
|
|
1366
|
+
setReportValue('manualReport', data);
|
|
1367
|
+
report(getReport());
|
|
1368
|
+
setReportValue('manualReport', undefined);
|
|
1369
|
+
}
|
|
1370
|
+
};
|
|
1371
|
+
}
|
|
1372
|
+
if (typeof window.getFullScreenShootFunc === 'undefined') {
|
|
1373
|
+
//下载页面截图
|
|
1374
|
+
window.getFullScreenShootFunc = function (filename) { return __awaiter(void 0, void 0, void 0, function () {
|
|
1375
|
+
var file, url, tagA;
|
|
1376
|
+
return __generator(this, function (_a) {
|
|
1377
|
+
switch (_a.label) {
|
|
1378
|
+
case 0: return [4 /*yield*/, getFullScreenShoot(filename)];
|
|
1379
|
+
case 1:
|
|
1380
|
+
file = _a.sent();
|
|
1381
|
+
url = window.URL.createObjectURL(file);
|
|
1382
|
+
tagA = document.createElement('a');
|
|
1383
|
+
tagA.setAttribute('href', url);
|
|
1384
|
+
tagA.setAttribute('download', file.name);
|
|
1385
|
+
tagA.setAttribute('target', '_blank');
|
|
1386
|
+
document.body.appendChild(tagA);
|
|
1387
|
+
tagA.click();
|
|
1388
|
+
document.body.removeChild(tagA);
|
|
1389
|
+
return [2 /*return*/];
|
|
1390
|
+
}
|
|
1391
|
+
});
|
|
1392
|
+
}); };
|
|
1393
|
+
}
|
|
1394
|
+
if (typeof window.getRRWebUserEventsCaptureFunc === 'undefined') {
|
|
1395
|
+
window.getRRWebUserEventsCaptureFunc = function () {
|
|
1396
|
+
setReportValue('type', 'error');
|
|
1397
|
+
setReportValue('error', {
|
|
1398
|
+
subType: 'manual',
|
|
1399
|
+
message: 'manual trigger',
|
|
1400
|
+
events: getUserEvents(true)
|
|
1401
|
+
});
|
|
1402
|
+
report(getReport());
|
|
1403
|
+
};
|
|
1404
|
+
}
|
|
1405
|
+
};
|
|
1406
|
+
var prewHref = '';
|
|
1407
|
+
var handleLocationChange = function (e) {
|
|
1408
|
+
var curHref = location.href;
|
|
1409
|
+
// href中`?`后面的值改变,不触发后续动作
|
|
1410
|
+
if (prewHref.split('?')[0] === curHref.split('?')[0]) {
|
|
1411
|
+
return;
|
|
1412
|
+
}
|
|
1413
|
+
prewHref = curHref;
|
|
1414
|
+
handleHistoryChange();
|
|
1415
|
+
elements.forEach(function (element) {
|
|
1416
|
+
element === null || element === void 0 ? void 0 : element.removeEventListener('click', clickEventFunc);
|
|
1417
|
+
});
|
|
1418
|
+
//清空可视化埋点之前的数据
|
|
1419
|
+
existTrackListenEvent = {};
|
|
1420
|
+
elements.length = 0;
|
|
1421
|
+
visualTrackFunc();
|
|
1422
|
+
};
|
|
1423
|
+
|
|
1424
|
+
var Track = /** @class */ (function () {
|
|
1425
|
+
function Track() {
|
|
1426
|
+
this.visualTrack = function () {
|
|
1427
|
+
window.onload = function () {
|
|
1428
|
+
visualTrackFunc();
|
|
1429
|
+
};
|
|
1430
|
+
};
|
|
1431
|
+
}
|
|
1432
|
+
Track.prototype.init = function (config) {
|
|
1433
|
+
var _a;
|
|
1434
|
+
// 是否开启日志收集
|
|
1435
|
+
if (!config || !config.enable) {
|
|
1436
|
+
return;
|
|
1437
|
+
}
|
|
1438
|
+
// 没有项目ID,则不监听任何事件
|
|
1439
|
+
if (!config.projectID) {
|
|
1440
|
+
// eslint-disable-next-line no-console
|
|
1441
|
+
console.warn('缺少项目ID或token!');
|
|
1442
|
+
return;
|
|
1443
|
+
}
|
|
1444
|
+
// 没有reportUrl
|
|
1445
|
+
if (!config.reportUrl) {
|
|
1446
|
+
// eslint-disable-next-line no-console
|
|
1447
|
+
console.warn('缺少上报地址!');
|
|
1448
|
+
return;
|
|
1449
|
+
}
|
|
1450
|
+
if (this.ignoreUrl(((_a = config === null || config === void 0 ? void 0 : config.ignore) === null || _a === void 0 ? void 0 : _a.urls) || [])) {
|
|
1451
|
+
return;
|
|
1452
|
+
}
|
|
1453
|
+
setConfig(config);
|
|
1454
|
+
initReport();
|
|
1455
|
+
recordXMLHttpRequestLog(config.XMLHttpRequestTimeout);
|
|
1456
|
+
hackFetch(config.XMLHttpRequestTimeout);
|
|
1457
|
+
Config.spa && this.addListenRouterChange();
|
|
1458
|
+
Config.enableBehavior && this.addListenUserActivity();
|
|
1459
|
+
Config.enableError && this.addListenJSUncaught();
|
|
1460
|
+
this.visualTrack();
|
|
1461
|
+
this.addListenUnload();
|
|
1462
|
+
initWindowObjectFunction();
|
|
1463
|
+
};
|
|
1464
|
+
/**
|
|
1465
|
+
* @description 监听错误异常
|
|
1466
|
+
*/
|
|
1467
|
+
Track.prototype.addListenJSUncaught = function () {
|
|
1468
|
+
on('error', handleError);
|
|
1469
|
+
on('unhandledrejection', handleError);
|
|
1470
|
+
};
|
|
1471
|
+
/**
|
|
1472
|
+
* @description 监听行为
|
|
1473
|
+
*/
|
|
1474
|
+
Track.prototype.addListenUserActivity = function () {
|
|
1475
|
+
on('click', handleClick); // 非输入框点击
|
|
1476
|
+
on('blur', handleBlur); // 输入框失焦
|
|
1477
|
+
on('scroll', handleScroll); // 输入框失焦
|
|
1478
|
+
};
|
|
1479
|
+
/**
|
|
1480
|
+
* @description 监听路由变化
|
|
1481
|
+
*/
|
|
1482
|
+
Track.prototype.addListenRouterChange = function () {
|
|
1483
|
+
on('popstate', handleLocationChange);
|
|
1484
|
+
// on('hashchange', pv.handleHashChange); //触发hashchange的同时也会触发popstate
|
|
1485
|
+
if (Config.spa && !Config.hash) {
|
|
1486
|
+
history.pushState = _history('pushState');
|
|
1487
|
+
history.replaceState = _history('replaceState');
|
|
1488
|
+
on('pushState', handleLocationChange);
|
|
1489
|
+
on('replaceState', handleLocationChange);
|
|
1490
|
+
}
|
|
1491
|
+
};
|
|
1492
|
+
/**
|
|
1493
|
+
* @description 页面将要关闭
|
|
1494
|
+
*/
|
|
1495
|
+
Track.prototype.addListenUnload = function () {
|
|
1496
|
+
var _this = this;
|
|
1497
|
+
on('beforeunload', function () {
|
|
1498
|
+
_this.destroy();
|
|
1499
|
+
});
|
|
1500
|
+
};
|
|
1501
|
+
/**
|
|
1502
|
+
* 忽略的url
|
|
1503
|
+
* @param urls
|
|
1504
|
+
*/
|
|
1505
|
+
Track.prototype.ignoreUrl = function (urls) {
|
|
1506
|
+
var someUrl = urls.some(function (url) { return location.href.includes(url); });
|
|
1507
|
+
return someUrl;
|
|
1508
|
+
};
|
|
1509
|
+
/**
|
|
1510
|
+
* @description 销毁监听器
|
|
1511
|
+
*/
|
|
1512
|
+
Track.prototype.destroy = function () {
|
|
1513
|
+
if (Config.spa) {
|
|
1514
|
+
// off('hashchange', pv.handleHashChange);
|
|
1515
|
+
off('pushState', handleHistoryChange);
|
|
1516
|
+
off('popstate', handleHistoryChange);
|
|
1517
|
+
off('replaceState', handleHistoryChange);
|
|
1518
|
+
}
|
|
1519
|
+
if (Config.enableBehavior) {
|
|
1520
|
+
off('click', handleClick);
|
|
1521
|
+
off('blur', handleBlur);
|
|
1522
|
+
off('scroll', handleScroll);
|
|
1523
|
+
}
|
|
1524
|
+
if (Config.enableError) {
|
|
1525
|
+
off('error', handleError);
|
|
1526
|
+
}
|
|
1527
|
+
sessionStorage.removeItem(shuyunTrackSessionId);
|
|
1528
|
+
};
|
|
1529
|
+
return Track;
|
|
1530
|
+
}());
|
|
1531
|
+
|
|
1532
|
+
return Track;
|
|
1533
|
+
|
|
1534
|
+
})(axios, uuid, uaParserJs, rrweb, ErrorStackParser, html2canvas);
|