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
package/src/index.js CHANGED
@@ -1,25 +1,25 @@
1
- /**
2
- * Foliko Framework
3
- * 简约的插件化 Agent 框架
4
- */
5
-
6
- const { Framework, Agent, Plugin, PluginManager, ToolRegistry, EventEmitter } = require('./core');
7
-
8
- const { SkillManagerPlugin, WorkflowPlugin } = require('./capabilities');
9
-
10
- const { MCPExecutorPlugin } = require('./executors/mcp-executor');
11
-
12
- module.exports = {
13
- // 核心
14
- Framework,
15
- Agent,
16
- Plugin,
17
- PluginManager,
18
- ToolRegistry,
19
- EventEmitter,
20
-
21
- // 能力插件
22
- SkillManagerPlugin,
23
- WorkflowPlugin,
24
- MCPExecutorPlugin,
25
- };
1
+ /**
2
+ * Foliko Framework
3
+ * 简约的插件化 Agent 框架
4
+ */
5
+
6
+ const { Framework, Agent, Plugin, PluginManager, ToolRegistry, EventEmitter } = require('./core');
7
+
8
+ const { SkillManagerPlugin, WorkflowPlugin } = require('./capabilities');
9
+
10
+ const { MCPExecutorPlugin } = require('./executors/mcp-executor');
11
+
12
+ module.exports = {
13
+ // 核心
14
+ Framework,
15
+ Agent,
16
+ Plugin,
17
+ PluginManager,
18
+ ToolRegistry,
19
+ EventEmitter,
20
+
21
+ // 能力插件
22
+ SkillManagerPlugin,
23
+ WorkflowPlugin,
24
+ MCPExecutorPlugin,
25
+ };
@@ -1,301 +1,301 @@
1
- /**
2
- * CircuitBreaker 熔断器
3
- * 用于防止工具调用失败时持续重试导致资源耗尽
4
- */
5
-
6
- const { logger } = require('./logger');
7
-
8
- /**
9
- * 熔断器状态
10
- */
11
- const CircuitState = {
12
- CLOSED: 'closed', // 正常状态,允许请求通过
13
- OPEN: 'open', // 熔断状态,拒绝请求
14
- HALF_OPEN: 'half_open', // 半开状态,允许一个测试请求
15
- };
16
-
17
- /**
18
- * 熔断器选项
19
- * @typedef {Object} CircuitBreakerOptions
20
- * @property {number} [failureThreshold=3] - 连续失败次数达到此值时触发熔断
21
- * @property {number} [successThreshold=2] - 半开状态下连续成功次数达到此值时关闭熔断
22
- * @property {number} [timeout=60000] - 熔断持续时间(毫秒),超时后进入半开状态
23
- * @property {string} [name] - 熔断器名称(用于日志)
24
- */
25
-
26
- /**
27
- * CircuitBreaker 熔断器类
28
- * 使用滑动窗口记录请求成功/失败,帮助防止持续失败的工具调用
29
- */
30
- class CircuitBreaker {
31
- /**
32
- * @param {CircuitBreakerOptions} options
33
- */
34
- constructor(options = {}) {
35
- this.options = {
36
- failureThreshold: options.failureThreshold || 3,
37
- successThreshold: options.successThreshold || 2,
38
- timeout: options.timeout || 60000, // 默认 1 分钟
39
- name: options.name || 'default',
40
- };
41
-
42
- this._state = CircuitState.CLOSED;
43
- this._failureCount = 0;
44
- this._successCount = 0;
45
- this._lastFailureTime = null;
46
- this._halfOpenAttempts = 0;
47
-
48
- this._log = logger.child('CircuitBreaker', { name: this.options.name });
49
- }
50
-
51
- /**
52
- * 获取当前状态
53
- * @returns {string}
54
- */
55
- getState() {
56
- // 检查是否需要从 OPEN 转为 HALF_OPEN
57
- if (this._state === CircuitState.OPEN && this._lastFailureTime) {
58
- const elapsed = Date.now() - this._lastFailureTime;
59
- if (elapsed >= this.options.timeout) {
60
- this._transitionTo(CircuitState.HALF_OPEN);
61
- }
62
- }
63
- return this._state;
64
- }
65
-
66
- /**
67
- * 检查是否允许执行
68
- * @returns {boolean}
69
- */
70
- canExecute() {
71
- const state = this.getState();
72
- return state === CircuitState.CLOSED || state === CircuitState.HALF_OPEN;
73
- }
74
-
75
- /**
76
- * 执行带熔断保护的函数
77
- * @param {Function} fn - 要执行的异步函数
78
- * @returns {Promise<any>}
79
- * @throws {Error} 如果熔断器处于 OPEN 状态,抛出错误
80
- */
81
- async execute(fn) {
82
- if (!this.canExecute()) {
83
- throw new Error(
84
- `Circuit breaker is OPEN for '${this.options.name}'. ` +
85
- `Tool calls are temporarily blocked. Try again later.`
86
- );
87
- }
88
-
89
- try {
90
- const result = await fn();
91
- this._onSuccess();
92
- return result;
93
- } catch (err) {
94
- this._onFailure(err);
95
- throw err;
96
- }
97
- }
98
-
99
- /**
100
- * 记录成功
101
- * @private
102
- */
103
- _onSuccess() {
104
- if (this._state === CircuitState.HALF_OPEN) {
105
- this._successCount++;
106
- this._halfOpenAttempts++;
107
- this._log.debug(`Half-open success: ${this._successCount}/${this.options.successThreshold}`);
108
-
109
- if (this._successCount >= this.options.successThreshold) {
110
- this._transitionTo(CircuitState.CLOSED);
111
- }
112
- } else if (this._state === CircuitState.CLOSED) {
113
- // 成功后重置失败计数
114
- this._failureCount = 0;
115
- }
116
- }
117
-
118
- /**
119
- * 记录失败
120
- * @param {Error} err
121
- * @private
122
- */
123
- _onFailure(err) {
124
- this._failureCount++;
125
- this._lastFailureTime = Date.now();
126
-
127
- if (this._state === CircuitState.HALF_OPEN) {
128
- // 半开状态下任何失败都立即打开熔断
129
- this._log.warn(`Half-open failure, reopening circuit: ${err.message}`);
130
- this._transitionTo(CircuitState.OPEN);
131
- } else if (this._state === CircuitState.CLOSED) {
132
- this._log.warn(
133
- `Tool call failed (${this._failureCount}/${this.options.failureThreshold}): ${err.message}`
134
- );
135
-
136
- if (this._failureCount >= this.options.failureThreshold) {
137
- this._log.error(
138
- `Circuit breaker triggered for '${this.options.name}' after ${this._failureCount} consecutive failures`
139
- );
140
- this._transitionTo(CircuitState.OPEN);
141
- }
142
- }
143
- }
144
-
145
- /**
146
- * 状态转换
147
- * @param {string} newState
148
- * @private
149
- */
150
- _transitionTo(newState) {
151
- const oldState = this._state;
152
- this._state = newState;
153
-
154
- if (newState === CircuitState.CLOSED) {
155
- this._failureCount = 0;
156
- this._successCount = 0;
157
- this._halfOpenAttempts = 0;
158
- this._log.info(`Circuit breaker CLOSED (reset)`);
159
- } else if (newState === CircuitState.HALF_OPEN) {
160
- this._successCount = 0;
161
- this._halfOpenAttempts = 0;
162
- this._log.info(`Circuit breaker HALF_OPEN (testing after ${this.options.timeout}ms timeout)`);
163
- } else if (newState === CircuitState.OPEN) {
164
- this._successCount = 0;
165
- this._halfOpenAttempts = 0;
166
- this._log.warn(`Circuit breaker OPEN (blocking tool calls for ${this.options.timeout}ms)`);
167
- }
168
- }
169
-
170
- /**
171
- * 手动重置熔断器
172
- */
173
- reset() {
174
- this._transitionTo(CircuitState.CLOSED);
175
- }
176
-
177
- /**
178
- * 获取熔断器状态信息
179
- * @returns {Object}
180
- */
181
- getStatus() {
182
- return {
183
- name: this.options.name,
184
- state: this.getState(),
185
- failureCount: this._failureCount,
186
- successCount: this._successCount,
187
- lastFailureTime: this._lastFailureTime,
188
- nextRetryTime:
189
- this._state === CircuitState.OPEN && this._lastFailureTime
190
- ? new Date(this._lastFailureTime + this.options.timeout).toISOString()
191
- : null,
192
- config: {
193
- failureThreshold: this.options.failureThreshold,
194
- successThreshold: this.options.successThreshold,
195
- timeout: this.options.timeout,
196
- },
197
- };
198
- }
199
- }
200
-
201
- /**
202
- * CircuitBreakerRegistry - 熔断器注册表
203
- * 管理多个工具的熔断器
204
- */
205
- class CircuitBreakerRegistry {
206
- constructor() {
207
- this._breakers = new Map();
208
- this._log = logger.child('CircuitBreakerRegistry');
209
- }
210
-
211
- /**
212
- * 获取或创建熔断器
213
- * @param {string} name - 熔断器名称(通常为工具名)
214
- * @param {CircuitBreakerOptions} options - 选项
215
- * @returns {CircuitBreaker}
216
- */
217
- getOrCreate(name, options = {}) {
218
- if (!this._breakers.has(name)) {
219
- this._breakers.set(name, new CircuitBreaker({ ...options, name }));
220
- }
221
- return this._breakers.get(name);
222
- }
223
-
224
- /**
225
- * 获取熔断器(如果不存在返回 null)
226
- * @param {string} name
227
- * @returns {CircuitBreaker|null}
228
- */
229
- get(name) {
230
- return this._breakers.get(name) || null;
231
- }
232
-
233
- /**
234
- * 检查是否允许执行
235
- * @param {string} name
236
- * @returns {boolean}
237
- */
238
- canExecute(name) {
239
- const breaker = this._breakers.get(name);
240
- return breaker ? breaker.canExecute() : true;
241
- }
242
-
243
- /**
244
- * 执行带熔断保护的函数
245
- * @param {string} name - 熔断器名称
246
- * @param {Function} fn - 要执行的函数
247
- * @param {CircuitBreakerOptions} options - 选项(首次创建时使用)
248
- * @returns {Promise<any>}
249
- */
250
- async execute(name, fn, options = {}) {
251
- const breaker = this.getOrCreate(name, options);
252
- return breaker.execute(fn);
253
- }
254
-
255
- /**
256
- * 获取所有熔断器状态
257
- * @returns {Object[]}
258
- */
259
- getAllStatus() {
260
- const result = [];
261
- for (const [name, breaker] of this._breakers) {
262
- result.push(breaker.getStatus());
263
- }
264
- return result;
265
- }
266
-
267
- /**
268
- * 重置所有熔断器
269
- */
270
- resetAll() {
271
- for (const breaker of this._breakers.values()) {
272
- breaker.reset();
273
- }
274
- this._log.info('All circuit breakers reset');
275
- }
276
-
277
- /**
278
- * 移除熔断器
279
- * @param {string} name
280
- */
281
- remove(name) {
282
- this._breakers.delete(name);
283
- }
284
-
285
- /**
286
- * 清空所有熔断器
287
- */
288
- clear() {
289
- this._breakers.clear();
290
- }
291
- }
292
-
293
- // 导出单例
294
- const globalRegistry = new CircuitBreakerRegistry();
295
-
296
- module.exports = {
297
- CircuitBreaker,
298
- CircuitBreakerRegistry,
299
- CircuitState,
300
- globalCircuitBreaker: globalRegistry,
301
- };
1
+ /**
2
+ * CircuitBreaker 熔断器
3
+ * 用于防止工具调用失败时持续重试导致资源耗尽
4
+ */
5
+
6
+ const { logger } = require('./logger');
7
+
8
+ /**
9
+ * 熔断器状态
10
+ */
11
+ const CircuitState = {
12
+ CLOSED: 'closed', // 正常状态,允许请求通过
13
+ OPEN: 'open', // 熔断状态,拒绝请求
14
+ HALF_OPEN: 'half_open', // 半开状态,允许一个测试请求
15
+ };
16
+
17
+ /**
18
+ * 熔断器选项
19
+ * @typedef {Object} CircuitBreakerOptions
20
+ * @property {number} [failureThreshold=3] - 连续失败次数达到此值时触发熔断
21
+ * @property {number} [successThreshold=2] - 半开状态下连续成功次数达到此值时关闭熔断
22
+ * @property {number} [timeout=60000] - 熔断持续时间(毫秒),超时后进入半开状态
23
+ * @property {string} [name] - 熔断器名称(用于日志)
24
+ */
25
+
26
+ /**
27
+ * CircuitBreaker 熔断器类
28
+ * 使用滑动窗口记录请求成功/失败,帮助防止持续失败的工具调用
29
+ */
30
+ class CircuitBreaker {
31
+ /**
32
+ * @param {CircuitBreakerOptions} options
33
+ */
34
+ constructor(options = {}) {
35
+ this.options = {
36
+ failureThreshold: options.failureThreshold || 3,
37
+ successThreshold: options.successThreshold || 2,
38
+ timeout: options.timeout || 60000, // 默认 1 分钟
39
+ name: options.name || 'default',
40
+ };
41
+
42
+ this._state = CircuitState.CLOSED;
43
+ this._failureCount = 0;
44
+ this._successCount = 0;
45
+ this._lastFailureTime = null;
46
+ this._halfOpenAttempts = 0;
47
+
48
+ this._log = logger.child('CircuitBreaker', { name: this.options.name });
49
+ }
50
+
51
+ /**
52
+ * 获取当前状态
53
+ * @returns {string}
54
+ */
55
+ getState() {
56
+ // 检查是否需要从 OPEN 转为 HALF_OPEN
57
+ if (this._state === CircuitState.OPEN && this._lastFailureTime) {
58
+ const elapsed = Date.now() - this._lastFailureTime;
59
+ if (elapsed >= this.options.timeout) {
60
+ this._transitionTo(CircuitState.HALF_OPEN);
61
+ }
62
+ }
63
+ return this._state;
64
+ }
65
+
66
+ /**
67
+ * 检查是否允许执行
68
+ * @returns {boolean}
69
+ */
70
+ canExecute() {
71
+ const state = this.getState();
72
+ return state === CircuitState.CLOSED || state === CircuitState.HALF_OPEN;
73
+ }
74
+
75
+ /**
76
+ * 执行带熔断保护的函数
77
+ * @param {Function} fn - 要执行的异步函数
78
+ * @returns {Promise<any>}
79
+ * @throws {Error} 如果熔断器处于 OPEN 状态,抛出错误
80
+ */
81
+ async execute(fn) {
82
+ if (!this.canExecute()) {
83
+ throw new Error(
84
+ `Circuit breaker is OPEN for '${this.options.name}'. ` +
85
+ `Tool calls are temporarily blocked. Try again later.`
86
+ );
87
+ }
88
+
89
+ try {
90
+ const result = await fn();
91
+ this._onSuccess();
92
+ return result;
93
+ } catch (err) {
94
+ this._onFailure(err);
95
+ throw err;
96
+ }
97
+ }
98
+
99
+ /**
100
+ * 记录成功
101
+ * @private
102
+ */
103
+ _onSuccess() {
104
+ if (this._state === CircuitState.HALF_OPEN) {
105
+ this._successCount++;
106
+ this._halfOpenAttempts++;
107
+ this._log.debug(`Half-open success: ${this._successCount}/${this.options.successThreshold}`);
108
+
109
+ if (this._successCount >= this.options.successThreshold) {
110
+ this._transitionTo(CircuitState.CLOSED);
111
+ }
112
+ } else if (this._state === CircuitState.CLOSED) {
113
+ // 成功后重置失败计数
114
+ this._failureCount = 0;
115
+ }
116
+ }
117
+
118
+ /**
119
+ * 记录失败
120
+ * @param {Error} err
121
+ * @private
122
+ */
123
+ _onFailure(err) {
124
+ this._failureCount++;
125
+ this._lastFailureTime = Date.now();
126
+
127
+ if (this._state === CircuitState.HALF_OPEN) {
128
+ // 半开状态下任何失败都立即打开熔断
129
+ this._log.warn(`Half-open failure, reopening circuit: ${err.message}`);
130
+ this._transitionTo(CircuitState.OPEN);
131
+ } else if (this._state === CircuitState.CLOSED) {
132
+ this._log.warn(
133
+ `Tool call failed (${this._failureCount}/${this.options.failureThreshold}): ${err.message}`
134
+ );
135
+
136
+ if (this._failureCount >= this.options.failureThreshold) {
137
+ this._log.error(
138
+ `Circuit breaker triggered for '${this.options.name}' after ${this._failureCount} consecutive failures`
139
+ );
140
+ this._transitionTo(CircuitState.OPEN);
141
+ }
142
+ }
143
+ }
144
+
145
+ /**
146
+ * 状态转换
147
+ * @param {string} newState
148
+ * @private
149
+ */
150
+ _transitionTo(newState) {
151
+ const oldState = this._state;
152
+ this._state = newState;
153
+
154
+ if (newState === CircuitState.CLOSED) {
155
+ this._failureCount = 0;
156
+ this._successCount = 0;
157
+ this._halfOpenAttempts = 0;
158
+ this._log.info(`Circuit breaker CLOSED (reset)`);
159
+ } else if (newState === CircuitState.HALF_OPEN) {
160
+ this._successCount = 0;
161
+ this._halfOpenAttempts = 0;
162
+ this._log.info(`Circuit breaker HALF_OPEN (testing after ${this.options.timeout}ms timeout)`);
163
+ } else if (newState === CircuitState.OPEN) {
164
+ this._successCount = 0;
165
+ this._halfOpenAttempts = 0;
166
+ this._log.warn(`Circuit breaker OPEN (blocking tool calls for ${this.options.timeout}ms)`);
167
+ }
168
+ }
169
+
170
+ /**
171
+ * 手动重置熔断器
172
+ */
173
+ reset() {
174
+ this._transitionTo(CircuitState.CLOSED);
175
+ }
176
+
177
+ /**
178
+ * 获取熔断器状态信息
179
+ * @returns {Object}
180
+ */
181
+ getStatus() {
182
+ return {
183
+ name: this.options.name,
184
+ state: this.getState(),
185
+ failureCount: this._failureCount,
186
+ successCount: this._successCount,
187
+ lastFailureTime: this._lastFailureTime,
188
+ nextRetryTime:
189
+ this._state === CircuitState.OPEN && this._lastFailureTime
190
+ ? new Date(this._lastFailureTime + this.options.timeout).toISOString()
191
+ : null,
192
+ config: {
193
+ failureThreshold: this.options.failureThreshold,
194
+ successThreshold: this.options.successThreshold,
195
+ timeout: this.options.timeout,
196
+ },
197
+ };
198
+ }
199
+ }
200
+
201
+ /**
202
+ * CircuitBreakerRegistry - 熔断器注册表
203
+ * 管理多个工具的熔断器
204
+ */
205
+ class CircuitBreakerRegistry {
206
+ constructor() {
207
+ this._breakers = new Map();
208
+ this._log = logger.child('CircuitBreakerRegistry');
209
+ }
210
+
211
+ /**
212
+ * 获取或创建熔断器
213
+ * @param {string} name - 熔断器名称(通常为工具名)
214
+ * @param {CircuitBreakerOptions} options - 选项
215
+ * @returns {CircuitBreaker}
216
+ */
217
+ getOrCreate(name, options = {}) {
218
+ if (!this._breakers.has(name)) {
219
+ this._breakers.set(name, new CircuitBreaker({ ...options, name }));
220
+ }
221
+ return this._breakers.get(name);
222
+ }
223
+
224
+ /**
225
+ * 获取熔断器(如果不存在返回 null)
226
+ * @param {string} name
227
+ * @returns {CircuitBreaker|null}
228
+ */
229
+ get(name) {
230
+ return this._breakers.get(name) || null;
231
+ }
232
+
233
+ /**
234
+ * 检查是否允许执行
235
+ * @param {string} name
236
+ * @returns {boolean}
237
+ */
238
+ canExecute(name) {
239
+ const breaker = this._breakers.get(name);
240
+ return breaker ? breaker.canExecute() : true;
241
+ }
242
+
243
+ /**
244
+ * 执行带熔断保护的函数
245
+ * @param {string} name - 熔断器名称
246
+ * @param {Function} fn - 要执行的函数
247
+ * @param {CircuitBreakerOptions} options - 选项(首次创建时使用)
248
+ * @returns {Promise<any>}
249
+ */
250
+ async execute(name, fn, options = {}) {
251
+ const breaker = this.getOrCreate(name, options);
252
+ return breaker.execute(fn);
253
+ }
254
+
255
+ /**
256
+ * 获取所有熔断器状态
257
+ * @returns {Object[]}
258
+ */
259
+ getAllStatus() {
260
+ const result = [];
261
+ for (const [name, breaker] of this._breakers) {
262
+ result.push(breaker.getStatus());
263
+ }
264
+ return result;
265
+ }
266
+
267
+ /**
268
+ * 重置所有熔断器
269
+ */
270
+ resetAll() {
271
+ for (const breaker of this._breakers.values()) {
272
+ breaker.reset();
273
+ }
274
+ this._log.info('All circuit breakers reset');
275
+ }
276
+
277
+ /**
278
+ * 移除熔断器
279
+ * @param {string} name
280
+ */
281
+ remove(name) {
282
+ this._breakers.delete(name);
283
+ }
284
+
285
+ /**
286
+ * 清空所有熔断器
287
+ */
288
+ clear() {
289
+ this._breakers.clear();
290
+ }
291
+ }
292
+
293
+ // 导出单例
294
+ const globalRegistry = new CircuitBreakerRegistry();
295
+
296
+ module.exports = {
297
+ CircuitBreaker,
298
+ CircuitBreakerRegistry,
299
+ CircuitState,
300
+ globalCircuitBreaker: globalRegistry,
301
+ };