chengyi-frondend-monitor-sdk 1.0.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/README.md +110 -0
- package/dist/errorHandler.d.ts +9 -0
- package/dist/index.cjs.js +471 -0
- package/dist/index.cjs.js.map +1 -0
- package/dist/index.d.ts +54 -0
- package/dist/index.esm.js +467 -0
- package/dist/index.esm.js.map +1 -0
- package/dist/index.umd.js +2 -0
- package/dist/index.umd.js.map +1 -0
- package/dist/networkMonitor.d.ts +9 -0
- package/dist/resourceMonitor.d.ts +9 -0
- package/dist/sender.d.ts +11 -0
- package/dist/utils.d.ts +18 -0
- package/package.json +32 -0
package/README.md
ADDED
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
# Monitor SDK
|
|
2
|
+
|
|
3
|
+
一个轻量级的前端错误监控 SDK,可以捕获 JavaScript 错误、Promise 异常、网络请求错误以及资源加载错误。
|
|
4
|
+
|
|
5
|
+
## 功能特性
|
|
6
|
+
|
|
7
|
+
- **JavaScript 错误监控**: 捕获未捕获的 JavaScript 错误 (`window.onerror`)。
|
|
8
|
+
- **Promise 异常监控**: 捕获未处理的 Promise 拒绝 (`window.onunhandledrejection`)。
|
|
9
|
+
- **网络错误监控**:
|
|
10
|
+
- 拦截并监控 `XMLHttpRequest` 错误。
|
|
11
|
+
- 拦截并监控 `fetch` API 错误。
|
|
12
|
+
- **资源加载错误监控**: 捕获资源(如图片 `<img>` 和脚本 `<script>`)加载失败的错误。
|
|
13
|
+
- **数据上报**: 优先使用 `navigator.sendBeacon` 进行可靠的数据上报,是一种在页面卸载或关闭时可靠发送数据的方法。
|
|
14
|
+
|
|
15
|
+
## 安装
|
|
16
|
+
|
|
17
|
+
```bash
|
|
18
|
+
pnpm i chengyi-frondend-monitor-sdk
|
|
19
|
+
```
|
|
20
|
+
|
|
21
|
+
## 使用方法
|
|
22
|
+
|
|
23
|
+
在你的应用入口文件(例如 `main.js` 或 `App.tsx`)中初始化 SDK。
|
|
24
|
+
|
|
25
|
+
```javascript
|
|
26
|
+
import { initErrorMonitor } from 'chengyi-frondend-monitor-sdk';
|
|
27
|
+
|
|
28
|
+
initErrorMonitor({
|
|
29
|
+
reportUrl: 'https://', // 你的错误上报接口地址
|
|
30
|
+
projectName: 'oms', // 项目名称
|
|
31
|
+
environment: 'production',
|
|
32
|
+
});
|
|
33
|
+
```
|
|
34
|
+
|
|
35
|
+
## 配置项
|
|
36
|
+
|
|
37
|
+
`initErrorMonitor` 函数接受一个配置对象,包含以下属性:
|
|
38
|
+
|
|
39
|
+
| 属性名 | 类型 | 说明 |
|
|
40
|
+
| ------------- | -------- | ------------------------------------------------------------ |
|
|
41
|
+
| `reportUrl` | `string` | **必填**。错误数据将通过 POST 请求发送到的 URL 地址。 |
|
|
42
|
+
| `projectName` | `string` | **必填**。用于标识错误来源的项目名称。 |
|
|
43
|
+
| `environment` | `string` | **必填**。当前运行环境(例如 'production', 'development')。 |
|
|
44
|
+
|
|
45
|
+
## 请求项
|
|
46
|
+
请求体示例
|
|
47
|
+
|
|
48
|
+
- **`userId`** (可选):用户 ID,用于关联用户相关错误。
|
|
49
|
+
- **`errorType`**:错误类型,例如 `'JavaScript Error'`、`'Promise Rejection'` 等。
|
|
50
|
+
- **`message`**:错误描述信息。
|
|
51
|
+
- **`stack`** (可选):错误的调用栈信息。
|
|
52
|
+
- **`source`**:错误发生的资源 URL。
|
|
53
|
+
- **`projectName`**:项目名称。
|
|
54
|
+
- **`environment`**:当前运行环境。
|
|
55
|
+
- **`userAgent`**:用户代理字符串。
|
|
56
|
+
- **`resource`**:错误发生的资源 URL。
|
|
57
|
+
- **`timestamp`**:错误发生的时间戳,ISO 格式字符串。
|
|
58
|
+
|
|
59
|
+
| 属性名 | 类型 | 说明 |
|
|
60
|
+
| ------------- | -------- | ------------------------------------------------------------ |
|
|
61
|
+
| `userId` | `string` | 可选。用户 ID,用于关联错误数据到具体用户。 |
|
|
62
|
+
| `errorType` | `string` | 可选。错误类型,例如 'JavaScript Error', 'Promise Rejection', 'Network Error', 'Resource Load Error', 'Vue Error'。 |
|
|
63
|
+
| `message` | `string` | 可选。错误描述信息。 |
|
|
64
|
+
| `stack` | `string` | 可选。错误的调用栈信息。 |
|
|
65
|
+
| `source` | `string` | 可选。错误发生的文件路径或 URL。 |
|
|
66
|
+
| `projectName` | `string` | 可选。项目名称,用于标识错误来源。 |
|
|
67
|
+
| `environment` | `string` | 可选。运行环境,例如 'production', 'development'。 |
|
|
68
|
+
| `userAgent` | `string` | 可选。用户代理字符串,用于标识客户端浏览器信息。 |
|
|
69
|
+
| `resource` | `string` | 可选。错误发生的资源 URL。 |
|
|
70
|
+
| `timestamp` | `string` | 可选。错误发生的时间戳,ISO 格式字符串。 |
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
## 本地开发
|
|
75
|
+
|
|
76
|
+
1. **克隆仓库**
|
|
77
|
+
|
|
78
|
+
```bash
|
|
79
|
+
git clone http://git.chengyieco.com/chengyi-frondend-monitor-sdk.git
|
|
80
|
+
cd chengyi-frondend-monitor-sdk
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
2. **安装依赖**
|
|
84
|
+
|
|
85
|
+
```bash
|
|
86
|
+
npm install
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
3. **运行开发服务器**
|
|
90
|
+
|
|
91
|
+
```bash
|
|
92
|
+
npm run dev
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
4. **构建生产版本**
|
|
96
|
+
|
|
97
|
+
```bash
|
|
98
|
+
npm run build
|
|
99
|
+
```
|
|
100
|
+
|
|
101
|
+
5. **运行演示 Demo**
|
|
102
|
+
启动本地服务器以测试 SDK 功能。
|
|
103
|
+
```bash
|
|
104
|
+
npm run demo
|
|
105
|
+
```
|
|
106
|
+
然后在浏览器中打开 `http://localhost:3000`(或控制台显示的端口)查看演示。
|
|
107
|
+
|
|
108
|
+
## 许可证
|
|
109
|
+
|
|
110
|
+
MIT
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 开启 JavaScript 错误监控
|
|
3
|
+
* 监听并上报全局的 JavaScript 运行时错误和未处理的 Promise 异常
|
|
4
|
+
*
|
|
5
|
+
* @param {string} reportUrl - 错误上报的服务端接口地址
|
|
6
|
+
* @param {string} projectName - 项目名称,用于在后台区分错误来源
|
|
7
|
+
* @param {string} environment - 运行环境(如 'production', 'development')
|
|
8
|
+
*/
|
|
9
|
+
export declare const monitorJavaScriptErrors: (reportUrl: string, projectName: string, environment: string) => void;
|
|
@@ -0,0 +1,471 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/******************************************************************************
|
|
4
|
+
Copyright (c) Microsoft Corporation.
|
|
5
|
+
|
|
6
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
|
7
|
+
purpose with or without fee is hereby granted.
|
|
8
|
+
|
|
9
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
10
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
11
|
+
AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
12
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
13
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
|
14
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
15
|
+
PERFORMANCE OF THIS SOFTWARE.
|
|
16
|
+
***************************************************************************** */
|
|
17
|
+
/* global Reflect, Promise, SuppressedError, Symbol, Iterator */
|
|
18
|
+
|
|
19
|
+
var extendStatics = function(d, b) {
|
|
20
|
+
extendStatics = Object.setPrototypeOf ||
|
|
21
|
+
({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||
|
|
22
|
+
function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };
|
|
23
|
+
return extendStatics(d, b);
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
function __extends(d, b) {
|
|
27
|
+
if (typeof b !== "function" && b !== null)
|
|
28
|
+
throw new TypeError("Class extends value " + String(b) + " is not a constructor or null");
|
|
29
|
+
extendStatics(d, b);
|
|
30
|
+
function __() { this.constructor = d; }
|
|
31
|
+
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
var __assign = function() {
|
|
35
|
+
__assign = Object.assign || function __assign(t) {
|
|
36
|
+
for (var s, i = 1, n = arguments.length; i < n; i++) {
|
|
37
|
+
s = arguments[i];
|
|
38
|
+
for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];
|
|
39
|
+
}
|
|
40
|
+
return t;
|
|
41
|
+
};
|
|
42
|
+
return __assign.apply(this, arguments);
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
function __awaiter(thisArg, _arguments, P, generator) {
|
|
46
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
47
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
48
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
49
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
50
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
51
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function __generator(thisArg, body) {
|
|
56
|
+
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
|
|
57
|
+
return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
|
|
58
|
+
function verb(n) { return function (v) { return step([n, v]); }; }
|
|
59
|
+
function step(op) {
|
|
60
|
+
if (f) throw new TypeError("Generator is already executing.");
|
|
61
|
+
while (g && (g = 0, op[0] && (_ = 0)), _) try {
|
|
62
|
+
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;
|
|
63
|
+
if (y = 0, t) op = [op[0] & 2, t.value];
|
|
64
|
+
switch (op[0]) {
|
|
65
|
+
case 0: case 1: t = op; break;
|
|
66
|
+
case 4: _.label++; return { value: op[1], done: false };
|
|
67
|
+
case 5: _.label++; y = op[1]; op = [0]; continue;
|
|
68
|
+
case 7: op = _.ops.pop(); _.trys.pop(); continue;
|
|
69
|
+
default:
|
|
70
|
+
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
|
|
71
|
+
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
|
|
72
|
+
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
|
|
73
|
+
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
|
|
74
|
+
if (t[2]) _.ops.pop();
|
|
75
|
+
_.trys.pop(); continue;
|
|
76
|
+
}
|
|
77
|
+
op = body.call(thisArg, _);
|
|
78
|
+
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
|
|
79
|
+
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
function __spreadArray(to, from, pack) {
|
|
84
|
+
if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
|
|
85
|
+
if (ar || !(i in from)) {
|
|
86
|
+
if (!ar) ar = Array.prototype.slice.call(from, 0, i);
|
|
87
|
+
ar[i] = from[i];
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
return to.concat(ar || Array.prototype.slice.call(from));
|
|
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
|
+
/**
|
|
99
|
+
* 获取浏览器信息
|
|
100
|
+
* 解析 userAgent 获取浏览器名称和版本
|
|
101
|
+
*
|
|
102
|
+
* @returns {{ name: string, version: string }} 包含浏览器名称和版本的对象
|
|
103
|
+
*/
|
|
104
|
+
var getBrowserInfo = function () {
|
|
105
|
+
var ua = navigator.userAgent;
|
|
106
|
+
var tem;
|
|
107
|
+
var match = ua.match(/(opera|chrome|safari|firefox|msie|trident(?=\/))\/?\s*(\d+)/i) || [];
|
|
108
|
+
if (/trident/i.test(match[1])) {
|
|
109
|
+
tem = /\brv[ :]+(\d+)/g.exec(ua) || [];
|
|
110
|
+
return { name: 'IE', version: tem[1] || '' };
|
|
111
|
+
}
|
|
112
|
+
if (match[1] === 'Chrome') {
|
|
113
|
+
tem = ua.match(/\b(OPR|Edge)\/(\d+)/);
|
|
114
|
+
if (tem != null) {
|
|
115
|
+
return { name: tem[1].replace('OPR', 'Opera'), version: tem[2] };
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
match[2] = match[2] || '';
|
|
119
|
+
var name = match[1];
|
|
120
|
+
var version = match[2];
|
|
121
|
+
return { name: name, version: version };
|
|
122
|
+
};
|
|
123
|
+
/**
|
|
124
|
+
* 格式化错误消息
|
|
125
|
+
*
|
|
126
|
+
* @param {any} err - 错误对象
|
|
127
|
+
* @returns {string} 格式化后的错误字符串
|
|
128
|
+
*/
|
|
129
|
+
var formatErrorMessage = function (err) {
|
|
130
|
+
if (err instanceof Error) {
|
|
131
|
+
return err.message;
|
|
132
|
+
}
|
|
133
|
+
if (typeof err === 'string') {
|
|
134
|
+
return err;
|
|
135
|
+
}
|
|
136
|
+
try {
|
|
137
|
+
return JSON.stringify(err);
|
|
138
|
+
}
|
|
139
|
+
catch (_a) {
|
|
140
|
+
return String(err);
|
|
141
|
+
}
|
|
142
|
+
};
|
|
143
|
+
var getUserInfo = function () {
|
|
144
|
+
var info = JSON.parse(localStorage.getItem('user-info') || '{}') || {
|
|
145
|
+
id: null
|
|
146
|
+
};
|
|
147
|
+
return info;
|
|
148
|
+
};
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* 发送错误数据到指定URL
|
|
152
|
+
*
|
|
153
|
+
* @param {Record<string, any>} errorData - 要发送的错误数据对象
|
|
154
|
+
* @param {string} url - 目标API端点URL
|
|
155
|
+
*
|
|
156
|
+
* @description
|
|
157
|
+
* 优先使用navigator.sendBeacon方法发送数据,如果浏览器不支持则回退到fetch API。
|
|
158
|
+
* sendBeacon方法在页面卸载时也能可靠地发送数据。
|
|
159
|
+
*/
|
|
160
|
+
var sendErrorData = function (errorData, url) {
|
|
161
|
+
// 获取浏览器信息并合并到错误数据中
|
|
162
|
+
var browserInfo = getBrowserInfo();
|
|
163
|
+
var userInfo = getUserInfo();
|
|
164
|
+
var dataToSend = __assign(__assign(__assign({}, errorData), browserInfo), { resource: window.location.href });
|
|
165
|
+
if (userInfo.id) {
|
|
166
|
+
dataToSend.userId = userInfo.id;
|
|
167
|
+
}
|
|
168
|
+
if (navigator.sendBeacon) {
|
|
169
|
+
var blob = new Blob([JSON.stringify(dataToSend)], { type: 'application/json' });
|
|
170
|
+
navigator.sendBeacon(url, blob);
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
fetch(url, {
|
|
174
|
+
method: 'POST',
|
|
175
|
+
headers: { 'Content-Type': 'application/json' },
|
|
176
|
+
body: JSON.stringify(dataToSend),
|
|
177
|
+
keepalive: true, // 即使页面关闭也能发送请求,作用类似 sendBeacon
|
|
178
|
+
}).catch(function (error) { return console.error('Error reporting failed:', error); });
|
|
179
|
+
}
|
|
180
|
+
};
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* 开启 JavaScript 错误监控
|
|
184
|
+
* 监听并上报全局的 JavaScript 运行时错误和未处理的 Promise 异常
|
|
185
|
+
*
|
|
186
|
+
* @param {string} reportUrl - 错误上报的服务端接口地址
|
|
187
|
+
* @param {string} projectName - 项目名称,用于在后台区分错误来源
|
|
188
|
+
* @param {string} environment - 运行环境(如 'production', 'development')
|
|
189
|
+
*/
|
|
190
|
+
var monitorJavaScriptErrors = function (reportUrl, projectName, environment) {
|
|
191
|
+
// 保存原有的 onerror 处理函数,防止覆盖
|
|
192
|
+
var originalOnError = window.onerror;
|
|
193
|
+
/**
|
|
194
|
+
* 捕获未处理的 JavaScript 错误
|
|
195
|
+
* @param message (string): 错误的描述信息(例如 "Uncaught ReferenceError: x is not defined")。
|
|
196
|
+
* @param source (string): 发生错误的脚本文件的 URL。
|
|
197
|
+
* @param lineno (number): 错误发生的行号。
|
|
198
|
+
* @param colno (number): 错误发生的列号。
|
|
199
|
+
* @param error (Error Object): 包含详细堆栈信息(Stack Trace)的 Error 对象。这是最有用的部分,因为它能告诉你错误调用的上下文。
|
|
200
|
+
*/
|
|
201
|
+
window.onerror = function (message, source, lineno, colno, error) {
|
|
202
|
+
var errorInfo = {
|
|
203
|
+
message: message,
|
|
204
|
+
source: source,
|
|
205
|
+
lineno: lineno,
|
|
206
|
+
colno: colno,
|
|
207
|
+
stack: error ? error.stack : null,
|
|
208
|
+
projectName: projectName,
|
|
209
|
+
environment: environment,
|
|
210
|
+
errorType: 'JavaScript Error',
|
|
211
|
+
timestamp: new Date().toISOString(),
|
|
212
|
+
userAgent: navigator.userAgent,
|
|
213
|
+
};
|
|
214
|
+
// 上报错误数据
|
|
215
|
+
sendErrorData(errorInfo, reportUrl);
|
|
216
|
+
// 如果原来有 onerror 处理函数,继续执行它
|
|
217
|
+
// 这样做是为了不破坏宿主环境(例如用户自己写的或其他 SDK)已有的错误处理逻辑
|
|
218
|
+
if (originalOnError) {
|
|
219
|
+
return originalOnError(message, source, lineno, colno, error);
|
|
220
|
+
}
|
|
221
|
+
};
|
|
222
|
+
// 保存原有的处理函数
|
|
223
|
+
var originalOnUnhandledRejection = window.onunhandledrejection;
|
|
224
|
+
// 捕获未处理的 Promise 错误
|
|
225
|
+
window.onunhandledrejection = function (event) {
|
|
226
|
+
var reason = event.reason;
|
|
227
|
+
var errorInfo = {
|
|
228
|
+
message: reason ? (reason.message || String(reason)) : 'Unknown Promise Error',
|
|
229
|
+
stack: reason && reason.stack ? reason.stack : null,
|
|
230
|
+
projectName: projectName,
|
|
231
|
+
environment: environment,
|
|
232
|
+
errorType: 'Unhandled Promise Rejection',
|
|
233
|
+
timestamp: new Date().toISOString(),
|
|
234
|
+
userAgent: navigator.userAgent,
|
|
235
|
+
};
|
|
236
|
+
// 上报错误数据
|
|
237
|
+
sendErrorData(errorInfo, reportUrl);
|
|
238
|
+
// 如果原来有处理函数,继续执行它
|
|
239
|
+
// 保持对原有 Promise 错误处理逻辑的兼容性
|
|
240
|
+
if (originalOnUnhandledRejection) {
|
|
241
|
+
return originalOnUnhandledRejection.call(window, event);
|
|
242
|
+
}
|
|
243
|
+
};
|
|
244
|
+
};
|
|
245
|
+
|
|
246
|
+
/**
|
|
247
|
+
* 开启网络错误监控
|
|
248
|
+
* 劫持并监听 XMLHttpRequest 和 fetch 请求,捕获网络异常
|
|
249
|
+
*
|
|
250
|
+
* @param {string} reportUrl - 错误上报地址
|
|
251
|
+
* @param {string} projectName - 项目名称
|
|
252
|
+
* @param {string} environment - 运行环境
|
|
253
|
+
*/
|
|
254
|
+
var monitorNetworkErrors = function (reportUrl, projectName, environment) {
|
|
255
|
+
// 劫持 XMLHttpRequest
|
|
256
|
+
// 保存原生的 open 方法,以便后续调用
|
|
257
|
+
var originalXhrOpen = XMLHttpRequest.prototype.open;
|
|
258
|
+
XMLHttpRequest.prototype.open = function (method, url) {
|
|
259
|
+
var args = [];
|
|
260
|
+
for (var _i = 2; _i < arguments.length; _i++) {
|
|
261
|
+
args[_i - 2] = arguments[_i];
|
|
262
|
+
}
|
|
263
|
+
var _urlStr = typeof url === 'string' ? url : String(url);
|
|
264
|
+
if (_urlStr.includes(reportUrl)) {
|
|
265
|
+
return originalXhrOpen.apply(this, __spreadArray([method, url], args, true));
|
|
266
|
+
}
|
|
267
|
+
// 监听 error 事件,当请求失败时触发
|
|
268
|
+
this.addEventListener('error', function () {
|
|
269
|
+
var errorInfo = {
|
|
270
|
+
message: "Network Error: ".concat(method, " ").concat(url),
|
|
271
|
+
projectName: projectName,
|
|
272
|
+
environment: environment,
|
|
273
|
+
errorType: 'Network Error',
|
|
274
|
+
timestamp: new Date().toISOString(),
|
|
275
|
+
userAgent: navigator.userAgent,
|
|
276
|
+
};
|
|
277
|
+
sendErrorData(errorInfo, reportUrl);
|
|
278
|
+
});
|
|
279
|
+
// 调用原生的 open 方法,保证正常请求流程
|
|
280
|
+
return originalXhrOpen.apply(this, __spreadArray([method, url], args, true));
|
|
281
|
+
};
|
|
282
|
+
// 劫持 fetch
|
|
283
|
+
// 保存原生的 fetch 方法
|
|
284
|
+
var originalFetch = window.fetch;
|
|
285
|
+
window.fetch = function (input, init) { return __awaiter(void 0, void 0, void 0, function () {
|
|
286
|
+
var urlStr, response, errorInfo, error_1, errorInfo;
|
|
287
|
+
return __generator(this, function (_a) {
|
|
288
|
+
switch (_a.label) {
|
|
289
|
+
case 0:
|
|
290
|
+
urlStr = (input instanceof Request) ? input.url : String(input);
|
|
291
|
+
if (urlStr.includes(reportUrl)) {
|
|
292
|
+
return [2 /*return*/, originalFetch(input, init)];
|
|
293
|
+
}
|
|
294
|
+
_a.label = 1;
|
|
295
|
+
case 1:
|
|
296
|
+
_a.trys.push([1, 3, , 4]);
|
|
297
|
+
return [4 /*yield*/, originalFetch(input, init)];
|
|
298
|
+
case 2:
|
|
299
|
+
response = _a.sent();
|
|
300
|
+
// 检查响应状态,如果不是 2xx 则视为错误
|
|
301
|
+
if (!response.ok) {
|
|
302
|
+
errorInfo = {
|
|
303
|
+
message: "Network Error: ".concat(response.status, " ").concat(response.statusText),
|
|
304
|
+
url: input instanceof Request ? input.url : input,
|
|
305
|
+
projectName: projectName,
|
|
306
|
+
environment: environment,
|
|
307
|
+
errorType: 'Fetch Error',
|
|
308
|
+
timestamp: new Date().toISOString(),
|
|
309
|
+
userAgent: navigator.userAgent,
|
|
310
|
+
};
|
|
311
|
+
sendErrorData(errorInfo, reportUrl);
|
|
312
|
+
}
|
|
313
|
+
return [2 /*return*/, response];
|
|
314
|
+
case 3:
|
|
315
|
+
error_1 = _a.sent();
|
|
316
|
+
errorInfo = {
|
|
317
|
+
message: "Fetch failed: ".concat((input instanceof Request ? input.url : input)),
|
|
318
|
+
projectName: projectName,
|
|
319
|
+
environment: environment,
|
|
320
|
+
errorType: 'Fetch Error',
|
|
321
|
+
timestamp: new Date().toISOString(),
|
|
322
|
+
userAgent: navigator.userAgent,
|
|
323
|
+
};
|
|
324
|
+
sendErrorData(errorInfo, reportUrl);
|
|
325
|
+
throw error_1;
|
|
326
|
+
case 4: return [2 /*return*/];
|
|
327
|
+
}
|
|
328
|
+
});
|
|
329
|
+
}); };
|
|
330
|
+
};
|
|
331
|
+
|
|
332
|
+
/**
|
|
333
|
+
* 开启资源加载错误监控
|
|
334
|
+
* 监听图片、脚本等资源加载失败的错误
|
|
335
|
+
*
|
|
336
|
+
* @param {string} reportUrl - 错误上报地址
|
|
337
|
+
* @param {string} projectName - 项目名称
|
|
338
|
+
* @param {string} environment - 运行环境
|
|
339
|
+
*/
|
|
340
|
+
var monitorResourceErrors = function (reportUrl, projectName, environment) {
|
|
341
|
+
/**
|
|
342
|
+
* 监听全局 error 事件
|
|
343
|
+
* 注意:资源加载错误不会冒泡,所以必须在捕获阶段处理 (useCapture = true)
|
|
344
|
+
*/
|
|
345
|
+
window.addEventListener('error', function (event) {
|
|
346
|
+
var target = event.target;
|
|
347
|
+
// 过滤出 IMG 和 SCRIPT 标签的错误
|
|
348
|
+
if (target && (target.tagName === 'IMG' || target.tagName === 'SCRIPT')) {
|
|
349
|
+
var errorInfo = {
|
|
350
|
+
message: "Resource Load Error: ".concat(target.tagName, " ").concat(target.getAttribute('src') || target.getAttribute('href')),
|
|
351
|
+
projectName: projectName,
|
|
352
|
+
environment: environment,
|
|
353
|
+
errorType: 'Resource Load Error',
|
|
354
|
+
timestamp: new Date().toISOString(),
|
|
355
|
+
userAgent: navigator.userAgent,
|
|
356
|
+
};
|
|
357
|
+
sendErrorData(errorInfo, reportUrl);
|
|
358
|
+
}
|
|
359
|
+
}, true); // useCapture 为 true 确保资源错误被捕获
|
|
360
|
+
};
|
|
361
|
+
|
|
362
|
+
var __monitorInitialized = false;
|
|
363
|
+
/**
|
|
364
|
+
* 初始化错误监控 SDK
|
|
365
|
+
* 该函数是 SDK 的入口,负责启动各类错误监控模块
|
|
366
|
+
*
|
|
367
|
+
* @param {ErrorMonitorConfig} config - 监控配置对象
|
|
368
|
+
*/
|
|
369
|
+
var initErrorMonitor = function (config) {
|
|
370
|
+
var reportUrl = config.reportUrl, projectName = config.projectName, environment = config.environment;
|
|
371
|
+
// 初始化各类错误监控
|
|
372
|
+
monitorJavaScriptErrors(reportUrl, projectName, environment);
|
|
373
|
+
monitorNetworkErrors(reportUrl, projectName, environment);
|
|
374
|
+
monitorResourceErrors(reportUrl, projectName, environment);
|
|
375
|
+
};
|
|
376
|
+
var VueErrorMonitorPlugin = {
|
|
377
|
+
/**
|
|
378
|
+
* Vue 插件安装函数
|
|
379
|
+
*
|
|
380
|
+
* @param {any} app - Vue 应用实例
|
|
381
|
+
* @param {object} options - 插件配置项
|
|
382
|
+
* @param {string} options.reportUrl - 上报错误服务端地址
|
|
383
|
+
* @param {string} options.projectName - 项目名称
|
|
384
|
+
* @param {string} options.environment - 当前环境
|
|
385
|
+
*/
|
|
386
|
+
install: function (app, options) {
|
|
387
|
+
if (!options || !options.reportUrl)
|
|
388
|
+
return;
|
|
389
|
+
if (!__monitorInitialized) {
|
|
390
|
+
initErrorMonitor(options);
|
|
391
|
+
__monitorInitialized = true;
|
|
392
|
+
}
|
|
393
|
+
var cfg = app.config;
|
|
394
|
+
var original = cfg.errorHandler;
|
|
395
|
+
/**
|
|
396
|
+
* 错误处理函数
|
|
397
|
+
*
|
|
398
|
+
* @param {unknown} err - 错误对象
|
|
399
|
+
* @param {unknown} [instance] - Vue 实例
|
|
400
|
+
* @param {unknown} [info] - 错误信息
|
|
401
|
+
*/
|
|
402
|
+
cfg.errorHandler = function (err, instance, info) {
|
|
403
|
+
var e = err;
|
|
404
|
+
sendErrorData({
|
|
405
|
+
message: formatErrorMessage(err),
|
|
406
|
+
stack: e && e.stack ? e.stack : null,
|
|
407
|
+
projectName: options.projectName,
|
|
408
|
+
environment: options.environment,
|
|
409
|
+
errorType: 'Vue Error',
|
|
410
|
+
timestamp: new Date().toISOString(),
|
|
411
|
+
userAgent: navigator.userAgent,
|
|
412
|
+
info: info,
|
|
413
|
+
}, options.reportUrl);
|
|
414
|
+
if (typeof original === 'function') {
|
|
415
|
+
try {
|
|
416
|
+
original(err, instance, info);
|
|
417
|
+
}
|
|
418
|
+
catch (_a) { }
|
|
419
|
+
}
|
|
420
|
+
};
|
|
421
|
+
}
|
|
422
|
+
};
|
|
423
|
+
/**
|
|
424
|
+
* 创建 React 错误边界组件
|
|
425
|
+
*
|
|
426
|
+
* @param {object} React - React 库对象
|
|
427
|
+
* @param {object} config - 错误监控配置
|
|
428
|
+
* @param {string} config.reportUrl - 上报错误服务端地址
|
|
429
|
+
* @param {string} config.projectName - 项目名称
|
|
430
|
+
* @param {string} config.environment - 当前环境
|
|
431
|
+
* @returns {React.Component} - 创建的错误边界组件
|
|
432
|
+
*/
|
|
433
|
+
var createReactErrorBoundary = function (React, config) {
|
|
434
|
+
// 确保在创建边界组件时启动全局监控(JS/网络/资源)
|
|
435
|
+
if (config && config.reportUrl && !__monitorInitialized) {
|
|
436
|
+
initErrorMonitor(config);
|
|
437
|
+
__monitorInitialized = true;
|
|
438
|
+
}
|
|
439
|
+
return /** @class */ (function (_super) {
|
|
440
|
+
__extends(ErrorMonitorBoundary, _super);
|
|
441
|
+
function ErrorMonitorBoundary(props) {
|
|
442
|
+
var _this = _super.call(this, props) || this;
|
|
443
|
+
_this.state = { hasError: false };
|
|
444
|
+
return _this;
|
|
445
|
+
}
|
|
446
|
+
ErrorMonitorBoundary.getDerivedStateFromError = function () { return { hasError: true }; };
|
|
447
|
+
ErrorMonitorBoundary.prototype.componentDidCatch = function (error, info) {
|
|
448
|
+
sendErrorData({
|
|
449
|
+
message: formatErrorMessage(error),
|
|
450
|
+
stack: (error === null || error === void 0 ? void 0 : error.stack) || null,
|
|
451
|
+
projectName: config.projectName,
|
|
452
|
+
environment: config.environment,
|
|
453
|
+
errorType: 'React Error',
|
|
454
|
+
timestamp: new Date().toISOString(),
|
|
455
|
+
userAgent: navigator.userAgent,
|
|
456
|
+
componentStack: (info === null || info === void 0 ? void 0 : info.componentStack) || null,
|
|
457
|
+
}, config.reportUrl);
|
|
458
|
+
};
|
|
459
|
+
ErrorMonitorBoundary.prototype.render = function () {
|
|
460
|
+
if (this.state.hasError)
|
|
461
|
+
return this.props.fallback || null;
|
|
462
|
+
return this.props.children;
|
|
463
|
+
};
|
|
464
|
+
return ErrorMonitorBoundary;
|
|
465
|
+
}(React.Component));
|
|
466
|
+
};
|
|
467
|
+
|
|
468
|
+
exports.VueErrorMonitorPlugin = VueErrorMonitorPlugin;
|
|
469
|
+
exports.createReactErrorBoundary = createReactErrorBoundary;
|
|
470
|
+
exports.initErrorMonitor = initErrorMonitor;
|
|
471
|
+
//# sourceMappingURL=index.cjs.js.map
|