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
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
|
+
};
|