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
@@ -0,0 +1,374 @@
1
+ /**
2
+ * Foliko Error - 统一错误处理系统
3
+ * 标准化项目中的所有错误类型
4
+ */
5
+
6
+ /**
7
+ * Foliko 错误基类
8
+ */
9
+ class FolikoError extends Error {
10
+ /**
11
+ * @param {string} message - 错误消息
12
+ * @param {Object} [options]
13
+ * @param {string} [options.code] - 错误代码
14
+ * @param {Object} [options.context] - 错误上下文
15
+ * @param {string} [options.stack] - 自定义堆栈
16
+ */
17
+ constructor(message, options = {}) {
18
+ super(message);
19
+ this.name = this.constructor.name;
20
+ this.code = options.code || 'FOLIKO_ERROR';
21
+ this.context = options.context || {};
22
+
23
+ // 保留原始堆栈跟踪
24
+ if (options.stack) {
25
+ this.stack = options.stack;
26
+ } else {
27
+ Error.captureStackTrace(this, this.constructor);
28
+ }
29
+ }
30
+
31
+ /**
32
+ * 将错误转换为 JSON
33
+ * @returns {Object}
34
+ */
35
+ toJSON() {
36
+ return {
37
+ name: this.name,
38
+ message: this.message,
39
+ code: this.code,
40
+ context: this.context,
41
+ stack: this.stack,
42
+ };
43
+ }
44
+ }
45
+
46
+ /**
47
+ * 插件相关错误
48
+ */
49
+ class PluginError extends FolikoError {
50
+ constructor(message, options = {}) {
51
+ super(message, {
52
+ code: 'PLUGIN_ERROR',
53
+ context: options.context,
54
+ });
55
+ }
56
+ }
57
+
58
+ /**
59
+ * 工具执行错误
60
+ */
61
+ class ToolError extends FolikoError {
62
+ constructor(message, options = {}) {
63
+ super(message, {
64
+ code: 'TOOL_ERROR',
65
+ context: {
66
+ toolName: options.toolName,
67
+ args: options.args,
68
+ ...options.context,
69
+ },
70
+ });
71
+ }
72
+ }
73
+
74
+ /**
75
+ * AI 执行错误
76
+ */
77
+ class AIError extends FolikoError {
78
+ constructor(message, options = {}) {
79
+ super(message, {
80
+ code: 'AI_ERROR',
81
+ context: {
82
+ model: options.model,
83
+ provider: options.provider,
84
+ statusCode: options.statusCode,
85
+ ...options.context,
86
+ },
87
+ });
88
+ }
89
+ }
90
+
91
+ /**
92
+ * 会话相关错误
93
+ */
94
+ class SessionError extends FolikoError {
95
+ constructor(message, options = {}) {
96
+ super(message, {
97
+ code: 'SESSION_ERROR',
98
+ context: {
99
+ sessionId: options.sessionId,
100
+ ...options.context,
101
+ },
102
+ });
103
+ }
104
+ }
105
+
106
+ /**
107
+ * 配置错误
108
+ */
109
+ class ConfigError extends FolikoError {
110
+ constructor(message, options = {}) {
111
+ super(message, {
112
+ code: 'CONFIG_ERROR',
113
+ context: {
114
+ configKey: options.configKey,
115
+ expected: options.expected,
116
+ actual: options.actual,
117
+ ...options.context,
118
+ },
119
+ });
120
+ }
121
+ }
122
+
123
+ /**
124
+ * 工作流执行错误
125
+ */
126
+ class WorkflowError extends FolikoError {
127
+ constructor(message, options = {}) {
128
+ super(message, {
129
+ code: 'WORKFLOW_ERROR',
130
+ context: {
131
+ workflowName: options.workflowName,
132
+ stepName: options.stepName,
133
+ stepIndex: options.stepIndex,
134
+ ...options.context,
135
+ },
136
+ });
137
+ }
138
+ }
139
+
140
+ /**
141
+ * 验证错误
142
+ */
143
+ class ValidationError extends FolikoError {
144
+ constructor(message, options = {}) {
145
+ super(message, {
146
+ code: 'VALIDATION_ERROR',
147
+ context: {
148
+ field: options.field,
149
+ value: options.value,
150
+ expected: options.expected,
151
+ ...options.context,
152
+ },
153
+ });
154
+ }
155
+ }
156
+
157
+ /**
158
+ * 工具未找到错误
159
+ */
160
+ class ToolNotFoundError extends ToolError {
161
+ constructor(toolName) {
162
+ super(`Tool not found: ${toolName}`, { toolName });
163
+ this.code = 'TOOL_NOT_FOUND';
164
+ }
165
+ }
166
+
167
+ /**
168
+ * 插件未找到错误
169
+ */
170
+ class PluginNotFoundError extends PluginError {
171
+ constructor(pluginName) {
172
+ super(`Plugin not found: ${pluginName}`, {
173
+ context: { pluginName },
174
+ });
175
+ this.code = 'PLUGIN_NOT_FOUND';
176
+ }
177
+ }
178
+
179
+ /**
180
+ * 检查错误类型
181
+ * @param {Error} error
182
+ * @param {Function} ErrorClass
183
+ * @returns {boolean}
184
+ */
185
+
186
+ /**
187
+ * 网络相关错误
188
+ */
189
+ class NetworkError extends FolikoError {
190
+ constructor(message, options = {}) {
191
+ super(message, {
192
+ code: 'NETWORK_ERROR',
193
+ context: {
194
+ host: options.host,
195
+ port: options.port,
196
+ code: options.code,
197
+ ...options.context,
198
+ },
199
+ });
200
+ }
201
+ }
202
+
203
+ /**
204
+ * 认证/授权错误
205
+ */
206
+ class AuthError extends FolikoError {
207
+ constructor(message, options = {}) {
208
+ super(message, {
209
+ code: 'AUTH_ERROR',
210
+ context: {
211
+ authType: options.authType,
212
+ userId: options.userId,
213
+ ...options.context,
214
+ },
215
+ });
216
+ }
217
+ }
218
+
219
+ /**
220
+ * 限流错误
221
+ */
222
+ class RateLimitError extends FolikoError {
223
+ constructor(message, options = {}) {
224
+ super(message, {
225
+ code: 'RATE_LIMIT_ERROR',
226
+ context: {
227
+ retryAfter: options.retryAfter,
228
+ limit: options.limit,
229
+ ...options.context,
230
+ },
231
+ });
232
+ }
233
+ }
234
+
235
+ /**
236
+ * 超时错误
237
+ */
238
+ class TimeoutError extends FolikoError {
239
+ constructor(message, options = {}) {
240
+ super(message, {
241
+ code: 'TIMEOUT_ERROR',
242
+ context: {
243
+ timeout: options.timeout,
244
+ operation: options.operation,
245
+ ...options.context,
246
+ },
247
+ });
248
+ }
249
+ }
250
+
251
+ /**
252
+ * 文件系统错误
253
+ */
254
+ class FileSystemError extends FolikoError {
255
+ constructor(message, options = {}) {
256
+ super(message, {
257
+ code: 'FS_ERROR',
258
+ context: {
259
+ path: options.path,
260
+ operation: options.operation,
261
+ ...options.context,
262
+ },
263
+ });
264
+ }
265
+ }
266
+
267
+ function isErrorOfType(error, ErrorClass) {
268
+ return error instanceof ErrorClass;
269
+ }
270
+
271
+ /**
272
+ * 安全地获取错误信息
273
+ * @param {Error} error
274
+ * @returns {Object}
275
+ */
276
+ function safeErrorInfo(error) {
277
+ if (error instanceof FolikoError) {
278
+ return error.toJSON();
279
+ }
280
+ return {
281
+ name: error.name || 'Error',
282
+ message: error.message,
283
+ code: 'UNKNOWN_ERROR',
284
+ context: {},
285
+ stack: error.stack,
286
+ };
287
+ }
288
+
289
+ /**
290
+ * 错误工厂函数 - 根据错误类型快速创建错误
291
+ * @param {string} type - 错误类型
292
+ * @param {string} message - 错误消息
293
+ * @param {Object} options - 错误选项
294
+ * @returns {FolikoError}
295
+ */
296
+ function createError(type, message, options = {}) {
297
+ const errorClasses = {
298
+ plugin: PluginError,
299
+ tool: ToolError,
300
+ ai: AIError,
301
+ session: SessionError,
302
+ config: ConfigError,
303
+ workflow: WorkflowError,
304
+ validation: ValidationError,
305
+ network: NetworkError,
306
+ auth: AuthError,
307
+ rateLimit: RateLimitError,
308
+ timeout: TimeoutError,
309
+ fs: FileSystemError,
310
+ };
311
+ const ErrorClass = errorClasses[type] || FolikoError;
312
+ return new ErrorClass(message, options);
313
+ }
314
+
315
+ /**
316
+ * 判断错误是否可重试
317
+ * @param {Error} error
318
+ * @returns {boolean}
319
+ */
320
+ function isRetryable(error) {
321
+ if (error instanceof RateLimitError) return true;
322
+ if (error instanceof TimeoutError) return true;
323
+ if (error instanceof NetworkError) return true;
324
+ if (error instanceof FolikoError) {
325
+ const retryableCodes = ['RATE_LIMIT_ERROR', 'TIMEOUT_ERROR', 'NETWORK_ERROR'];
326
+ return retryableCodes.includes(error.code);
327
+ }
328
+ return false;
329
+ }
330
+
331
+ /**
332
+ * 获取用户友好的错误消息
333
+ * @param {Error} error
334
+ * @returns {string}
335
+ */
336
+ function getFriendlyMessage(error) {
337
+ if (error instanceof FolikoError) {
338
+ const messages = {
339
+ PLUGIN_NOT_FOUND: '插件未找到,请检查配置',
340
+ TOOL_NOT_FOUND: '工具未找到,请检查工具名称',
341
+ CONFIG_ERROR: '配置错误,请检查配置文件',
342
+ VALIDATION_ERROR: '输入验证失败,请检查输入数据',
343
+ RATE_LIMIT_ERROR: '请求过于频繁,请稍后再试',
344
+ TIMEOUT_ERROR: '请求超时,请检查网络连接',
345
+ NETWORK_ERROR: '网络错误,请检查网络连接',
346
+ AUTH_ERROR: '认证失败,请检查凭据是否正确',
347
+ };
348
+ return messages[error.code] || error.message;
349
+ }
350
+ return error.message || '发生未知错误';
351
+ }
352
+
353
+ module.exports = {
354
+ FolikoError,
355
+ PluginError,
356
+ ToolError,
357
+ AIError,
358
+ SessionError,
359
+ ConfigError,
360
+ WorkflowError,
361
+ ValidationError,
362
+ ToolNotFoundError,
363
+ PluginNotFoundError,
364
+ NetworkError,
365
+ AuthError,
366
+ RateLimitError,
367
+ TimeoutError,
368
+ FileSystemError,
369
+ isErrorOfType,
370
+ safeErrorInfo,
371
+ createError,
372
+ isRetryable,
373
+ getFriendlyMessage,
374
+ };
@@ -1,97 +1,97 @@
1
- /**
2
- * EventEmitter 事件发射器
3
- * 简单的事件总线实现
4
- */
5
-
6
- class EventEmitter {
7
- constructor() {
8
- this._events = new Map()
9
- }
10
-
11
- /**
12
- * 监听事件
13
- * @param {string} event - 事件名
14
- * @param {Function} handler - 处理函数
15
- * @returns {this}
16
- */
17
- on(event, handler) {
18
- if (!this._events.has(event)) {
19
- this._events.set(event, new Set())
20
- }
21
- this._events.get(event).add(handler)
22
- return this
23
- }
24
-
25
- /**
26
- * 监听一次性事件
27
- * @param {string} event - 事件名
28
- * @param {Function} handler - 处理函数
29
- * @returns {this}
30
- */
31
- once(event, handler) {
32
- const wrapper = (...args) => {
33
- handler(...args)
34
- this.off(event, wrapper)
35
- }
36
- return this.on(event, wrapper)
37
- }
38
-
39
- /**
40
- * 取消监听
41
- * @param {string} event - 事件名
42
- * @param {Function} handler - 处理函数
43
- * @returns {this}
44
- */
45
- off(event, handler) {
46
- const handlers = this._events.get(event)
47
- if (handlers) {
48
- handlers.delete(handler)
49
- }
50
- return this
51
- }
52
-
53
- /**
54
- * 触发事件
55
- * @param {string} event - 事件名
56
- * @param {...any} args - 参数
57
- * @returns {this}
58
- */
59
- emit(event, ...args) {
60
- const handlers = this._events.get(event)
61
- if (handlers) {
62
- for (const handler of handlers) {
63
- try {
64
- handler(...args)
65
- } catch (err) {
66
- console.error(`[EventEmitter] Error in handler for '${event}':`, err)
67
- }
68
- }
69
- }
70
- return this
71
- }
72
-
73
- /**
74
- * 移除所有监听器
75
- * @param {string} [event] - 可选,指定事件名
76
- * @returns {this}
77
- */
78
- removeAllListeners(event) {
79
- if (event) {
80
- this._events.delete(event)
81
- } else {
82
- this._events.clear()
83
- }
84
- return this
85
- }
86
-
87
- /**
88
- * 获取监听器数量
89
- * @param {string} event - 事件名
90
- * @returns {number}
91
- */
92
- listenerCount(event) {
93
- return this._events.get(event)?.size || 0
94
- }
95
- }
96
-
97
- module.exports = { EventEmitter }
1
+ /**
2
+ * EventEmitter 事件发射器
3
+ * 简单的事件总线实现
4
+ */
5
+
6
+ class EventEmitter {
7
+ constructor() {
8
+ this._events = new Map();
9
+ }
10
+
11
+ /**
12
+ * 监听事件
13
+ * @param {string} event - 事件名
14
+ * @param {Function} handler - 处理函数
15
+ * @returns {this}
16
+ */
17
+ on(event, handler) {
18
+ if (!this._events.has(event)) {
19
+ this._events.set(event, new Set());
20
+ }
21
+ this._events.get(event).add(handler);
22
+ return this;
23
+ }
24
+
25
+ /**
26
+ * 监听一次性事件
27
+ * @param {string} event - 事件名
28
+ * @param {Function} handler - 处理函数
29
+ * @returns {this}
30
+ */
31
+ once(event, handler) {
32
+ const wrapper = (...args) => {
33
+ handler(...args);
34
+ this.off(event, wrapper);
35
+ };
36
+ return this.on(event, wrapper);
37
+ }
38
+
39
+ /**
40
+ * 取消监听
41
+ * @param {string} event - 事件名
42
+ * @param {Function} handler - 处理函数
43
+ * @returns {this}
44
+ */
45
+ off(event, handler) {
46
+ const handlers = this._events.get(event);
47
+ if (handlers) {
48
+ handlers.delete(handler);
49
+ }
50
+ return this;
51
+ }
52
+
53
+ /**
54
+ * 触发事件
55
+ * @param {string} event - 事件名
56
+ * @param {...any} args - 参数
57
+ * @returns {this}
58
+ */
59
+ emit(event, ...args) {
60
+ const handlers = this._events.get(event);
61
+ if (handlers) {
62
+ for (const handler of handlers) {
63
+ try {
64
+ handler(...args);
65
+ } catch (err) {
66
+ // EventEmitter error: Error in handler for '${event}':`, err)
67
+ }
68
+ }
69
+ }
70
+ return this;
71
+ }
72
+
73
+ /**
74
+ * 移除所有监听器
75
+ * @param {string} [event] - 可选,指定事件名
76
+ * @returns {this}
77
+ */
78
+ removeAllListeners(event) {
79
+ if (event) {
80
+ this._events.delete(event);
81
+ } else {
82
+ this._events.clear();
83
+ }
84
+ return this;
85
+ }
86
+
87
+ /**
88
+ * 获取监听器数量
89
+ * @param {string} event - 事件名
90
+ * @returns {number}
91
+ */
92
+ listenerCount(event) {
93
+ return this._events.get(event)?.size || 0;
94
+ }
95
+ }
96
+
97
+ module.exports = { EventEmitter };
@@ -0,0 +1,133 @@
1
+ /**
2
+ * Foliko ID 生成器 - 统一 ID 生成
3
+ * 支持 UUID、ULID、雪花ID、自定义前缀
4
+ */
5
+
6
+ const crypto = require('crypto');
7
+
8
+ /**
9
+ * 生成标准 UUID v4
10
+ * @returns {string}
11
+ */
12
+ function uuid() {
13
+ if (crypto.randomUUID) {
14
+ return crypto.randomUUID();
15
+ }
16
+ // 回退实现 (Node.js < 14.17)
17
+ return ([1e7] + -1e3 + -4e3 + -8e3 + -1e11).replace(/[018]/g, (c) =>
18
+ (c ^ (crypto.randomBytes(1)[0] & (15 >> (c / 4)))).toString(16)
19
+ );
20
+ }
21
+
22
+ /**
23
+ * 生成带前缀的 UUID
24
+ * @param {string} prefix - 前缀,如 'agent', 'session', 'task'
25
+ * @returns {string}
26
+ */
27
+ function prefixedId(prefix = 'id') {
28
+ return `${prefix}_${uuid()}`;
29
+ }
30
+
31
+ /**
32
+ * 生成短 ID (无连字符)
33
+ * @param {number} length - 长度,默认 12
34
+ * @returns {string}
35
+ */
36
+ function shortId(length = 12) {
37
+ return uuid().replace(/-/g, '').slice(0, length);
38
+ }
39
+
40
+ /**
41
+ * 生成雪花 ID (Twitter Snowflake 简化版)
42
+ * @returns {string}
43
+ */
44
+ let snowflakeEpoch = 1609459200000; // 2021-01-01
45
+ let snowflakeSequence = 0;
46
+ let snowflakeLastTime = 0;
47
+
48
+ function snowflake() {
49
+ const now = Date.now();
50
+
51
+ if (now === snowflakeLastTime) {
52
+ snowflakeSequence = (snowflakeSequence + 1) & 4095;
53
+ } else {
54
+ snowflakeSequence = 0;
55
+ snowflakeLastTime = now;
56
+ }
57
+
58
+ const timestamp = now - snowflakeEpoch;
59
+ const workerId = 1;
60
+
61
+ const id = (BigInt(timestamp) << 22n) | (BigInt(workerId) << 12n) | BigInt(snowflakeSequence);
62
+
63
+ return id.toString();
64
+ }
65
+
66
+ /**
67
+ * 生成任务 ID (向后兼容原有格式)
68
+ * @param {string} [prefix] - 可选前缀
69
+ * @returns {string}
70
+ */
71
+ function taskId(prefix) {
72
+ const suffix = crypto.randomBytes(8).toString('base64url');
73
+ return prefix ? `${prefix}_${suffix}` : suffix;
74
+ }
75
+
76
+ /**
77
+ * 生成目标 ID (向后兼容原有格式)
78
+ * @returns {string}
79
+ */
80
+ function goalId() {
81
+ const random = crypto.randomBytes(8).toString('base64url');
82
+ return `goal_${Date.now()}_${random}`;
83
+ }
84
+
85
+ /**
86
+ * 生成会话 ID (向后兼容原有格式)
87
+ * @returns {string}
88
+ */
89
+ function sessionId() {
90
+ const random = crypto.randomBytes(8).toString('base64url');
91
+ return `session_${Date.now()}_${random}`;
92
+ }
93
+
94
+ /**
95
+ * 生成邮箱消息 ID
96
+ * @returns {string}
97
+ */
98
+ function messageId() {
99
+ const timestamp = Date.now();
100
+ const random = crypto.randomBytes(10).toString('base64url');
101
+ return `<${timestamp}.${random}@foliko>`;
102
+ }
103
+
104
+ /**
105
+ * 生成规则 ID
106
+ * @returns {string}
107
+ */
108
+ function ruleId() {
109
+ const random = crypto.randomBytes(8).toString('base64url');
110
+ return `rule_${Date.now()}_${random}`;
111
+ }
112
+
113
+ /**
114
+ * 重置雪花 ID 计数器 (用于测试)
115
+ */
116
+ function resetSnowflake() {
117
+ snowflakeSequence = 0;
118
+ snowflakeLastTime = 0;
119
+ }
120
+
121
+ // 导出所有函数
122
+ module.exports = {
123
+ uuid,
124
+ prefixedId,
125
+ shortId,
126
+ snowflake,
127
+ taskId,
128
+ goalId,
129
+ sessionId,
130
+ messageId,
131
+ ruleId,
132
+ resetSnowflake,
133
+ };