foliko 1.0.74 → 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 (238) hide show
  1. package/.agent/.shared/ui-ux-pro-max/data/charts.csv +26 -0
  2. package/.agent/.shared/ui-ux-pro-max/data/colors.csv +97 -0
  3. package/.agent/.shared/ui-ux-pro-max/data/icons.csv +101 -0
  4. package/.agent/.shared/ui-ux-pro-max/data/landing.csv +31 -0
  5. package/.agent/.shared/ui-ux-pro-max/data/products.csv +97 -0
  6. package/.agent/.shared/ui-ux-pro-max/data/prompts.csv +24 -0
  7. package/.agent/.shared/ui-ux-pro-max/data/react-performance.csv +45 -0
  8. package/.agent/.shared/ui-ux-pro-max/data/stacks/flutter.csv +53 -0
  9. package/.agent/.shared/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -0
  10. package/.agent/.shared/ui-ux-pro-max/data/stacks/jetpack-compose.csv +53 -0
  11. package/.agent/.shared/ui-ux-pro-max/data/stacks/nextjs.csv +53 -0
  12. package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -0
  13. package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -0
  14. package/.agent/.shared/ui-ux-pro-max/data/stacks/react-native.csv +52 -0
  15. package/.agent/.shared/ui-ux-pro-max/data/stacks/react.csv +54 -0
  16. package/.agent/.shared/ui-ux-pro-max/data/stacks/shadcn.csv +61 -0
  17. package/.agent/.shared/ui-ux-pro-max/data/stacks/svelte.csv +54 -0
  18. package/.agent/.shared/ui-ux-pro-max/data/stacks/swiftui.csv +51 -0
  19. package/.agent/.shared/ui-ux-pro-max/data/stacks/vue.csv +50 -0
  20. package/.agent/.shared/ui-ux-pro-max/data/styles.csv +59 -0
  21. package/.agent/.shared/ui-ux-pro-max/data/typography.csv +58 -0
  22. package/.agent/.shared/ui-ux-pro-max/data/ui-reasoning.csv +101 -0
  23. package/.agent/.shared/ui-ux-pro-max/data/ux-guidelines.csv +100 -0
  24. package/.agent/.shared/ui-ux-pro-max/data/web-interface.csv +31 -0
  25. package/.agent/.shared/ui-ux-pro-max/scripts/__pycache__/core.cpython-313.pyc +0 -0
  26. package/.agent/.shared/ui-ux-pro-max/scripts/__pycache__/design_system.cpython-313.pyc +0 -0
  27. package/.agent/.shared/ui-ux-pro-max/scripts/core.py +258 -0
  28. package/.agent/.shared/ui-ux-pro-max/scripts/design_system.py +1067 -0
  29. package/.agent/.shared/ui-ux-pro-max/scripts/search.py +106 -0
  30. package/.agent/ARCHITECTURE.md +288 -0
  31. package/.agent/agents/ambient-agent.md +57 -0
  32. package/.agent/agents/debugger.md +55 -0
  33. package/.agent/agents/email-assistant.md +49 -0
  34. package/.agent/agents/file-manager.md +42 -0
  35. package/.agent/agents/python-developer.md +60 -0
  36. package/.agent/agents/scheduler.md +59 -0
  37. package/.agent/agents/web-developer.md +45 -0
  38. package/.agent/data/default.json +29 -0
  39. package/.agent/data/plugins-state.json +255 -0
  40. package/.agent/mcp_config.json +4 -0
  41. package/.agent/mcp_config_updated.json +12 -0
  42. package/.agent/plugins.json +5 -0
  43. package/.agent/rules/GEMINI.md +273 -0
  44. package/.agent/rules/allow-rule.md +77 -0
  45. package/.agent/rules/log-rule.md +83 -0
  46. package/.agent/rules/security-rule.md +93 -0
  47. package/.agent/scripts/auto_preview.py +148 -0
  48. package/.agent/scripts/checklist.py +217 -0
  49. package/.agent/scripts/session_manager.py +120 -0
  50. package/.agent/scripts/verify_all.py +327 -0
  51. package/.agent/skills/api-patterns/SKILL.md +81 -0
  52. package/.agent/skills/api-patterns/api-style.md +42 -0
  53. package/.agent/skills/api-patterns/auth.md +24 -0
  54. package/.agent/skills/api-patterns/documentation.md +26 -0
  55. package/.agent/skills/api-patterns/graphql.md +41 -0
  56. package/.agent/skills/api-patterns/rate-limiting.md +31 -0
  57. package/.agent/skills/api-patterns/response.md +37 -0
  58. package/.agent/skills/api-patterns/rest.md +40 -0
  59. package/.agent/skills/api-patterns/scripts/api_validator.py +211 -0
  60. package/.agent/skills/api-patterns/security-testing.md +122 -0
  61. package/.agent/skills/api-patterns/trpc.md +41 -0
  62. package/.agent/skills/api-patterns/versioning.md +22 -0
  63. package/.agent/skills/app-builder/SKILL.md +75 -0
  64. package/.agent/skills/app-builder/agent-coordination.md +71 -0
  65. package/.agent/skills/app-builder/feature-building.md +53 -0
  66. package/.agent/skills/app-builder/project-detection.md +34 -0
  67. package/.agent/skills/app-builder/scaffolding.md +118 -0
  68. package/.agent/skills/app-builder/tech-stack.md +40 -0
  69. package/.agent/skills/app-builder/templates/SKILL.md +39 -0
  70. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -0
  71. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -0
  72. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -0
  73. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -0
  74. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -0
  75. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -0
  76. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -0
  77. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +122 -0
  78. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +122 -0
  79. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +169 -0
  80. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +134 -0
  81. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -0
  82. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +119 -0
  83. package/.agent/skills/architecture/SKILL.md +55 -0
  84. package/.agent/skills/architecture/context-discovery.md +43 -0
  85. package/.agent/skills/architecture/examples.md +94 -0
  86. package/.agent/skills/architecture/pattern-selection.md +68 -0
  87. package/.agent/skills/architecture/patterns-reference.md +50 -0
  88. package/.agent/skills/architecture/trade-off-analysis.md +77 -0
  89. package/.agent/skills/clean-code/SKILL.md +201 -0
  90. package/.agent/skills/doc.md +177 -0
  91. package/.agent/skills/frontend-design/SKILL.md +418 -0
  92. package/.agent/skills/frontend-design/animation-guide.md +331 -0
  93. package/.agent/skills/frontend-design/color-system.md +311 -0
  94. package/.agent/skills/frontend-design/decision-trees.md +418 -0
  95. package/.agent/skills/frontend-design/motion-graphics.md +306 -0
  96. package/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -0
  97. package/.agent/skills/frontend-design/scripts/ux_audit.py +722 -0
  98. package/.agent/skills/frontend-design/typography-system.md +345 -0
  99. package/.agent/skills/frontend-design/ux-psychology.md +1116 -0
  100. package/.agent/skills/frontend-design/visual-effects.md +383 -0
  101. package/.agent/skills/i18n-localization/SKILL.md +154 -0
  102. package/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -0
  103. package/.agent/skills/mcp-builder/SKILL.md +176 -0
  104. package/.agent/skills/web-design-guidelines/SKILL.md +57 -0
  105. package/.agent/workflows/brainstorm.md +113 -0
  106. package/.agent/workflows/create.md +59 -0
  107. package/.agent/workflows/debug.md +103 -0
  108. package/.agent/workflows/deploy.md +176 -0
  109. package/.agent/workflows/enhance.md +63 -0
  110. package/.agent/workflows/orchestrate.md +237 -0
  111. package/.agent/workflows/plan.md +89 -0
  112. package/.agent/workflows/preview.md +81 -0
  113. package/.agent/workflows/simple-test.md +42 -0
  114. package/.agent/workflows/status.md +86 -0
  115. package/.agent/workflows/structured-orchestrate.md +180 -0
  116. package/.agent/workflows/test.md +144 -0
  117. package/.agent/workflows/ui-ux-pro-max.md +296 -0
  118. package/.claude/settings.local.json +11 -1
  119. package/.editorconfig +56 -0
  120. package/.husky/pre-commit +4 -0
  121. package/.lintstagedrc +7 -0
  122. package/.prettierignore +29 -0
  123. package/.prettierrc +11 -0
  124. package/CLAUDE.md +2 -0
  125. package/README.md +64 -55
  126. package/SPEC.md +102 -61
  127. package/cli/bin/foliko.js +11 -11
  128. package/cli/src/commands/chat.js +143 -141
  129. package/cli/src/commands/list.js +93 -90
  130. package/cli/src/index.js +75 -75
  131. package/cli/src/ui/chat-ui.js +201 -199
  132. package/cli/src/utils/ansi.js +40 -40
  133. package/cli/src/utils/markdown.js +292 -296
  134. package/docker-compose.yml +1 -1
  135. package/docs/ai-sdk-optimization.md +655 -643
  136. package/docs/features.md +80 -80
  137. package/docs/quick-reference.md +49 -46
  138. package/docs/user-manual.md +411 -380
  139. package/examples/ambient-example.js +194 -196
  140. package/examples/basic.js +50 -45
  141. package/examples/bootstrap.js +121 -112
  142. package/examples/mcp-example.js +19 -16
  143. package/examples/skill-example.js +20 -20
  144. package/examples/test-chat.js +137 -135
  145. package/examples/test-mcp.js +85 -79
  146. package/examples/test-reload.js +59 -61
  147. package/examples/test-telegram.js +50 -50
  148. package/examples/test-tg-bot.js +45 -42
  149. package/examples/test-tg-simple.js +47 -46
  150. package/examples/test-tg.js +62 -62
  151. package/examples/test-think.js +43 -37
  152. package/examples/test-web-plugin.js +103 -98
  153. package/examples/test-weixin-feishu.js +103 -100
  154. package/examples/workflow.js +158 -158
  155. package/package.json +37 -3
  156. package/plugins/ai-plugin.js +102 -100
  157. package/plugins/ambient-agent/EventWatcher.js +113 -0
  158. package/plugins/ambient-agent/ExplorerLoop.js +640 -0
  159. package/plugins/ambient-agent/GoalManager.js +197 -0
  160. package/plugins/ambient-agent/Reflector.js +95 -0
  161. package/plugins/ambient-agent/StateStore.js +90 -0
  162. package/plugins/ambient-agent/constants.js +101 -0
  163. package/plugins/ambient-agent/index.js +579 -0
  164. package/plugins/audit-plugin.js +187 -187
  165. package/plugins/default-plugins.js +662 -649
  166. package/plugins/email/constants.js +64 -0
  167. package/plugins/email/handlers.js +461 -0
  168. package/plugins/email/index.js +278 -0
  169. package/plugins/email/monitor.js +269 -0
  170. package/plugins/email/parser.js +138 -0
  171. package/plugins/email/reply.js +151 -0
  172. package/plugins/email/utils.js +124 -0
  173. package/plugins/feishu-plugin.js +481 -477
  174. package/plugins/file-system-plugin.js +826 -476
  175. package/plugins/install-plugin.js +199 -197
  176. package/plugins/python-executor-plugin.js +367 -365
  177. package/plugins/python-plugin-loader.js +481 -479
  178. package/plugins/rules-plugin.js +294 -292
  179. package/plugins/scheduler-plugin.js +691 -689
  180. package/plugins/session-plugin.js +369 -367
  181. package/plugins/shell-executor-plugin.js +197 -197
  182. package/plugins/storage-plugin.js +240 -238
  183. package/plugins/subagent-plugin.js +845 -785
  184. package/plugins/telegram-plugin.js +482 -475
  185. package/plugins/think-plugin.js +345 -343
  186. package/plugins/tools-plugin.js +196 -194
  187. package/plugins/web-plugin.js +606 -604
  188. package/plugins/weixin-plugin.js +545 -538
  189. package/reports/system-health-report-20260401.md +79 -0
  190. package/skills/ambient-agent/SKILL.md +49 -39
  191. package/skills/foliko-dev/AGENTS.md +64 -61
  192. package/skills/foliko-dev/SKILL.md +125 -119
  193. package/skills/mcp-usage/SKILL.md +19 -17
  194. package/skills/python-plugin-dev/SKILL.md +16 -15
  195. package/skills/skill-guide/SKILL.md +12 -12
  196. package/skills/subagent-guide/SKILL.md +237 -0
  197. package/skills/workflow-guide/SKILL.md +90 -45
  198. package/skills/workflow-troubleshooting/DEBUGGING.md +36 -21
  199. package/skills/workflow-troubleshooting/SKILL.md +156 -79
  200. package/src/capabilities/index.js +11 -11
  201. package/src/capabilities/skill-manager.js +609 -595
  202. package/src/capabilities/workflow-engine.js +1109 -1195
  203. package/src/core/agent-chat.js +882 -735
  204. package/src/core/agent.js +892 -688
  205. package/src/core/framework.js +465 -431
  206. package/src/core/index.js +19 -19
  207. package/src/core/plugin-base.js +219 -219
  208. package/src/core/plugin-manager.js +863 -767
  209. package/src/core/provider.js +114 -111
  210. package/src/core/sub-agent-config.js +264 -0
  211. package/src/core/system-prompt-builder.js +120 -0
  212. package/src/core/tool-registry.js +517 -134
  213. package/src/core/tool-router.js +297 -216
  214. package/src/executors/executor-base.js +12 -12
  215. package/src/executors/mcp-executor.js +741 -729
  216. package/src/index.js +25 -37
  217. package/src/utils/circuit-breaker.js +301 -0
  218. package/src/utils/error-boundary.js +363 -0
  219. package/src/utils/error.js +374 -0
  220. package/src/utils/event-emitter.js +97 -97
  221. package/src/utils/id.js +133 -0
  222. package/src/utils/index.js +217 -3
  223. package/src/utils/logger.js +181 -0
  224. package/src/utils/plugin-helpers.js +90 -0
  225. package/src/utils/retry.js +122 -0
  226. package/src/utils/sandbox.js +292 -0
  227. package/test/tool-registry-validation.test.js +218 -0
  228. package/test_report.md +70 -0
  229. package/website/docs/api.html +169 -107
  230. package/website/docs/configuration.html +296 -144
  231. package/website/docs/plugin-development.html +154 -85
  232. package/website/docs/project-structure.html +110 -109
  233. package/website/docs/skill-development.html +117 -61
  234. package/website/index.html +209 -205
  235. package/website/script.js +136 -133
  236. package/website/styles.css +1 -1
  237. package/plugins/ambient-agent-plugin.js +0 -1565
  238. package/plugins/email.js +0 -1142
package/src/index.js CHANGED
@@ -1,37 +1,25 @@
1
- /**
2
- * Foliko Framework
3
- * 简约的插件化 Agent 框架
4
- */
5
-
6
- const {
7
- Framework,
8
- Agent,
9
- Plugin,
10
- PluginManager,
11
- ToolRegistry,
12
- EventEmitter
13
- } = require('./core')
14
-
15
- const {
16
- SkillManagerPlugin,
17
- WorkflowPlugin
18
- } = require('./capabilities')
19
-
20
- const {
21
- MCPExecutorPlugin
22
- } = require('./executors/mcp-executor')
23
-
24
- module.exports = {
25
- // 核心
26
- Framework,
27
- Agent,
28
- Plugin,
29
- PluginManager,
30
- ToolRegistry,
31
- EventEmitter,
32
-
33
- // 能力插件
34
- SkillManagerPlugin,
35
- WorkflowPlugin,
36
- MCPExecutorPlugin
37
- }
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
+ };
@@ -0,0 +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
+ };