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.
Files changed (88) hide show
  1. package/.claude/settings.local.json +159 -157
  2. package/cli/bin/foliko.js +12 -12
  3. package/cli/src/commands/chat.js +143 -143
  4. package/cli/src/commands/list.js +93 -93
  5. package/cli/src/index.js +75 -75
  6. package/cli/src/ui/chat-ui.js +201 -201
  7. package/cli/src/utils/ansi.js +40 -40
  8. package/cli/src/utils/markdown.js +292 -292
  9. package/examples/ambient-example.js +194 -194
  10. package/examples/basic.js +115 -115
  11. package/examples/bootstrap.js +121 -121
  12. package/examples/mcp-example.js +56 -56
  13. package/examples/skill-example.js +49 -49
  14. package/examples/test-chat.js +137 -137
  15. package/examples/test-mcp.js +85 -85
  16. package/examples/test-reload.js +59 -59
  17. package/examples/test-telegram.js +50 -50
  18. package/examples/test-tg-bot.js +45 -45
  19. package/examples/test-tg-simple.js +47 -47
  20. package/examples/test-tg.js +62 -62
  21. package/examples/test-think.js +43 -43
  22. package/examples/test-web-plugin.js +103 -103
  23. package/examples/test-weixin-feishu.js +103 -103
  24. package/examples/workflow.js +158 -158
  25. package/package.json +1 -1
  26. package/plugins/ai-plugin.js +102 -102
  27. package/plugins/ambient-agent/EventWatcher.js +113 -113
  28. package/plugins/ambient-agent/ExplorerLoop.js +640 -640
  29. package/plugins/ambient-agent/GoalManager.js +197 -197
  30. package/plugins/ambient-agent/Reflector.js +95 -95
  31. package/plugins/ambient-agent/StateStore.js +90 -90
  32. package/plugins/ambient-agent/constants.js +101 -101
  33. package/plugins/ambient-agent/index.js +579 -579
  34. package/plugins/audit-plugin.js +187 -187
  35. package/plugins/default-plugins.js +662 -662
  36. package/plugins/email/constants.js +64 -64
  37. package/plugins/email/handlers.js +461 -461
  38. package/plugins/email/index.js +278 -278
  39. package/plugins/email/monitor.js +269 -269
  40. package/plugins/email/parser.js +138 -138
  41. package/plugins/email/reply.js +151 -151
  42. package/plugins/email/utils.js +124 -124
  43. package/plugins/feishu-plugin.js +481 -481
  44. package/plugins/file-system-plugin.js +826 -826
  45. package/plugins/install-plugin.js +199 -199
  46. package/plugins/python-executor-plugin.js +367 -367
  47. package/plugins/python-plugin-loader.js +481 -481
  48. package/plugins/rules-plugin.js +294 -294
  49. package/plugins/scheduler-plugin.js +691 -691
  50. package/plugins/session-plugin.js +369 -369
  51. package/plugins/shell-executor-plugin.js +197 -197
  52. package/plugins/storage-plugin.js +240 -240
  53. package/plugins/subagent-plugin.js +845 -845
  54. package/plugins/telegram-plugin.js +482 -482
  55. package/plugins/think-plugin.js +345 -345
  56. package/plugins/tools-plugin.js +196 -196
  57. package/plugins/web-plugin.js +606 -606
  58. package/plugins/weixin-plugin.js +545 -545
  59. package/src/capabilities/index.js +11 -11
  60. package/src/capabilities/skill-manager.js +609 -609
  61. package/src/capabilities/workflow-engine.js +1109 -1109
  62. package/src/core/agent-chat.js +882 -882
  63. package/src/core/agent.js +892 -892
  64. package/src/core/framework.js +465 -465
  65. package/src/core/index.js +19 -19
  66. package/src/core/plugin-base.js +219 -219
  67. package/src/core/plugin-manager.js +863 -863
  68. package/src/core/provider.js +114 -114
  69. package/src/core/sub-agent-config.js +264 -264
  70. package/src/core/system-prompt-builder.js +120 -120
  71. package/src/core/tool-registry.js +517 -517
  72. package/src/core/tool-router.js +297 -297
  73. package/src/executors/executor-base.js +58 -58
  74. package/src/executors/mcp-executor.js +741 -741
  75. package/src/index.js +25 -25
  76. package/src/utils/circuit-breaker.js +301 -301
  77. package/src/utils/error-boundary.js +363 -363
  78. package/src/utils/error.js +374 -374
  79. package/src/utils/event-emitter.js +97 -97
  80. package/src/utils/id.js +133 -133
  81. package/src/utils/index.js +217 -217
  82. package/src/utils/logger.js +181 -181
  83. package/src/utils/plugin-helpers.js +90 -90
  84. package/src/utils/retry.js +122 -122
  85. package/src/utils/sandbox.js +292 -292
  86. package/test/tool-registry-validation.test.js +218 -218
  87. package/website/script.js +136 -136
  88. 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
+ };