foliko 1.0.75 → 1.0.76
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/.claude/settings.local.json +159 -157
- package/cli/bin/foliko.js +12 -12
- package/cli/src/commands/chat.js +143 -143
- package/cli/src/commands/list.js +93 -93
- package/cli/src/index.js +75 -75
- package/cli/src/ui/chat-ui.js +201 -201
- package/cli/src/utils/ansi.js +40 -40
- package/cli/src/utils/markdown.js +292 -292
- package/examples/ambient-example.js +194 -194
- package/examples/basic.js +115 -115
- package/examples/bootstrap.js +121 -121
- package/examples/mcp-example.js +56 -56
- package/examples/skill-example.js +49 -49
- package/examples/test-chat.js +137 -137
- package/examples/test-mcp.js +85 -85
- package/examples/test-reload.js +59 -59
- package/examples/test-telegram.js +50 -50
- package/examples/test-tg-bot.js +45 -45
- package/examples/test-tg-simple.js +47 -47
- package/examples/test-tg.js +62 -62
- package/examples/test-think.js +43 -43
- package/examples/test-web-plugin.js +103 -103
- package/examples/test-weixin-feishu.js +103 -103
- package/examples/workflow.js +158 -158
- package/package.json +1 -1
- package/plugins/ai-plugin.js +102 -102
- package/plugins/ambient-agent/EventWatcher.js +113 -113
- package/plugins/ambient-agent/ExplorerLoop.js +640 -640
- package/plugins/ambient-agent/GoalManager.js +197 -197
- package/plugins/ambient-agent/Reflector.js +95 -95
- package/plugins/ambient-agent/StateStore.js +90 -90
- package/plugins/ambient-agent/constants.js +101 -101
- package/plugins/ambient-agent/index.js +579 -579
- package/plugins/audit-plugin.js +187 -187
- package/plugins/default-plugins.js +662 -662
- package/plugins/email/constants.js +64 -64
- package/plugins/email/handlers.js +461 -461
- package/plugins/email/index.js +278 -278
- package/plugins/email/monitor.js +269 -269
- package/plugins/email/parser.js +138 -138
- package/plugins/email/reply.js +151 -151
- package/plugins/email/utils.js +124 -124
- package/plugins/feishu-plugin.js +481 -481
- package/plugins/file-system-plugin.js +826 -826
- package/plugins/install-plugin.js +199 -199
- package/plugins/python-executor-plugin.js +367 -367
- package/plugins/python-plugin-loader.js +481 -481
- package/plugins/rules-plugin.js +294 -294
- package/plugins/scheduler-plugin.js +691 -691
- package/plugins/session-plugin.js +369 -369
- package/plugins/shell-executor-plugin.js +197 -197
- package/plugins/storage-plugin.js +240 -240
- package/plugins/subagent-plugin.js +845 -845
- package/plugins/telegram-plugin.js +482 -482
- package/plugins/think-plugin.js +345 -345
- package/plugins/tools-plugin.js +196 -196
- package/plugins/web-plugin.js +606 -606
- package/plugins/weixin-plugin.js +545 -545
- package/src/capabilities/index.js +11 -11
- package/src/capabilities/skill-manager.js +609 -609
- package/src/capabilities/workflow-engine.js +1109 -1109
- package/src/core/agent-chat.js +882 -882
- package/src/core/agent.js +892 -892
- package/src/core/framework.js +465 -465
- package/src/core/index.js +19 -19
- package/src/core/plugin-base.js +219 -219
- package/src/core/plugin-manager.js +863 -863
- package/src/core/provider.js +114 -114
- package/src/core/sub-agent-config.js +264 -264
- package/src/core/system-prompt-builder.js +120 -120
- package/src/core/tool-registry.js +517 -517
- package/src/core/tool-router.js +297 -297
- package/src/executors/executor-base.js +58 -58
- package/src/executors/mcp-executor.js +741 -741
- package/src/index.js +25 -25
- package/src/utils/circuit-breaker.js +301 -301
- package/src/utils/error-boundary.js +363 -363
- package/src/utils/error.js +374 -374
- package/src/utils/event-emitter.js +97 -97
- package/src/utils/id.js +133 -133
- package/src/utils/index.js +217 -217
- package/src/utils/logger.js +181 -181
- package/src/utils/plugin-helpers.js +90 -90
- package/src/utils/retry.js +122 -122
- package/src/utils/sandbox.js +292 -292
- package/test/tool-registry-validation.test.js +218 -218
- package/website/script.js +136 -136
- package/foliko-1.0.75.tgz +0 -0
|
@@ -1,363 +1,363 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Foliko Error Boundary - 错误边界系统
|
|
3
|
-
* 捕获、处理和恢复错误,提供降级方案
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
const { EventEmitter } = require('./event-emitter');
|
|
7
|
-
const { logger } = require('./logger');
|
|
8
|
-
const { safeErrorInfo, isErrorOfType } = require('./error');
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* 错误严重级别
|
|
12
|
-
*/
|
|
13
|
-
const Severity = {
|
|
14
|
-
LOW: 'low', // 可忽略的小问题
|
|
15
|
-
MEDIUM: 'medium', // 需要关注但不阻断流程
|
|
16
|
-
HIGH: 'high', // 阻断当前操作但可恢复
|
|
17
|
-
CRITICAL: 'critical', // 致命错误,需要完全降级
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
/**
|
|
21
|
-
* 错误恢复动作
|
|
22
|
-
*/
|
|
23
|
-
const RecoveryAction = {
|
|
24
|
-
RETRY: 'retry', // 重试操作
|
|
25
|
-
SKIP: 'skip', // 跳过当前步骤
|
|
26
|
-
FALLBACK: 'fallback', // 使用降级方案
|
|
27
|
-
ABORT: 'abort', // 中止流程
|
|
28
|
-
IGNORE: 'ignore', // 忽略错误继续
|
|
29
|
-
};
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* 错误上下文
|
|
33
|
-
* @typedef {Object} ErrorContext
|
|
34
|
-
* @property {string} operation - 操作名称
|
|
35
|
-
* @property {string} component - 组件名称
|
|
36
|
-
* @property {Object} metadata - 额外元数据
|
|
37
|
-
* @property {number} attempt - 当前尝试次数
|
|
38
|
-
*/
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* 错误边界配置
|
|
42
|
-
* @typedef {Object} ErrorBoundaryConfig
|
|
43
|
-
* @property {string} name - 边界名称
|
|
44
|
-
* @property {Function} [onError] - 全局错误处理器
|
|
45
|
-
* @property {Function} [onRecovery] - 恢复处理器
|
|
46
|
-
* @property {Function} [fallback] - 降级函数
|
|
47
|
-
* @property {boolean} [logErrors=true] - 是否记录错误
|
|
48
|
-
* @property {boolean} [propagateErrors=false] - 是否向上传播错误
|
|
49
|
-
*/
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* 错误边界类
|
|
53
|
-
* 用于捕获和管理错误,提供恢复机制
|
|
54
|
-
*/
|
|
55
|
-
class ErrorBoundary extends EventEmitter {
|
|
56
|
-
/**
|
|
57
|
-
* @param {ErrorBoundaryConfig} config
|
|
58
|
-
*/
|
|
59
|
-
constructor(config = {}) {
|
|
60
|
-
super();
|
|
61
|
-
|
|
62
|
-
this.name = config.name || 'AnonymousBoundary';
|
|
63
|
-
this._onError = config.onError || null;
|
|
64
|
-
this._onRecovery = config.onRecovery || null;
|
|
65
|
-
this._fallback = config.fallback || null;
|
|
66
|
-
this._logErrors = config.logErrors !== false;
|
|
67
|
-
this._propagateErrors = config.propagateErrors || false;
|
|
68
|
-
this._logger = logger.child('ErrorBoundary');
|
|
69
|
-
|
|
70
|
-
// 错误统计
|
|
71
|
-
this._errorCount = 0;
|
|
72
|
-
this._lastError = null;
|
|
73
|
-
this._errorHistory = [];
|
|
74
|
-
this._maxHistorySize = config.maxHistorySize || 50;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* 执行可能抛出错误的操作
|
|
79
|
-
* @param {Function} operation - 要执行的操作
|
|
80
|
-
* @param {Object} [context] - 错误上下文
|
|
81
|
-
* @returns {Promise<{success: boolean, result?: any, error?: Error, recovered?: boolean}>}
|
|
82
|
-
*/
|
|
83
|
-
async execute(operation, context = {}) {
|
|
84
|
-
const operationName = context.operation || 'anonymous';
|
|
85
|
-
|
|
86
|
-
try {
|
|
87
|
-
const result = await (typeof operation === 'function' ? operation() : operation);
|
|
88
|
-
return { success: true, result };
|
|
89
|
-
} catch (error) {
|
|
90
|
-
this._errorCount++;
|
|
91
|
-
this._lastError = error;
|
|
92
|
-
|
|
93
|
-
// 记录到历史
|
|
94
|
-
this._addToHistory(error, context);
|
|
95
|
-
|
|
96
|
-
// 记录错误日志
|
|
97
|
-
if (this._logErrors) {
|
|
98
|
-
this._logError(error, context);
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
// 触发错误事件
|
|
102
|
-
this.emit('error', {
|
|
103
|
-
error,
|
|
104
|
-
context,
|
|
105
|
-
boundary: this.name,
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
// 调用全局错误处理器
|
|
109
|
-
let recoveryAction = RecoveryAction.ABORT;
|
|
110
|
-
if (this._onError) {
|
|
111
|
-
try {
|
|
112
|
-
// 使用 Promise.resolve 以支持 async 函数
|
|
113
|
-
recoveryAction = await Promise.resolve(this._onError(error, context, this));
|
|
114
|
-
} catch (handlerError) {
|
|
115
|
-
this._logger.warn(`Error in onError handler: ${handlerError.message}`);
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
// 根据恢复动作处理
|
|
120
|
-
const { action, fallbackResult } = await this._handleRecovery(error, context, recoveryAction);
|
|
121
|
-
|
|
122
|
-
// 触发恢复事件
|
|
123
|
-
if (action !== RecoveryAction.ABORT) {
|
|
124
|
-
this.emit('recovery', {
|
|
125
|
-
error,
|
|
126
|
-
context,
|
|
127
|
-
action,
|
|
128
|
-
boundary: this.name,
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
if (this._onRecovery) {
|
|
132
|
-
try {
|
|
133
|
-
await this._onRecovery(error, context, action, this);
|
|
134
|
-
} catch (recoveryError) {
|
|
135
|
-
log.warn(`Error in onRecovery handler: ${recoveryError.message}`);
|
|
136
|
-
}
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
// 如果需要传播错误,或者没有降级方案且要求传播,则抛出错误
|
|
141
|
-
if (this._propagateErrors || (action === RecoveryAction.ABORT && !fallbackResult)) {
|
|
142
|
-
throw error;
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
return {
|
|
146
|
-
success: false,
|
|
147
|
-
error,
|
|
148
|
-
recovered: action !== RecoveryAction.ABORT,
|
|
149
|
-
fallbackResult,
|
|
150
|
-
action,
|
|
151
|
-
};
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
/**
|
|
156
|
-
* 执行带重试的操作
|
|
157
|
-
* @param {Function} operation - 要执行的操作
|
|
158
|
-
* @param {Object} retryConfig - 重试配置
|
|
159
|
-
* @param {Object} [context] - 错误上下文
|
|
160
|
-
* @returns {Promise<any>}
|
|
161
|
-
*/
|
|
162
|
-
async executeWithRetry(operation, retryConfig = {}, context = {}) {
|
|
163
|
-
const { maxAttempts = 3, baseDelay = 1000, shouldRetry = () => true } = retryConfig;
|
|
164
|
-
|
|
165
|
-
let lastError;
|
|
166
|
-
|
|
167
|
-
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
|
|
168
|
-
try {
|
|
169
|
-
return await this.execute(operation, { ...context, attempt });
|
|
170
|
-
} catch (error) {
|
|
171
|
-
lastError = error;
|
|
172
|
-
|
|
173
|
-
if (attempt >= maxAttempts || !shouldRetry(error, attempt)) {
|
|
174
|
-
throw error;
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
// 计算延迟
|
|
178
|
-
const delay = baseDelay * Math.pow(2, attempt - 1);
|
|
179
|
-
log.debug(`Retry attempt ${attempt}/${maxAttempts} after ${delay}ms...`);
|
|
180
|
-
|
|
181
|
-
await new Promise((resolve) => setTimeout(resolve, delay));
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
|
|
185
|
-
throw lastError;
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
/**
|
|
189
|
-
* 包装一个函数,使其错误被边界捕获
|
|
190
|
-
* @param {Function} fn - 要包装的函数
|
|
191
|
-
* @param {Object} [defaultContext] - 默认上下文
|
|
192
|
-
* @returns {Function}
|
|
193
|
-
*/
|
|
194
|
-
wrap(fn, defaultContext = {}) {
|
|
195
|
-
return async (...args) => {
|
|
196
|
-
const result = await this.execute(() => fn(...args), defaultContext);
|
|
197
|
-
return result.success ? result.result : result.fallbackResult;
|
|
198
|
-
};
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
/**
|
|
202
|
-
* 处理错误恢复
|
|
203
|
-
* @private
|
|
204
|
-
*/
|
|
205
|
-
async _handleRecovery(error, context, recoveryAction) {
|
|
206
|
-
let fallbackResult = null;
|
|
207
|
-
let action = recoveryAction;
|
|
208
|
-
|
|
209
|
-
switch (recoveryAction) {
|
|
210
|
-
case RecoveryAction.FALLBACK:
|
|
211
|
-
if (this._fallback) {
|
|
212
|
-
try {
|
|
213
|
-
fallbackResult = await this._fallback(error, context, this);
|
|
214
|
-
action = RecoveryAction.RETRY; // fallback 成功后视为恢复
|
|
215
|
-
} catch (fallbackError) {
|
|
216
|
-
log.warn(`Fallback also failed: ${fallbackError.message}`);
|
|
217
|
-
action = RecoveryAction.ABORT;
|
|
218
|
-
}
|
|
219
|
-
}
|
|
220
|
-
break;
|
|
221
|
-
|
|
222
|
-
case RecoveryAction.SKIP:
|
|
223
|
-
log.debug(`Skipping operation: ${context.operation}`);
|
|
224
|
-
fallbackResult = null;
|
|
225
|
-
break;
|
|
226
|
-
|
|
227
|
-
case RecoveryAction.IGNORE:
|
|
228
|
-
log.debug(`Ignoring error: ${error.message}`);
|
|
229
|
-
fallbackResult = undefined;
|
|
230
|
-
break;
|
|
231
|
-
|
|
232
|
-
default:
|
|
233
|
-
action = RecoveryAction.ABORT;
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
return { action, fallbackResult };
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
/**
|
|
240
|
-
* 记录错误
|
|
241
|
-
* @private
|
|
242
|
-
*/
|
|
243
|
-
_logError(error, context) {
|
|
244
|
-
if (!this._logErrors) return;
|
|
245
|
-
|
|
246
|
-
const errorInfo = safeErrorInfo(error);
|
|
247
|
-
const severity = this._determineSeverity(error);
|
|
248
|
-
|
|
249
|
-
const logMessage = `[${this.name}] Error in ${context.operation || 'unknown'}: ${error.message}`;
|
|
250
|
-
|
|
251
|
-
switch (severity) {
|
|
252
|
-
case Severity.LOW:
|
|
253
|
-
this._logger.debug(logMessage);
|
|
254
|
-
break;
|
|
255
|
-
case Severity.MEDIUM:
|
|
256
|
-
this._logger.warn(logMessage);
|
|
257
|
-
break;
|
|
258
|
-
case Severity.HIGH:
|
|
259
|
-
this._logger.error(logMessage);
|
|
260
|
-
break;
|
|
261
|
-
case Severity.CRITICAL:
|
|
262
|
-
this._logger.error(`CRITICAL: ${logMessage}`, { stack: error.stack });
|
|
263
|
-
break;
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
/**
|
|
268
|
-
* 确定错误严重级别
|
|
269
|
-
* @private
|
|
270
|
-
*/
|
|
271
|
-
_determineSeverity(error) {
|
|
272
|
-
if (isErrorOfType(error, require('./error').FolikoError)) {
|
|
273
|
-
return error.context?.severity || Severity.MEDIUM;
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
// 根据错误类型和消息判断
|
|
277
|
-
const message = (error.message || '').toLowerCase();
|
|
278
|
-
|
|
279
|
-
if (message.includes('timeout') || message.includes('network')) {
|
|
280
|
-
return Severity.MEDIUM;
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
if (message.includes('memory') || message.includes('fatal')) {
|
|
284
|
-
return Severity.CRITICAL;
|
|
285
|
-
}
|
|
286
|
-
|
|
287
|
-
return Severity.HIGH;
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
/**
|
|
291
|
-
* 添加错误到历史记录
|
|
292
|
-
* @private
|
|
293
|
-
*/
|
|
294
|
-
_addToHistory(error, context) {
|
|
295
|
-
this._errorHistory.push({
|
|
296
|
-
timestamp: new Date().toISOString(),
|
|
297
|
-
error: safeErrorInfo(error),
|
|
298
|
-
context,
|
|
299
|
-
});
|
|
300
|
-
|
|
301
|
-
// 保持历史记录大小
|
|
302
|
-
if (this._errorHistory.length > this._maxHistorySize) {
|
|
303
|
-
this._errorHistory.shift();
|
|
304
|
-
}
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
/**
|
|
308
|
-
* 获取错误统计
|
|
309
|
-
* @returns {Object}
|
|
310
|
-
*/
|
|
311
|
-
getStats() {
|
|
312
|
-
return {
|
|
313
|
-
totalErrors: this._errorCount,
|
|
314
|
-
lastError: this._lastError ? safeErrorInfo(this._lastError) : null,
|
|
315
|
-
historySize: this._errorHistory.length,
|
|
316
|
-
recentErrors: this._errorHistory.slice(-10),
|
|
317
|
-
};
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
/**
|
|
321
|
-
* 清空错误历史
|
|
322
|
-
*/
|
|
323
|
-
clearHistory() {
|
|
324
|
-
this._errorHistory = [];
|
|
325
|
-
this._lastError = null;
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
|
|
329
|
-
/**
|
|
330
|
-
* 创建错误边界实例的工厂函数
|
|
331
|
-
* @param {ErrorBoundaryConfig} config
|
|
332
|
-
* @returns {ErrorBoundary}
|
|
333
|
-
*/
|
|
334
|
-
function createErrorBoundary(config = {}) {
|
|
335
|
-
return new ErrorBoundary(config);
|
|
336
|
-
}
|
|
337
|
-
|
|
338
|
-
/**
|
|
339
|
-
* 组合多个错误边界
|
|
340
|
-
* @param {ErrorBoundary[]} boundaries
|
|
341
|
-
* @returns {ErrorBoundary}
|
|
342
|
-
*/
|
|
343
|
-
function combineBoundaries(boundaries) {
|
|
344
|
-
const combined = new ErrorBoundary({
|
|
345
|
-
name: 'CombinedBoundary',
|
|
346
|
-
});
|
|
347
|
-
|
|
348
|
-
// 将错误传播到所有子边界
|
|
349
|
-
for (const boundary of boundaries) {
|
|
350
|
-
boundary.on('error', (data) => combined.emit('error', data));
|
|
351
|
-
boundary.on('recovery', (data) => combined.emit('recovery', data));
|
|
352
|
-
}
|
|
353
|
-
|
|
354
|
-
return combined;
|
|
355
|
-
}
|
|
356
|
-
|
|
357
|
-
module.exports = {
|
|
358
|
-
ErrorBoundary,
|
|
359
|
-
Severity,
|
|
360
|
-
RecoveryAction,
|
|
361
|
-
createErrorBoundary,
|
|
362
|
-
combineBoundaries,
|
|
363
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* Foliko Error Boundary - 错误边界系统
|
|
3
|
+
* 捕获、处理和恢复错误,提供降级方案
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
const { EventEmitter } = require('./event-emitter');
|
|
7
|
+
const { logger } = require('./logger');
|
|
8
|
+
const { safeErrorInfo, isErrorOfType } = require('./error');
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* 错误严重级别
|
|
12
|
+
*/
|
|
13
|
+
const Severity = {
|
|
14
|
+
LOW: 'low', // 可忽略的小问题
|
|
15
|
+
MEDIUM: 'medium', // 需要关注但不阻断流程
|
|
16
|
+
HIGH: 'high', // 阻断当前操作但可恢复
|
|
17
|
+
CRITICAL: 'critical', // 致命错误,需要完全降级
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* 错误恢复动作
|
|
22
|
+
*/
|
|
23
|
+
const RecoveryAction = {
|
|
24
|
+
RETRY: 'retry', // 重试操作
|
|
25
|
+
SKIP: 'skip', // 跳过当前步骤
|
|
26
|
+
FALLBACK: 'fallback', // 使用降级方案
|
|
27
|
+
ABORT: 'abort', // 中止流程
|
|
28
|
+
IGNORE: 'ignore', // 忽略错误继续
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* 错误上下文
|
|
33
|
+
* @typedef {Object} ErrorContext
|
|
34
|
+
* @property {string} operation - 操作名称
|
|
35
|
+
* @property {string} component - 组件名称
|
|
36
|
+
* @property {Object} metadata - 额外元数据
|
|
37
|
+
* @property {number} attempt - 当前尝试次数
|
|
38
|
+
*/
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* 错误边界配置
|
|
42
|
+
* @typedef {Object} ErrorBoundaryConfig
|
|
43
|
+
* @property {string} name - 边界名称
|
|
44
|
+
* @property {Function} [onError] - 全局错误处理器
|
|
45
|
+
* @property {Function} [onRecovery] - 恢复处理器
|
|
46
|
+
* @property {Function} [fallback] - 降级函数
|
|
47
|
+
* @property {boolean} [logErrors=true] - 是否记录错误
|
|
48
|
+
* @property {boolean} [propagateErrors=false] - 是否向上传播错误
|
|
49
|
+
*/
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* 错误边界类
|
|
53
|
+
* 用于捕获和管理错误,提供恢复机制
|
|
54
|
+
*/
|
|
55
|
+
class ErrorBoundary extends EventEmitter {
|
|
56
|
+
/**
|
|
57
|
+
* @param {ErrorBoundaryConfig} config
|
|
58
|
+
*/
|
|
59
|
+
constructor(config = {}) {
|
|
60
|
+
super();
|
|
61
|
+
|
|
62
|
+
this.name = config.name || 'AnonymousBoundary';
|
|
63
|
+
this._onError = config.onError || null;
|
|
64
|
+
this._onRecovery = config.onRecovery || null;
|
|
65
|
+
this._fallback = config.fallback || null;
|
|
66
|
+
this._logErrors = config.logErrors !== false;
|
|
67
|
+
this._propagateErrors = config.propagateErrors || false;
|
|
68
|
+
this._logger = logger.child('ErrorBoundary');
|
|
69
|
+
|
|
70
|
+
// 错误统计
|
|
71
|
+
this._errorCount = 0;
|
|
72
|
+
this._lastError = null;
|
|
73
|
+
this._errorHistory = [];
|
|
74
|
+
this._maxHistorySize = config.maxHistorySize || 50;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* 执行可能抛出错误的操作
|
|
79
|
+
* @param {Function} operation - 要执行的操作
|
|
80
|
+
* @param {Object} [context] - 错误上下文
|
|
81
|
+
* @returns {Promise<{success: boolean, result?: any, error?: Error, recovered?: boolean}>}
|
|
82
|
+
*/
|
|
83
|
+
async execute(operation, context = {}) {
|
|
84
|
+
const operationName = context.operation || 'anonymous';
|
|
85
|
+
|
|
86
|
+
try {
|
|
87
|
+
const result = await (typeof operation === 'function' ? operation() : operation);
|
|
88
|
+
return { success: true, result };
|
|
89
|
+
} catch (error) {
|
|
90
|
+
this._errorCount++;
|
|
91
|
+
this._lastError = error;
|
|
92
|
+
|
|
93
|
+
// 记录到历史
|
|
94
|
+
this._addToHistory(error, context);
|
|
95
|
+
|
|
96
|
+
// 记录错误日志
|
|
97
|
+
if (this._logErrors) {
|
|
98
|
+
this._logError(error, context);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// 触发错误事件
|
|
102
|
+
this.emit('error', {
|
|
103
|
+
error,
|
|
104
|
+
context,
|
|
105
|
+
boundary: this.name,
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
// 调用全局错误处理器
|
|
109
|
+
let recoveryAction = RecoveryAction.ABORT;
|
|
110
|
+
if (this._onError) {
|
|
111
|
+
try {
|
|
112
|
+
// 使用 Promise.resolve 以支持 async 函数
|
|
113
|
+
recoveryAction = await Promise.resolve(this._onError(error, context, this));
|
|
114
|
+
} catch (handlerError) {
|
|
115
|
+
this._logger.warn(`Error in onError handler: ${handlerError.message}`);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// 根据恢复动作处理
|
|
120
|
+
const { action, fallbackResult } = await this._handleRecovery(error, context, recoveryAction);
|
|
121
|
+
|
|
122
|
+
// 触发恢复事件
|
|
123
|
+
if (action !== RecoveryAction.ABORT) {
|
|
124
|
+
this.emit('recovery', {
|
|
125
|
+
error,
|
|
126
|
+
context,
|
|
127
|
+
action,
|
|
128
|
+
boundary: this.name,
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
if (this._onRecovery) {
|
|
132
|
+
try {
|
|
133
|
+
await this._onRecovery(error, context, action, this);
|
|
134
|
+
} catch (recoveryError) {
|
|
135
|
+
log.warn(`Error in onRecovery handler: ${recoveryError.message}`);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// 如果需要传播错误,或者没有降级方案且要求传播,则抛出错误
|
|
141
|
+
if (this._propagateErrors || (action === RecoveryAction.ABORT && !fallbackResult)) {
|
|
142
|
+
throw error;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
return {
|
|
146
|
+
success: false,
|
|
147
|
+
error,
|
|
148
|
+
recovered: action !== RecoveryAction.ABORT,
|
|
149
|
+
fallbackResult,
|
|
150
|
+
action,
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* 执行带重试的操作
|
|
157
|
+
* @param {Function} operation - 要执行的操作
|
|
158
|
+
* @param {Object} retryConfig - 重试配置
|
|
159
|
+
* @param {Object} [context] - 错误上下文
|
|
160
|
+
* @returns {Promise<any>}
|
|
161
|
+
*/
|
|
162
|
+
async executeWithRetry(operation, retryConfig = {}, context = {}) {
|
|
163
|
+
const { maxAttempts = 3, baseDelay = 1000, shouldRetry = () => true } = retryConfig;
|
|
164
|
+
|
|
165
|
+
let lastError;
|
|
166
|
+
|
|
167
|
+
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
|
|
168
|
+
try {
|
|
169
|
+
return await this.execute(operation, { ...context, attempt });
|
|
170
|
+
} catch (error) {
|
|
171
|
+
lastError = error;
|
|
172
|
+
|
|
173
|
+
if (attempt >= maxAttempts || !shouldRetry(error, attempt)) {
|
|
174
|
+
throw error;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// 计算延迟
|
|
178
|
+
const delay = baseDelay * Math.pow(2, attempt - 1);
|
|
179
|
+
log.debug(`Retry attempt ${attempt}/${maxAttempts} after ${delay}ms...`);
|
|
180
|
+
|
|
181
|
+
await new Promise((resolve) => setTimeout(resolve, delay));
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
throw lastError;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
/**
|
|
189
|
+
* 包装一个函数,使其错误被边界捕获
|
|
190
|
+
* @param {Function} fn - 要包装的函数
|
|
191
|
+
* @param {Object} [defaultContext] - 默认上下文
|
|
192
|
+
* @returns {Function}
|
|
193
|
+
*/
|
|
194
|
+
wrap(fn, defaultContext = {}) {
|
|
195
|
+
return async (...args) => {
|
|
196
|
+
const result = await this.execute(() => fn(...args), defaultContext);
|
|
197
|
+
return result.success ? result.result : result.fallbackResult;
|
|
198
|
+
};
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* 处理错误恢复
|
|
203
|
+
* @private
|
|
204
|
+
*/
|
|
205
|
+
async _handleRecovery(error, context, recoveryAction) {
|
|
206
|
+
let fallbackResult = null;
|
|
207
|
+
let action = recoveryAction;
|
|
208
|
+
|
|
209
|
+
switch (recoveryAction) {
|
|
210
|
+
case RecoveryAction.FALLBACK:
|
|
211
|
+
if (this._fallback) {
|
|
212
|
+
try {
|
|
213
|
+
fallbackResult = await this._fallback(error, context, this);
|
|
214
|
+
action = RecoveryAction.RETRY; // fallback 成功后视为恢复
|
|
215
|
+
} catch (fallbackError) {
|
|
216
|
+
log.warn(`Fallback also failed: ${fallbackError.message}`);
|
|
217
|
+
action = RecoveryAction.ABORT;
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
break;
|
|
221
|
+
|
|
222
|
+
case RecoveryAction.SKIP:
|
|
223
|
+
log.debug(`Skipping operation: ${context.operation}`);
|
|
224
|
+
fallbackResult = null;
|
|
225
|
+
break;
|
|
226
|
+
|
|
227
|
+
case RecoveryAction.IGNORE:
|
|
228
|
+
log.debug(`Ignoring error: ${error.message}`);
|
|
229
|
+
fallbackResult = undefined;
|
|
230
|
+
break;
|
|
231
|
+
|
|
232
|
+
default:
|
|
233
|
+
action = RecoveryAction.ABORT;
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
return { action, fallbackResult };
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
/**
|
|
240
|
+
* 记录错误
|
|
241
|
+
* @private
|
|
242
|
+
*/
|
|
243
|
+
_logError(error, context) {
|
|
244
|
+
if (!this._logErrors) return;
|
|
245
|
+
|
|
246
|
+
const errorInfo = safeErrorInfo(error);
|
|
247
|
+
const severity = this._determineSeverity(error);
|
|
248
|
+
|
|
249
|
+
const logMessage = `[${this.name}] Error in ${context.operation || 'unknown'}: ${error.message}`;
|
|
250
|
+
|
|
251
|
+
switch (severity) {
|
|
252
|
+
case Severity.LOW:
|
|
253
|
+
this._logger.debug(logMessage);
|
|
254
|
+
break;
|
|
255
|
+
case Severity.MEDIUM:
|
|
256
|
+
this._logger.warn(logMessage);
|
|
257
|
+
break;
|
|
258
|
+
case Severity.HIGH:
|
|
259
|
+
this._logger.error(logMessage);
|
|
260
|
+
break;
|
|
261
|
+
case Severity.CRITICAL:
|
|
262
|
+
this._logger.error(`CRITICAL: ${logMessage}`, { stack: error.stack });
|
|
263
|
+
break;
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
/**
|
|
268
|
+
* 确定错误严重级别
|
|
269
|
+
* @private
|
|
270
|
+
*/
|
|
271
|
+
_determineSeverity(error) {
|
|
272
|
+
if (isErrorOfType(error, require('./error').FolikoError)) {
|
|
273
|
+
return error.context?.severity || Severity.MEDIUM;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// 根据错误类型和消息判断
|
|
277
|
+
const message = (error.message || '').toLowerCase();
|
|
278
|
+
|
|
279
|
+
if (message.includes('timeout') || message.includes('network')) {
|
|
280
|
+
return Severity.MEDIUM;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
if (message.includes('memory') || message.includes('fatal')) {
|
|
284
|
+
return Severity.CRITICAL;
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
return Severity.HIGH;
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
/**
|
|
291
|
+
* 添加错误到历史记录
|
|
292
|
+
* @private
|
|
293
|
+
*/
|
|
294
|
+
_addToHistory(error, context) {
|
|
295
|
+
this._errorHistory.push({
|
|
296
|
+
timestamp: new Date().toISOString(),
|
|
297
|
+
error: safeErrorInfo(error),
|
|
298
|
+
context,
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
// 保持历史记录大小
|
|
302
|
+
if (this._errorHistory.length > this._maxHistorySize) {
|
|
303
|
+
this._errorHistory.shift();
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
/**
|
|
308
|
+
* 获取错误统计
|
|
309
|
+
* @returns {Object}
|
|
310
|
+
*/
|
|
311
|
+
getStats() {
|
|
312
|
+
return {
|
|
313
|
+
totalErrors: this._errorCount,
|
|
314
|
+
lastError: this._lastError ? safeErrorInfo(this._lastError) : null,
|
|
315
|
+
historySize: this._errorHistory.length,
|
|
316
|
+
recentErrors: this._errorHistory.slice(-10),
|
|
317
|
+
};
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
/**
|
|
321
|
+
* 清空错误历史
|
|
322
|
+
*/
|
|
323
|
+
clearHistory() {
|
|
324
|
+
this._errorHistory = [];
|
|
325
|
+
this._lastError = null;
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
/**
|
|
330
|
+
* 创建错误边界实例的工厂函数
|
|
331
|
+
* @param {ErrorBoundaryConfig} config
|
|
332
|
+
* @returns {ErrorBoundary}
|
|
333
|
+
*/
|
|
334
|
+
function createErrorBoundary(config = {}) {
|
|
335
|
+
return new ErrorBoundary(config);
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
/**
|
|
339
|
+
* 组合多个错误边界
|
|
340
|
+
* @param {ErrorBoundary[]} boundaries
|
|
341
|
+
* @returns {ErrorBoundary}
|
|
342
|
+
*/
|
|
343
|
+
function combineBoundaries(boundaries) {
|
|
344
|
+
const combined = new ErrorBoundary({
|
|
345
|
+
name: 'CombinedBoundary',
|
|
346
|
+
});
|
|
347
|
+
|
|
348
|
+
// 将错误传播到所有子边界
|
|
349
|
+
for (const boundary of boundaries) {
|
|
350
|
+
boundary.on('error', (data) => combined.emit('error', data));
|
|
351
|
+
boundary.on('recovery', (data) => combined.emit('recovery', data));
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
return combined;
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
module.exports = {
|
|
358
|
+
ErrorBoundary,
|
|
359
|
+
Severity,
|
|
360
|
+
RecoveryAction,
|
|
361
|
+
createErrorBoundary,
|
|
362
|
+
combineBoundaries,
|
|
363
|
+
};
|