foliko 1.0.74 → 1.0.75

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 (237) 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 +157 -149
  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 +4 -4
  128. package/cli/src/commands/chat.js +53 -51
  129. package/cli/src/commands/list.js +40 -37
  130. package/cli/src/index.js +18 -18
  131. package/cli/src/ui/chat-ui.js +78 -76
  132. package/cli/src/utils/ansi.js +15 -15
  133. package/cli/src/utils/markdown.js +112 -116
  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 +95 -97
  140. package/examples/basic.js +115 -110
  141. package/examples/bootstrap.js +52 -43
  142. package/examples/mcp-example.js +56 -53
  143. package/examples/skill-example.js +49 -49
  144. package/examples/test-chat.js +60 -58
  145. package/examples/test-mcp.js +49 -43
  146. package/examples/test-reload.js +38 -40
  147. package/examples/test-telegram.js +3 -3
  148. package/examples/test-tg-bot.js +7 -4
  149. package/examples/test-tg-simple.js +4 -3
  150. package/examples/test-tg.js +3 -3
  151. package/examples/test-think.js +13 -7
  152. package/examples/test-web-plugin.js +61 -56
  153. package/examples/test-weixin-feishu.js +40 -37
  154. package/examples/workflow.js +49 -49
  155. package/foliko-1.0.75.tgz +0 -0
  156. package/package.json +37 -3
  157. package/plugins/ai-plugin.js +7 -5
  158. package/plugins/ambient-agent/EventWatcher.js +113 -0
  159. package/plugins/ambient-agent/ExplorerLoop.js +640 -0
  160. package/plugins/ambient-agent/GoalManager.js +197 -0
  161. package/plugins/ambient-agent/Reflector.js +95 -0
  162. package/plugins/ambient-agent/StateStore.js +90 -0
  163. package/plugins/ambient-agent/constants.js +101 -0
  164. package/plugins/ambient-agent/index.js +579 -0
  165. package/plugins/default-plugins.js +62 -49
  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 +23 -19
  174. package/plugins/file-system-plugin.js +456 -106
  175. package/plugins/install-plugin.js +6 -4
  176. package/plugins/python-executor-plugin.js +3 -1
  177. package/plugins/python-plugin-loader.js +10 -8
  178. package/plugins/rules-plugin.js +5 -3
  179. package/plugins/scheduler-plugin.js +18 -16
  180. package/plugins/session-plugin.js +3 -1
  181. package/plugins/storage-plugin.js +5 -3
  182. package/plugins/subagent-plugin.js +152 -92
  183. package/plugins/telegram-plugin.js +26 -19
  184. package/plugins/think-plugin.js +4 -2
  185. package/plugins/tools-plugin.js +3 -1
  186. package/plugins/web-plugin.js +15 -13
  187. package/plugins/weixin-plugin.js +43 -36
  188. package/reports/system-health-report-20260401.md +79 -0
  189. package/skills/ambient-agent/SKILL.md +49 -39
  190. package/skills/foliko-dev/AGENTS.md +64 -61
  191. package/skills/foliko-dev/SKILL.md +125 -119
  192. package/skills/mcp-usage/SKILL.md +19 -17
  193. package/skills/python-plugin-dev/SKILL.md +16 -15
  194. package/skills/skill-guide/SKILL.md +12 -12
  195. package/skills/subagent-guide/SKILL.md +237 -0
  196. package/skills/workflow-guide/SKILL.md +90 -45
  197. package/skills/workflow-troubleshooting/DEBUGGING.md +36 -21
  198. package/skills/workflow-troubleshooting/SKILL.md +156 -79
  199. package/src/capabilities/index.js +4 -4
  200. package/src/capabilities/skill-manager.js +211 -197
  201. package/src/capabilities/workflow-engine.js +461 -547
  202. package/src/core/agent-chat.js +426 -279
  203. package/src/core/agent.js +453 -249
  204. package/src/core/framework.js +183 -149
  205. package/src/core/index.js +8 -8
  206. package/src/core/plugin-base.js +52 -52
  207. package/src/core/plugin-manager.js +377 -281
  208. package/src/core/provider.js +35 -32
  209. package/src/core/sub-agent-config.js +264 -0
  210. package/src/core/system-prompt-builder.js +120 -0
  211. package/src/core/tool-registry.js +416 -33
  212. package/src/core/tool-router.js +149 -68
  213. package/src/executors/executor-base.js +58 -58
  214. package/src/executors/mcp-executor.js +269 -257
  215. package/src/index.js +5 -17
  216. package/src/utils/circuit-breaker.js +301 -0
  217. package/src/utils/error-boundary.js +363 -0
  218. package/src/utils/error.js +374 -0
  219. package/src/utils/event-emitter.js +20 -20
  220. package/src/utils/id.js +133 -0
  221. package/src/utils/index.js +217 -3
  222. package/src/utils/logger.js +181 -0
  223. package/src/utils/plugin-helpers.js +90 -0
  224. package/src/utils/retry.js +122 -0
  225. package/src/utils/sandbox.js +292 -0
  226. package/test/tool-registry-validation.test.js +218 -0
  227. package/test_report.md +70 -0
  228. package/website/docs/api.html +169 -107
  229. package/website/docs/configuration.html +296 -144
  230. package/website/docs/plugin-development.html +154 -85
  231. package/website/docs/project-structure.html +110 -109
  232. package/website/docs/skill-development.html +117 -61
  233. package/website/index.html +209 -205
  234. package/website/script.js +20 -17
  235. package/website/styles.css +1 -1
  236. package/plugins/ambient-agent-plugin.js +0 -1565
  237. 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
+ };
@@ -5,7 +5,7 @@
5
5
 
6
6
  class EventEmitter {
7
7
  constructor() {
8
- this._events = new Map()
8
+ this._events = new Map();
9
9
  }
10
10
 
11
11
  /**
@@ -16,10 +16,10 @@ class EventEmitter {
16
16
  */
17
17
  on(event, handler) {
18
18
  if (!this._events.has(event)) {
19
- this._events.set(event, new Set())
19
+ this._events.set(event, new Set());
20
20
  }
21
- this._events.get(event).add(handler)
22
- return this
21
+ this._events.get(event).add(handler);
22
+ return this;
23
23
  }
24
24
 
25
25
  /**
@@ -30,10 +30,10 @@ class EventEmitter {
30
30
  */
31
31
  once(event, handler) {
32
32
  const wrapper = (...args) => {
33
- handler(...args)
34
- this.off(event, wrapper)
35
- }
36
- return this.on(event, wrapper)
33
+ handler(...args);
34
+ this.off(event, wrapper);
35
+ };
36
+ return this.on(event, wrapper);
37
37
  }
38
38
 
39
39
  /**
@@ -43,11 +43,11 @@ class EventEmitter {
43
43
  * @returns {this}
44
44
  */
45
45
  off(event, handler) {
46
- const handlers = this._events.get(event)
46
+ const handlers = this._events.get(event);
47
47
  if (handlers) {
48
- handlers.delete(handler)
48
+ handlers.delete(handler);
49
49
  }
50
- return this
50
+ return this;
51
51
  }
52
52
 
53
53
  /**
@@ -57,17 +57,17 @@ class EventEmitter {
57
57
  * @returns {this}
58
58
  */
59
59
  emit(event, ...args) {
60
- const handlers = this._events.get(event)
60
+ const handlers = this._events.get(event);
61
61
  if (handlers) {
62
62
  for (const handler of handlers) {
63
63
  try {
64
- handler(...args)
64
+ handler(...args);
65
65
  } catch (err) {
66
- console.error(`[EventEmitter] Error in handler for '${event}':`, err)
66
+ // EventEmitter error: Error in handler for '${event}':`, err)
67
67
  }
68
68
  }
69
69
  }
70
- return this
70
+ return this;
71
71
  }
72
72
 
73
73
  /**
@@ -77,11 +77,11 @@ class EventEmitter {
77
77
  */
78
78
  removeAllListeners(event) {
79
79
  if (event) {
80
- this._events.delete(event)
80
+ this._events.delete(event);
81
81
  } else {
82
- this._events.clear()
82
+ this._events.clear();
83
83
  }
84
- return this
84
+ return this;
85
85
  }
86
86
 
87
87
  /**
@@ -90,8 +90,8 @@ class EventEmitter {
90
90
  * @returns {number}
91
91
  */
92
92
  listenerCount(event) {
93
- return this._events.get(event)?.size || 0
93
+ return this._events.get(event)?.size || 0;
94
94
  }
95
95
  }
96
96
 
97
- module.exports = { EventEmitter }
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
+ };