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/utils/error.js
CHANGED
|
@@ -1,374 +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
|
+
/**
|
|
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
|
+
};
|