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.
Files changed (88) hide show
  1. package/.claude/settings.local.json +159 -157
  2. package/cli/bin/foliko.js +12 -12
  3. package/cli/src/commands/chat.js +143 -143
  4. package/cli/src/commands/list.js +93 -93
  5. package/cli/src/index.js +75 -75
  6. package/cli/src/ui/chat-ui.js +201 -201
  7. package/cli/src/utils/ansi.js +40 -40
  8. package/cli/src/utils/markdown.js +292 -292
  9. package/examples/ambient-example.js +194 -194
  10. package/examples/basic.js +115 -115
  11. package/examples/bootstrap.js +121 -121
  12. package/examples/mcp-example.js +56 -56
  13. package/examples/skill-example.js +49 -49
  14. package/examples/test-chat.js +137 -137
  15. package/examples/test-mcp.js +85 -85
  16. package/examples/test-reload.js +59 -59
  17. package/examples/test-telegram.js +50 -50
  18. package/examples/test-tg-bot.js +45 -45
  19. package/examples/test-tg-simple.js +47 -47
  20. package/examples/test-tg.js +62 -62
  21. package/examples/test-think.js +43 -43
  22. package/examples/test-web-plugin.js +103 -103
  23. package/examples/test-weixin-feishu.js +103 -103
  24. package/examples/workflow.js +158 -158
  25. package/package.json +1 -1
  26. package/plugins/ai-plugin.js +102 -102
  27. package/plugins/ambient-agent/EventWatcher.js +113 -113
  28. package/plugins/ambient-agent/ExplorerLoop.js +640 -640
  29. package/plugins/ambient-agent/GoalManager.js +197 -197
  30. package/plugins/ambient-agent/Reflector.js +95 -95
  31. package/plugins/ambient-agent/StateStore.js +90 -90
  32. package/plugins/ambient-agent/constants.js +101 -101
  33. package/plugins/ambient-agent/index.js +579 -579
  34. package/plugins/audit-plugin.js +187 -187
  35. package/plugins/default-plugins.js +662 -662
  36. package/plugins/email/constants.js +64 -64
  37. package/plugins/email/handlers.js +461 -461
  38. package/plugins/email/index.js +278 -278
  39. package/plugins/email/monitor.js +269 -269
  40. package/plugins/email/parser.js +138 -138
  41. package/plugins/email/reply.js +151 -151
  42. package/plugins/email/utils.js +124 -124
  43. package/plugins/feishu-plugin.js +481 -481
  44. package/plugins/file-system-plugin.js +826 -826
  45. package/plugins/install-plugin.js +199 -199
  46. package/plugins/python-executor-plugin.js +367 -367
  47. package/plugins/python-plugin-loader.js +481 -481
  48. package/plugins/rules-plugin.js +294 -294
  49. package/plugins/scheduler-plugin.js +691 -691
  50. package/plugins/session-plugin.js +369 -369
  51. package/plugins/shell-executor-plugin.js +197 -197
  52. package/plugins/storage-plugin.js +240 -240
  53. package/plugins/subagent-plugin.js +845 -845
  54. package/plugins/telegram-plugin.js +482 -482
  55. package/plugins/think-plugin.js +345 -345
  56. package/plugins/tools-plugin.js +196 -196
  57. package/plugins/web-plugin.js +606 -606
  58. package/plugins/weixin-plugin.js +545 -545
  59. package/src/capabilities/index.js +11 -11
  60. package/src/capabilities/skill-manager.js +609 -609
  61. package/src/capabilities/workflow-engine.js +1109 -1109
  62. package/src/core/agent-chat.js +882 -882
  63. package/src/core/agent.js +892 -892
  64. package/src/core/framework.js +465 -465
  65. package/src/core/index.js +19 -19
  66. package/src/core/plugin-base.js +219 -219
  67. package/src/core/plugin-manager.js +863 -863
  68. package/src/core/provider.js +114 -114
  69. package/src/core/sub-agent-config.js +264 -264
  70. package/src/core/system-prompt-builder.js +120 -120
  71. package/src/core/tool-registry.js +517 -517
  72. package/src/core/tool-router.js +297 -297
  73. package/src/executors/executor-base.js +58 -58
  74. package/src/executors/mcp-executor.js +741 -741
  75. package/src/index.js +25 -25
  76. package/src/utils/circuit-breaker.js +301 -301
  77. package/src/utils/error-boundary.js +363 -363
  78. package/src/utils/error.js +374 -374
  79. package/src/utils/event-emitter.js +97 -97
  80. package/src/utils/id.js +133 -133
  81. package/src/utils/index.js +217 -217
  82. package/src/utils/logger.js +181 -181
  83. package/src/utils/plugin-helpers.js +90 -90
  84. package/src/utils/retry.js +122 -122
  85. package/src/utils/sandbox.js +292 -292
  86. package/test/tool-registry-validation.test.js +218 -218
  87. package/website/script.js +136 -136
  88. package/foliko-1.0.75.tgz +0 -0
@@ -1,662 +1,662 @@
1
- /**
2
- * 默认插件配置加载器
3
- * 检测 .agent/ 目录下的配置,提供给 bootstrap() 使用
4
- */
5
-
6
- const fs = require('fs')
7
- const path = require('path')
8
- const { Plugin } = require('../src/core/plugin-base')
9
- const { logger } = require('../src/utils/logger')
10
- const log = logger.child('AgentConfig')
11
- const bootstrapLog = logger.child('Bootstrap')
12
-
13
- /**
14
- * 加载 .agent 目录下的配置
15
- * 返回配置对象,可用于手动加载插件
16
- */
17
- function loadAgentConfig(agentDir = '.agent') {
18
- const config = {
19
- agentDir,
20
- ai: {},
21
- mcpServers: {},
22
- plugins: [],
23
- skillsDirs: [],
24
- agentsDir: null // 子Agent配置目录
25
- }
26
- const cmdDir=path.resolve(process.cwd())
27
- const resolvedDir = path.resolve(process.cwd(), agentDir)
28
-
29
- if (!fs.existsSync(resolvedDir)) {
30
- log.info(` .agent directory not found: ${resolvedDir}`)
31
- return config
32
- }
33
-
34
- log.info(` Loading config from: ${resolvedDir}`)
35
-
36
- // 添加 agents 目录(如果存在)
37
- const agentsDir = path.join(resolvedDir, 'agents')
38
- if (fs.existsSync(agentsDir)) {
39
- config.agentsDir = agentsDir
40
- log.info(` Found agents directory: ${agentsDir}`)
41
- }
42
-
43
- // 加载 config 文件(key=value 格式)
44
- const configFile = path.join(resolvedDir, 'config')
45
- if (fs.existsSync(configFile)) {
46
- try {
47
- const lines = fs.readFileSync(configFile, 'utf-8').split('\n')
48
- for (const line of lines) {
49
- const trimmed = line.trim()
50
- if (!trimmed || trimmed.startsWith('#')) continue
51
-
52
- const colonIndex = trimmed.indexOf(':')
53
- if (colonIndex === -1) continue
54
-
55
- const key = trimmed.substring(0, colonIndex).trim()
56
- const value = trimmed.substring(colonIndex + 1).trim()
57
-
58
- if (key === 'ai_key' || key === 'api_key') {
59
- config.ai.apiKey = value
60
- } else if (key === 'ai_model' || key === 'model') {
61
- config.ai.model = value
62
- } else if (key === 'ai_provider' || key === 'provider') {
63
- config.ai.provider = value
64
- } else if (key === 'ai_base_url' || key === 'baseURL') {
65
- config.ai.baseURL = value
66
- }
67
- }
68
- } catch (err) {
69
- log.error(` Failed to load config:`, err.message)
70
- }
71
- }
72
-
73
- // 加载 ai.json
74
- const aiFile = path.join(resolvedDir, 'ai.json')
75
- if (fs.existsSync(aiFile)) {
76
- try {
77
- const content = fs.readFileSync(aiFile, 'utf-8')
78
- const aiConfig = JSON.parse(content)
79
- config.ai = { ...config.ai, ...aiConfig }
80
- } catch (err) {
81
- log.error(` Failed to load ai.json:`, err.message)
82
- }
83
- }
84
-
85
- // 加载 plugins.json (支持 telegram, weixin, email 等插件配置)
86
- const pluginsFile = path.join(resolvedDir, 'plugins.json')
87
- if (fs.existsSync(pluginsFile)) {
88
- try {
89
- const content = fs.readFileSync(pluginsFile, 'utf-8')
90
- const pluginsConfig = JSON.parse(content)
91
- // telegram 配置
92
- if (pluginsConfig.telegram) {
93
- config.telegram = pluginsConfig.telegram
94
- }
95
- // weixin 配置
96
- if (pluginsConfig.weixin) {
97
- config.weixin = pluginsConfig.weixin
98
- }
99
- // email 配置
100
- if (pluginsConfig.email) {
101
- config.email = pluginsConfig.email
102
- }
103
- } catch (err) {
104
- log.error(` Failed to load plugins.json:`, err.message)
105
- }
106
- }
107
-
108
- // 加载 weixin.json (如果存在)
109
- const weixinFile = path.join(resolvedDir, 'weixin.json')
110
- if (fs.existsSync(weixinFile)) {
111
- try {
112
- const content = fs.readFileSync(weixinFile, 'utf-8')
113
- config.weixin = { ...config.weixin, ...JSON.parse(content) }
114
- } catch (err) {
115
- log.error(` Failed to load weixin.json:`, err.message)
116
- }
117
- }
118
-
119
- // 加载 mcp_config.json
120
- const mcpFile = path.join(resolvedDir, 'mcp_config.json')
121
- if (fs.existsSync(mcpFile)) {
122
- try {
123
- const content = fs.readFileSync(mcpFile, 'utf-8')
124
- const mcpConfig = JSON.parse(content)
125
- config.mcpServers = mcpConfig.mcpServers || {}
126
- } catch (err) {
127
- log.error(` Failed to load mcp_config.json:`, err.message)
128
- }
129
- }
130
-
131
- // 添加 .agent/skills 目录(不存在则创建)
132
- const skillsDir = path.join(resolvedDir, 'skills')
133
- if (fs.existsSync(resolvedDir) && !fs.existsSync(skillsDir)) {
134
- fs.mkdirSync(skillsDir, { recursive: true })
135
- log.info(` Created skills directory: ${skillsDir}`)
136
- }
137
- if (fs.existsSync(skillsDir)) {
138
- config.skillsDirs.push(skillsDir)
139
- }
140
-
141
- // 添加 .agent/skills 目录(不存在则创建)
142
- const cmdskillsDir = path.join(cmdDir, 'skills')
143
- if (fs.existsSync(resolvedDir) && !fs.existsSync(cmdskillsDir)) {
144
- fs.mkdirSync(cmdskillsDir, { recursive: true })
145
- log.info(` Created skills directory: ${cmdskillsDir}`)
146
- }
147
- if (fs.existsSync(cmdskillsDir)) {
148
- config.skillsDirs.push(cmdskillsDir)
149
- }
150
-
151
- // 添加 agentDir 父目录的 skills/ 目录
152
- const parentDir = path.dirname(__dirname)
153
- const rootSkillsDir = path.join(parentDir, 'skills')
154
- if (fs.existsSync(rootSkillsDir)) {
155
- config.skillsDirs.push(rootSkillsDir)
156
- }
157
-
158
- // const defaultSkillsDir = path.join(process.cwd(), 'defaultSkills')
159
- // if (fs.existsSync(defaultSkillsDir)) {
160
- // config.skillsDirs.push(defaultSkillsDir)
161
- // }
162
-
163
- return config
164
- }
165
-
166
- /**
167
- * DefaultPlugins - 默认插件配置加载器
168
- */
169
- class DefaultPlugins extends Plugin {
170
- constructor(config = {}) {
171
- super()
172
- this.name = 'defaults'
173
- this.version = '1.0.0'
174
- this.description = '默认插件配置加载器'
175
- this.priority = 0 // 最先加载
176
- this.system = true
177
-
178
- this._framework = null
179
- this._agentDir = config.agentDir || '.agent'
180
- this._config = null
181
- }
182
-
183
- install(framework) {
184
- this._framework = framework
185
- return this
186
- }
187
-
188
- start(framework) {
189
- // 加载配置
190
- this._config = loadAgentConfig(this._agentDir)
191
- return this
192
- }
193
-
194
- /**
195
- * 获取加载的配置
196
- */
197
- getConfig() {
198
- return this._config || loadAgentConfig(this._agentDir)
199
- }
200
-
201
- reload(framework) {
202
- this._framework = framework
203
- this._config = loadAgentConfig(this._agentDir)
204
- }
205
-
206
- uninstall(framework) {
207
- this._framework = null
208
- this._config = null
209
- }
210
- }
211
-
212
- /**
213
- * Bootstrap 辅助函数
214
- * 根据配置加载所有默认插件
215
- */
216
- async function bootstrapDefaults(framework, config = {}) {
217
- // 如果已经有配置,使用现有配置;否则加载
218
- const agentConfig = config._config
219
- if (!agentConfig) {
220
- bootstrapLog.error(' No config provided, skipping plugin loading')
221
- return
222
- }
223
- // 设置 bootstrap 模式,避免重复启动
224
- framework.pluginManager.setBootstrapping(true)
225
-
226
- // 合并 AI 配置
227
- const aiConfig = { ...agentConfig.ai, ...config.aiConfig }
228
-
229
- // 核心插件列表(不能禁用,必须加载)
230
- const CORE_PLUGINS = new Set([
231
- 'install', 'ai', 'storage', 'tools', 'workflow', 'skill-manager',
232
- 'mcp-executor', 'shell-executor', 'python-executor', 'session', 'web',
233
- 'audit', 'rules', 'scheduler', 'file-system', 'think', 'ambient',
234
- 'python-plugin-loader', 'telegram', 'weixin', 'subagent-manager'
235
- ])
236
-
237
- // 辅助函数:检查插件是否应该加载
238
- // 传入插件名称(字符串)、已创建的实例,或插件类
239
- const shouldLoad = (plugin) => {
240
- const name = typeof plugin === 'string' ? plugin : (plugin.name || plugin.prototype?.name)
241
- if (framework.pluginManager.has(name)) {
242
- framework._debug&&bootstrapLog.debug(` ${name} Plugin already loaded, skipping`)
243
- return false
244
- }
245
- // 系统插件(system: true)不能禁用
246
- let isSystem = false
247
- if (typeof plugin === 'function') {
248
- isSystem = plugin.prototype?.system === true
249
- } else if (typeof plugin === 'object') {
250
- isSystem = plugin.system === true
251
- }
252
- if (isSystem && framework.pluginManager.isEnabled(name) === false) {
253
- framework._debug&&bootstrapLog.debug(` ${name} is a system plugin, cannot be disabled`)
254
- }
255
- return true
256
- }
257
-
258
- // 0. Install 工具插件(最先加载,让其他插件能用到它的 node_modules)
259
- if (shouldLoad('install')) {
260
- const { InstallPlugin } = require('./install-plugin')
261
- await framework.loadPlugin(new InstallPlugin({ agentDir: agentConfig.agentDir }))
262
- framework._debug&&bootstrapLog.debug(' Install Plugin loaded')
263
- }
264
-
265
- // 合并 skillsDirs 配置(bootstrap 传入的优先)
266
- const skillsDirs = [
267
- ...(config.skillsDirs || []),
268
- ...(agentConfig.skillsDirs || [])
269
- ]
270
-
271
- framework._debug&&bootstrapLog.debug(' Loading default plugins...')
272
- // AI 插件(如果已禁用则跳过)
273
- if (!shouldLoad('ai') || !(aiConfig.provider || aiConfig.model || aiConfig.apiKey)) {
274
- // 跳过或已禁用
275
- } else {
276
- const { AIPlugin } = require('./ai-plugin')
277
- // 根据 provider 获取对应的 API key
278
- const upperProvider = (aiConfig.provider || 'deepseek').toUpperCase().replace(/-/g, '_')
279
- const envApiKey = process.env[`${upperProvider}_API_KEY`] || process.env.FOLIKO_API_KEY
280
- const aiPlugin = new AIPlugin({
281
- provider: aiConfig.provider || 'deepseek',
282
- model: aiConfig.model || 'deepseek-chat',
283
- apiKey: aiConfig.apiKey || envApiKey,
284
- baseURL: aiConfig.baseURL
285
- })
286
- await framework.loadPlugin(aiPlugin)
287
- framework._debug&&bootstrapLog.debug(' AI Plugin loaded')
288
- }
289
-
290
- // 1.5 创建主 Agent(供 Telegram 等需要绑定 Agent 的插件使用)
291
- if (!framework._mainAgent && aiConfig.provider) {
292
- const { Agent } = require('../src/core/agent')
293
- const aiPlugin = framework.pluginManager.get('ai')
294
- const aiClient = aiPlugin ? aiPlugin.getAIClient() : null
295
- framework._debug&&bootstrapLog.debug(' Creating Main Agent - aiClient:', !!aiClient)
296
-
297
- framework._mainAgent = framework.createAgent({
298
- name: 'MainAgent',
299
- systemPrompt: '你是一个智能助手。当用户提出问题或任务时,你会主动分析需求,选择合适的工具来获取信息或执行操作。你善于将复杂任务拆解为多个步骤,通过工具协作完成。',
300
- model: aiConfig.model,
301
- provider: aiConfig.provider,
302
- apiKey: aiConfig.apiKey,
303
- baseURL: aiConfig.baseURL
304
- })
305
- framework._agents.push(framework._mainAgent)
306
- framework._debug&&bootstrapLog.debug(' Main Agent created, has _chatHandler:', !!framework._mainAgent._chatHandler)
307
- }
308
-
309
- // 2. Storage 存储插件
310
- if (shouldLoad('storage')) {
311
- const { StoragePlugin } = require('./storage-plugin')
312
- await framework.loadPlugin(new StoragePlugin())
313
- framework._debug&&bootstrapLog.debug(' Storage Plugin loaded')
314
- }
315
-
316
- // 3. 内置工具插件
317
- if (shouldLoad('tools')) {
318
- const { ToolsPlugin } = require('./tools-plugin')
319
- await framework.loadPlugin(new ToolsPlugin())
320
- framework._debug&&bootstrapLog.debug(' Tools Plugin loaded')
321
- }
322
-
323
- // 4. 工作流插件
324
- if (shouldLoad('workflow')) {
325
- const { WorkflowPlugin } = require('../src/capabilities/workflow-engine')
326
- await framework.loadPlugin(new WorkflowPlugin())
327
- framework._debug&&bootstrapLog.debug(' Workflow Plugin loaded')
328
- }
329
-
330
- // 5. Skill 管理器插件
331
- if (shouldLoad('skill-manager')) {
332
- if (skillsDirs.length > 0) {
333
- const { SkillManagerPlugin } = require('../src/capabilities/skill-manager')
334
- // 传递所有 skills 目录
335
- await framework.loadPlugin(new SkillManagerPlugin({ skillsDirs }))
336
- framework._debug&&bootstrapLog.debug(' Skill Manager loaded')
337
- }
338
- }
339
-
340
- // 6. MCP 执行器插件(始终加载,确保 mcp_reload 工具可用)
341
- if (shouldLoad('mcp-executor')) {
342
- const mcpServers = Object.entries(agentConfig.mcpServers || {})
343
- const servers = mcpServers.map(([name, cfg]) => ({
344
- ...cfg,
345
- name,
346
- command: cfg.command,
347
- args: cfg.args || [],
348
- env: cfg.env || {},
349
- url: cfg.url,
350
- headers: cfg.headers
351
- }))
352
- const { MCPExecutorPlugin } = require('../src/executors/mcp-executor')
353
- await framework.loadPlugin(new MCPExecutorPlugin({ servers }))
354
- framework._debug&&bootstrapLog.debug(` MCP Executor loaded${servers.length > 0 ? ` (${servers.length} servers)` : ' (no servers)'}`)
355
- }
356
-
357
- // 7. Shell 执行器插件
358
- if (shouldLoad('shell-executor')) {
359
- const { ShellExecutorPlugin } = require('./shell-executor-plugin')
360
- await framework.loadPlugin(new ShellExecutorPlugin())
361
- framework._debug&&bootstrapLog.debug(' Shell Executor loaded')
362
- }
363
-
364
- // 8. Python 执行器插件
365
- if (shouldLoad('python-executor')) {
366
- const { PythonExecutorPlugin } = require('./python-executor-plugin')
367
- await framework.loadPlugin(new PythonExecutorPlugin())
368
- framework._debug&&bootstrapLog.debug(' Python Executor loaded')
369
- }
370
-
371
- // 8.5 Web Web服务插件
372
- if (shouldLoad('web')) {
373
- const { WebPlugin } = require('./web-plugin')
374
- await framework.loadPlugin(new WebPlugin())
375
- framework._debug&&bootstrapLog.debug(' Web Plugin loaded')
376
- }
377
-
378
- // 9. Session 会话管理插件
379
- if (shouldLoad('session')) {
380
- const { SessionPlugin } = require('./session-plugin')
381
- await framework.loadPlugin(new SessionPlugin())
382
- framework._debug&&bootstrapLog.debug(' Session Plugin loaded')
383
- }
384
-
385
- // 10. Audit 审计日志插件
386
- if (shouldLoad('audit')) {
387
- const { AuditPlugin } = require('./audit-plugin')
388
- await framework.loadPlugin(new AuditPlugin())
389
- framework._debug&&bootstrapLog.debug(' Audit Plugin loaded')
390
- }
391
-
392
- // 10. Rules 规则引擎插件
393
- if (shouldLoad('rules')) {
394
- const { RulesPlugin } = require('./rules-plugin')
395
- await framework.loadPlugin(new RulesPlugin())
396
- framework._debug&&bootstrapLog.debug(' Rules Plugin loaded')
397
- }
398
-
399
- // 11. Scheduler 定时任务插件
400
- if (shouldLoad('scheduler')) {
401
- const { SchedulerPlugin } = require('./scheduler-plugin')
402
- await framework.loadPlugin(new SchedulerPlugin())
403
- framework._debug&&bootstrapLog.debug(' Scheduler Plugin loaded')
404
- }
405
-
406
- // 11. FileSystem 文件系统插件
407
- if (shouldLoad('file-system')) {
408
- const { FileSystemPlugin } = require('./file-system-plugin')
409
- await framework.loadPlugin(new FileSystemPlugin())
410
- framework._debug&&bootstrapLog.debug(' FileSystem Plugin loaded')
411
- }
412
-
413
- // 12. Think 主动思考插件
414
- if (shouldLoad('think')) {
415
- const { ThinkPlugin } = require('./think-plugin')
416
- await framework.loadPlugin(new ThinkPlugin())
417
- framework._debug&&bootstrapLog.debug(' Think Plugin loaded')
418
- }
419
-
420
- // 12.1 Ambient Agent 插件
421
- if (shouldLoad('ambient')) {
422
- const { AmbientAgentPlugin } = require('./ambient-agent')
423
- await framework.loadPlugin(new AmbientAgentPlugin())
424
- framework._debug&&bootstrapLog.debug(' Ambient Agent Plugin loaded')
425
- }
426
-
427
- // 12.5 Python 插件加载器
428
- if (shouldLoad('python-plugin-loader')) {
429
- const { PythonPluginLoader } = require('./python-plugin-loader')
430
- await framework.loadPlugin(new PythonPluginLoader({ agentDir: agentConfig.agentDir }))
431
- framework._debug&&bootstrapLog.debug(' Python Plugin Loader loaded')
432
- }
433
-
434
- // 12.6 Telegram 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
435
- if (shouldLoad('telegram')) {
436
- const { TelegramPlugin } = require('./telegram-plugin')
437
- const telegramConfig = {
438
- botToken: process.env.TELEGRAM_BOT_TOKEN || agentConfig.telegram?.botToken
439
- }
440
- await framework.loadPlugin(new TelegramPlugin(telegramConfig))
441
- }
442
-
443
- // 12.7 WeChat 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
444
- if (shouldLoad('weixin')) {
445
- try {
446
- const { WeixinPlugin } = require('./weixin-plugin')
447
- const weixinConfig = {
448
- forceLogin: agentConfig.weixin?.forceLogin || false,
449
- qrcodeTerminal: agentConfig.weixin?.qrcodeTerminal !== false,
450
- allowedUsers: agentConfig.weixin?.allowedUsers || []
451
- }
452
- await framework.loadPlugin(new WeixinPlugin(weixinConfig))
453
- } catch (err) {
454
- bootstrapLog.warn(' WeChat Plugin not available:', err.message)
455
- }
456
- }
457
-
458
- // 12.8 Email 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
459
- if (shouldLoad('email')) {
460
- try {
461
- const { Plugin } = require('../src/core/plugin-base')
462
- const { logger } = require('../src/utils/logger')
463
- const log = logger.child('AgentConfig')
464
- const bootstrapLog = logger.child('Bootstrap')
465
- const { EmailPlugin } = require('./email')
466
- // 支持两种格式:email.smtp 或 email.config.smtp
467
- const emailData = agentConfig.email || {}
468
- const emailCfg = emailData.config || {}
469
- const emailConfig = {
470
- smtp: emailCfg.smtp || emailData.smtp || {},
471
- imap: emailCfg.imap || emailData.imap || {},
472
- clientId: emailCfg.clientId || emailData.clientId
473
- }
474
- await framework.loadPlugin(new EmailPlugin(emailConfig))
475
- } catch (err) {
476
- bootstrapLog.warn(' Email Plugin not available:', err.message)
477
- }
478
- }
479
-
480
- // 12.9 Feishu 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
481
- if (shouldLoad('feishu')) {
482
- try {
483
- const { FeishuPlugin } = require('./feishu-plugin')
484
- const feishuConfig = {
485
- allowedUsers: agentConfig.feishu?.allowedUsers || []
486
- }
487
- await framework.loadPlugin(new FeishuPlugin(feishuConfig))
488
- } catch (err) {
489
- bootstrapLog.warn(' Feishu Plugin not available:', err.message)
490
- }
491
- }
492
-
493
- // 12.10 SubAgent 管理器
494
- if (shouldLoad('subagent-manager')) {
495
- try {
496
- // 先初始化 SubAgentConfigManager 并加载所有配置
497
- const { SubAgentConfigManager } = require('../src/core/sub-agent-config')
498
- const subAgentConfigManager = new SubAgentConfigManager(agentConfig.agentsDir)
499
- subAgentConfigManager.loadAll()
500
- framework._subAgentConfigManager = subAgentConfigManager
501
-
502
- // 然后加载 SubAgentManagerPlugin
503
- const { SubAgentManagerPlugin } = require('./subagent-plugin')
504
- await framework.loadPlugin(new SubAgentManagerPlugin({ agentsDir: agentConfig.agentsDir }))
505
- framework._debug&&bootstrapLog.debug(' SubAgent Manager loaded')
506
- } catch (err) {
507
- bootstrapLog.warn(' SubAgent Manager failed:', err.message)
508
- }
509
- }
510
-
511
- // 13. 加载自定义插件
512
- await loadCustomPlugins(framework, agentConfig)
513
-
514
- framework._debug&&bootstrapLog.debug(' All plugins loaded')
515
-
516
- // 统一启动所有插件(避免重复启动)
517
- await framework.pluginManager.startAll()
518
-
519
- // 清除 bootstrap 模式
520
- framework.pluginManager.setBootstrapping(false)
521
-
522
- framework._debug&&bootstrapLog.debug(' Loaded plugins ', framework.pluginManager.getAll().length)
523
- framework._debug&&bootstrapLog.debug(' Loaded tools ', framework.getTools().length)
524
- }
525
-
526
- /**
527
- * 解析插件路径
528
- * 支持两种结构:
529
- * 1. 文件夹结构: .agent/plugins/my-plugin/index.js
530
- * 2. 单文件结构: .agent/plugins/my-plugin.js
531
- * @param {string} pluginsDir - 插件目录
532
- * @param {string} name - 插件名称
533
- * @returns {{path: string, type: 'folder'|'file'}|null} 插件路径和类型
534
- */
535
- function resolvePluginPath(pluginsDir, name) {
536
- const folderPath = path.join(pluginsDir, name)
537
- const filePath = path.join(pluginsDir, `${name}.js`)
538
-
539
- // 文件夹优先
540
- if (fs.existsSync(folderPath) && fs.statSync(folderPath).isDirectory()) {
541
- const pkgPath = path.join(folderPath, 'package.json')
542
- if (fs.existsSync(pkgPath)) {
543
- try {
544
- const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'))
545
- const main = pkg.main || 'index.js'
546
- const mainPath = path.join(folderPath, main)
547
- if (fs.existsSync(mainPath)) {
548
- return { path: mainPath, type: 'folder' }
549
- }
550
- } catch (err) {
551
- log.warn(` Failed to parse package.json for ${name}:`, err.message)
552
- }
553
- }
554
- // 默认加载 index.js
555
- const indexPath = path.join(folderPath, 'index.js')
556
- if (fs.existsSync(indexPath)) {
557
- return { path: indexPath, type: 'folder' }
558
- }
559
- //log.warn(` No entry point found for plugin folder: ${name}`)
560
- return null
561
- }
562
-
563
- // 单文件回退
564
- if (fs.existsSync(filePath)) {
565
- return { path: filePath, type: 'file' }
566
- }
567
-
568
- return null
569
- }
570
-
571
- /**
572
- * 扫描插件目录,返回所有插件名称
573
- * @param {string} pluginsDir - 插件目录
574
- * @returns {string[]} 插件名称列表
575
- */
576
- function scanPluginNames(pluginsDir) {
577
- if (!fs.existsSync(pluginsDir)) {
578
- return []
579
- }
580
-
581
- const names = new Set()
582
- const entries = fs.readdirSync(pluginsDir, { withFileTypes: true })
583
-
584
- for (const entry of entries) {
585
- if (entry.isDirectory()) {
586
- // 文件夹插件
587
- names.add(entry.name)
588
- } else if (entry.isFile() && entry.name.endsWith('.js')) {
589
- // 单文件插件(排除与文件夹同名的)
590
- const baseName = entry.name.replace(/\.js$/, '')
591
- if (!names.has(baseName)) {
592
- names.add(baseName)
593
- }
594
- }
595
- }
596
-
597
- return Array.from(names)
598
- }
599
-
600
- async function loadCustomPlugins(framework, agentConfig) {
601
- // 加载 .agent/plugins 目录下的自定义插件
602
- const pluginsDir = path.resolve(process.cwd(), '.agent', 'plugins')
603
-
604
- if (!fs.existsSync(pluginsDir)) {
605
- return
606
- }
607
-
608
- // 扫描所有插件名称(支持文件夹和单文件)
609
- const pluginNames = scanPluginNames(pluginsDir)
610
-
611
- for (const pluginName of pluginNames) {
612
- try {
613
- const resolved = resolvePluginPath(pluginsDir, pluginName)
614
- if (!resolved) {
615
- //log.warn(` Cannot resolve plugin: ${pluginName}`)
616
- continue
617
- }
618
-
619
- const { path: pluginPath, type } = resolved
620
-
621
- // 清除缓存
622
- delete require.cache[require.resolve(pluginPath)]
623
- const pluginModule = require(pluginPath)
624
-
625
- let plugin
626
- if (typeof pluginModule === 'function') {
627
- // 免引入写法:直接传递函数,让 loadPlugin 内部处理
628
- plugin = pluginModule
629
- } else if (pluginModule.default) {
630
- plugin = pluginModule.default
631
- } else {
632
- plugin = pluginModule
633
- }
634
-
635
- // 获取插件名称
636
- const tempPlugin = typeof plugin === 'function' && plugin.prototype?.name
637
- ? new plugin() : plugin
638
- const resolvedPluginName = tempPlugin.name || pluginName
639
-
640
- // 如果插件已加载且已启动,跳过
641
- if (framework.pluginManager.has(resolvedPluginName) &&
642
- framework.pluginManager.get(resolvedPluginName)?._started) {
643
- continue
644
- }
645
-
646
- //bootstrapLog.debug(` Loading custom plugin: ${pluginName} (${type})`)
647
- // .agent/plugins 目录下的插件强制启用,不受 state 文件 enabled: false 影响
648
- await framework.pluginManager.load(plugin, { forceEnabled: true })
649
- } catch (err) {
650
- log.error(` Failed to load plugin ${pluginName}:`, err.message)
651
- }
652
- }
653
- }
654
-
655
- module.exports = {
656
- DefaultPlugins,
657
- loadAgentConfig,
658
- bootstrapDefaults,
659
- loadCustomPlugins,
660
- resolvePluginPath,
661
- scanPluginNames
662
- }
1
+ /**
2
+ * 默认插件配置加载器
3
+ * 检测 .agent/ 目录下的配置,提供给 bootstrap() 使用
4
+ */
5
+
6
+ const fs = require('fs')
7
+ const path = require('path')
8
+ const { Plugin } = require('../src/core/plugin-base')
9
+ const { logger } = require('../src/utils/logger')
10
+ const log = logger.child('AgentConfig')
11
+ const bootstrapLog = logger.child('Bootstrap')
12
+
13
+ /**
14
+ * 加载 .agent 目录下的配置
15
+ * 返回配置对象,可用于手动加载插件
16
+ */
17
+ function loadAgentConfig(agentDir = '.agent') {
18
+ const config = {
19
+ agentDir,
20
+ ai: {},
21
+ mcpServers: {},
22
+ plugins: [],
23
+ skillsDirs: [],
24
+ agentsDir: null // 子Agent配置目录
25
+ }
26
+ const cmdDir=path.resolve(process.cwd())
27
+ const resolvedDir = path.resolve(process.cwd(), agentDir)
28
+
29
+ if (!fs.existsSync(resolvedDir)) {
30
+ log.info(` .agent directory not found: ${resolvedDir}`)
31
+ return config
32
+ }
33
+
34
+ log.info(` Loading config from: ${resolvedDir}`)
35
+
36
+ // 添加 agents 目录(如果存在)
37
+ const agentsDir = path.join(resolvedDir, 'agents')
38
+ if (fs.existsSync(agentsDir)) {
39
+ config.agentsDir = agentsDir
40
+ log.info(` Found agents directory: ${agentsDir}`)
41
+ }
42
+
43
+ // 加载 config 文件(key=value 格式)
44
+ const configFile = path.join(resolvedDir, 'config')
45
+ if (fs.existsSync(configFile)) {
46
+ try {
47
+ const lines = fs.readFileSync(configFile, 'utf-8').split('\n')
48
+ for (const line of lines) {
49
+ const trimmed = line.trim()
50
+ if (!trimmed || trimmed.startsWith('#')) continue
51
+
52
+ const colonIndex = trimmed.indexOf(':')
53
+ if (colonIndex === -1) continue
54
+
55
+ const key = trimmed.substring(0, colonIndex).trim()
56
+ const value = trimmed.substring(colonIndex + 1).trim()
57
+
58
+ if (key === 'ai_key' || key === 'api_key') {
59
+ config.ai.apiKey = value
60
+ } else if (key === 'ai_model' || key === 'model') {
61
+ config.ai.model = value
62
+ } else if (key === 'ai_provider' || key === 'provider') {
63
+ config.ai.provider = value
64
+ } else if (key === 'ai_base_url' || key === 'baseURL') {
65
+ config.ai.baseURL = value
66
+ }
67
+ }
68
+ } catch (err) {
69
+ log.error(` Failed to load config:`, err.message)
70
+ }
71
+ }
72
+
73
+ // 加载 ai.json
74
+ const aiFile = path.join(resolvedDir, 'ai.json')
75
+ if (fs.existsSync(aiFile)) {
76
+ try {
77
+ const content = fs.readFileSync(aiFile, 'utf-8')
78
+ const aiConfig = JSON.parse(content)
79
+ config.ai = { ...config.ai, ...aiConfig }
80
+ } catch (err) {
81
+ log.error(` Failed to load ai.json:`, err.message)
82
+ }
83
+ }
84
+
85
+ // 加载 plugins.json (支持 telegram, weixin, email 等插件配置)
86
+ const pluginsFile = path.join(resolvedDir, 'plugins.json')
87
+ if (fs.existsSync(pluginsFile)) {
88
+ try {
89
+ const content = fs.readFileSync(pluginsFile, 'utf-8')
90
+ const pluginsConfig = JSON.parse(content)
91
+ // telegram 配置
92
+ if (pluginsConfig.telegram) {
93
+ config.telegram = pluginsConfig.telegram
94
+ }
95
+ // weixin 配置
96
+ if (pluginsConfig.weixin) {
97
+ config.weixin = pluginsConfig.weixin
98
+ }
99
+ // email 配置
100
+ if (pluginsConfig.email) {
101
+ config.email = pluginsConfig.email
102
+ }
103
+ } catch (err) {
104
+ log.error(` Failed to load plugins.json:`, err.message)
105
+ }
106
+ }
107
+
108
+ // 加载 weixin.json (如果存在)
109
+ const weixinFile = path.join(resolvedDir, 'weixin.json')
110
+ if (fs.existsSync(weixinFile)) {
111
+ try {
112
+ const content = fs.readFileSync(weixinFile, 'utf-8')
113
+ config.weixin = { ...config.weixin, ...JSON.parse(content) }
114
+ } catch (err) {
115
+ log.error(` Failed to load weixin.json:`, err.message)
116
+ }
117
+ }
118
+
119
+ // 加载 mcp_config.json
120
+ const mcpFile = path.join(resolvedDir, 'mcp_config.json')
121
+ if (fs.existsSync(mcpFile)) {
122
+ try {
123
+ const content = fs.readFileSync(mcpFile, 'utf-8')
124
+ const mcpConfig = JSON.parse(content)
125
+ config.mcpServers = mcpConfig.mcpServers || {}
126
+ } catch (err) {
127
+ log.error(` Failed to load mcp_config.json:`, err.message)
128
+ }
129
+ }
130
+
131
+ // 添加 .agent/skills 目录(不存在则创建)
132
+ const skillsDir = path.join(resolvedDir, 'skills')
133
+ if (fs.existsSync(resolvedDir) && !fs.existsSync(skillsDir)) {
134
+ fs.mkdirSync(skillsDir, { recursive: true })
135
+ log.info(` Created skills directory: ${skillsDir}`)
136
+ }
137
+ if (fs.existsSync(skillsDir)) {
138
+ config.skillsDirs.push(skillsDir)
139
+ }
140
+
141
+ // 添加 .agent/skills 目录(不存在则创建)
142
+ const cmdskillsDir = path.join(cmdDir, 'skills')
143
+ if (fs.existsSync(resolvedDir) && !fs.existsSync(cmdskillsDir)) {
144
+ fs.mkdirSync(cmdskillsDir, { recursive: true })
145
+ log.info(` Created skills directory: ${cmdskillsDir}`)
146
+ }
147
+ if (fs.existsSync(cmdskillsDir)) {
148
+ config.skillsDirs.push(cmdskillsDir)
149
+ }
150
+
151
+ // 添加 agentDir 父目录的 skills/ 目录
152
+ const parentDir = path.dirname(__dirname)
153
+ const rootSkillsDir = path.join(parentDir, 'skills')
154
+ if (fs.existsSync(rootSkillsDir)) {
155
+ config.skillsDirs.push(rootSkillsDir)
156
+ }
157
+
158
+ // const defaultSkillsDir = path.join(process.cwd(), 'defaultSkills')
159
+ // if (fs.existsSync(defaultSkillsDir)) {
160
+ // config.skillsDirs.push(defaultSkillsDir)
161
+ // }
162
+
163
+ return config
164
+ }
165
+
166
+ /**
167
+ * DefaultPlugins - 默认插件配置加载器
168
+ */
169
+ class DefaultPlugins extends Plugin {
170
+ constructor(config = {}) {
171
+ super()
172
+ this.name = 'defaults'
173
+ this.version = '1.0.0'
174
+ this.description = '默认插件配置加载器'
175
+ this.priority = 0 // 最先加载
176
+ this.system = true
177
+
178
+ this._framework = null
179
+ this._agentDir = config.agentDir || '.agent'
180
+ this._config = null
181
+ }
182
+
183
+ install(framework) {
184
+ this._framework = framework
185
+ return this
186
+ }
187
+
188
+ start(framework) {
189
+ // 加载配置
190
+ this._config = loadAgentConfig(this._agentDir)
191
+ return this
192
+ }
193
+
194
+ /**
195
+ * 获取加载的配置
196
+ */
197
+ getConfig() {
198
+ return this._config || loadAgentConfig(this._agentDir)
199
+ }
200
+
201
+ reload(framework) {
202
+ this._framework = framework
203
+ this._config = loadAgentConfig(this._agentDir)
204
+ }
205
+
206
+ uninstall(framework) {
207
+ this._framework = null
208
+ this._config = null
209
+ }
210
+ }
211
+
212
+ /**
213
+ * Bootstrap 辅助函数
214
+ * 根据配置加载所有默认插件
215
+ */
216
+ async function bootstrapDefaults(framework, config = {}) {
217
+ // 如果已经有配置,使用现有配置;否则加载
218
+ const agentConfig = config._config
219
+ if (!agentConfig) {
220
+ bootstrapLog.error(' No config provided, skipping plugin loading')
221
+ return
222
+ }
223
+ // 设置 bootstrap 模式,避免重复启动
224
+ framework.pluginManager.setBootstrapping(true)
225
+
226
+ // 合并 AI 配置
227
+ const aiConfig = { ...agentConfig.ai, ...config.aiConfig }
228
+
229
+ // 核心插件列表(不能禁用,必须加载)
230
+ const CORE_PLUGINS = new Set([
231
+ 'install', 'ai', 'storage', 'tools', 'workflow', 'skill-manager',
232
+ 'mcp-executor', 'shell-executor', 'python-executor', 'session', 'web',
233
+ 'audit', 'rules', 'scheduler', 'file-system', 'think', 'ambient',
234
+ 'python-plugin-loader', 'telegram', 'weixin', 'subagent-manager'
235
+ ])
236
+
237
+ // 辅助函数:检查插件是否应该加载
238
+ // 传入插件名称(字符串)、已创建的实例,或插件类
239
+ const shouldLoad = (plugin) => {
240
+ const name = typeof plugin === 'string' ? plugin : (plugin.name || plugin.prototype?.name)
241
+ if (framework.pluginManager.has(name)) {
242
+ framework._debug&&bootstrapLog.debug(` ${name} Plugin already loaded, skipping`)
243
+ return false
244
+ }
245
+ // 系统插件(system: true)不能禁用
246
+ let isSystem = false
247
+ if (typeof plugin === 'function') {
248
+ isSystem = plugin.prototype?.system === true
249
+ } else if (typeof plugin === 'object') {
250
+ isSystem = plugin.system === true
251
+ }
252
+ if (isSystem && framework.pluginManager.isEnabled(name) === false) {
253
+ framework._debug&&bootstrapLog.debug(` ${name} is a system plugin, cannot be disabled`)
254
+ }
255
+ return true
256
+ }
257
+
258
+ // 0. Install 工具插件(最先加载,让其他插件能用到它的 node_modules)
259
+ if (shouldLoad('install')) {
260
+ const { InstallPlugin } = require('./install-plugin')
261
+ await framework.loadPlugin(new InstallPlugin({ agentDir: agentConfig.agentDir }))
262
+ framework._debug&&bootstrapLog.debug(' Install Plugin loaded')
263
+ }
264
+
265
+ // 合并 skillsDirs 配置(bootstrap 传入的优先)
266
+ const skillsDirs = [
267
+ ...(config.skillsDirs || []),
268
+ ...(agentConfig.skillsDirs || [])
269
+ ]
270
+
271
+ framework._debug&&bootstrapLog.debug(' Loading default plugins...')
272
+ // AI 插件(如果已禁用则跳过)
273
+ if (!shouldLoad('ai') || !(aiConfig.provider || aiConfig.model || aiConfig.apiKey)) {
274
+ // 跳过或已禁用
275
+ } else {
276
+ const { AIPlugin } = require('./ai-plugin')
277
+ // 根据 provider 获取对应的 API key
278
+ const upperProvider = (aiConfig.provider || 'deepseek').toUpperCase().replace(/-/g, '_')
279
+ const envApiKey = process.env[`${upperProvider}_API_KEY`] || process.env.FOLIKO_API_KEY
280
+ const aiPlugin = new AIPlugin({
281
+ provider: aiConfig.provider || 'deepseek',
282
+ model: aiConfig.model || 'deepseek-chat',
283
+ apiKey: aiConfig.apiKey || envApiKey,
284
+ baseURL: aiConfig.baseURL
285
+ })
286
+ await framework.loadPlugin(aiPlugin)
287
+ framework._debug&&bootstrapLog.debug(' AI Plugin loaded')
288
+ }
289
+
290
+ // 1.5 创建主 Agent(供 Telegram 等需要绑定 Agent 的插件使用)
291
+ if (!framework._mainAgent && aiConfig.provider) {
292
+ const { Agent } = require('../src/core/agent')
293
+ const aiPlugin = framework.pluginManager.get('ai')
294
+ const aiClient = aiPlugin ? aiPlugin.getAIClient() : null
295
+ framework._debug&&bootstrapLog.debug(' Creating Main Agent - aiClient:', !!aiClient)
296
+
297
+ framework._mainAgent = framework.createAgent({
298
+ name: 'MainAgent',
299
+ systemPrompt: '你是一个智能助手。当用户提出问题或任务时,你会主动分析需求,选择合适的工具来获取信息或执行操作。你善于将复杂任务拆解为多个步骤,通过工具协作完成。',
300
+ model: aiConfig.model,
301
+ provider: aiConfig.provider,
302
+ apiKey: aiConfig.apiKey,
303
+ baseURL: aiConfig.baseURL
304
+ })
305
+ framework._agents.push(framework._mainAgent)
306
+ framework._debug&&bootstrapLog.debug(' Main Agent created, has _chatHandler:', !!framework._mainAgent._chatHandler)
307
+ }
308
+
309
+ // 2. Storage 存储插件
310
+ if (shouldLoad('storage')) {
311
+ const { StoragePlugin } = require('./storage-plugin')
312
+ await framework.loadPlugin(new StoragePlugin())
313
+ framework._debug&&bootstrapLog.debug(' Storage Plugin loaded')
314
+ }
315
+
316
+ // 3. 内置工具插件
317
+ if (shouldLoad('tools')) {
318
+ const { ToolsPlugin } = require('./tools-plugin')
319
+ await framework.loadPlugin(new ToolsPlugin())
320
+ framework._debug&&bootstrapLog.debug(' Tools Plugin loaded')
321
+ }
322
+
323
+ // 4. 工作流插件
324
+ if (shouldLoad('workflow')) {
325
+ const { WorkflowPlugin } = require('../src/capabilities/workflow-engine')
326
+ await framework.loadPlugin(new WorkflowPlugin())
327
+ framework._debug&&bootstrapLog.debug(' Workflow Plugin loaded')
328
+ }
329
+
330
+ // 5. Skill 管理器插件
331
+ if (shouldLoad('skill-manager')) {
332
+ if (skillsDirs.length > 0) {
333
+ const { SkillManagerPlugin } = require('../src/capabilities/skill-manager')
334
+ // 传递所有 skills 目录
335
+ await framework.loadPlugin(new SkillManagerPlugin({ skillsDirs }))
336
+ framework._debug&&bootstrapLog.debug(' Skill Manager loaded')
337
+ }
338
+ }
339
+
340
+ // 6. MCP 执行器插件(始终加载,确保 mcp_reload 工具可用)
341
+ if (shouldLoad('mcp-executor')) {
342
+ const mcpServers = Object.entries(agentConfig.mcpServers || {})
343
+ const servers = mcpServers.map(([name, cfg]) => ({
344
+ ...cfg,
345
+ name,
346
+ command: cfg.command,
347
+ args: cfg.args || [],
348
+ env: cfg.env || {},
349
+ url: cfg.url,
350
+ headers: cfg.headers
351
+ }))
352
+ const { MCPExecutorPlugin } = require('../src/executors/mcp-executor')
353
+ await framework.loadPlugin(new MCPExecutorPlugin({ servers }))
354
+ framework._debug&&bootstrapLog.debug(` MCP Executor loaded${servers.length > 0 ? ` (${servers.length} servers)` : ' (no servers)'}`)
355
+ }
356
+
357
+ // 7. Shell 执行器插件
358
+ if (shouldLoad('shell-executor')) {
359
+ const { ShellExecutorPlugin } = require('./shell-executor-plugin')
360
+ await framework.loadPlugin(new ShellExecutorPlugin())
361
+ framework._debug&&bootstrapLog.debug(' Shell Executor loaded')
362
+ }
363
+
364
+ // 8. Python 执行器插件
365
+ if (shouldLoad('python-executor')) {
366
+ const { PythonExecutorPlugin } = require('./python-executor-plugin')
367
+ await framework.loadPlugin(new PythonExecutorPlugin())
368
+ framework._debug&&bootstrapLog.debug(' Python Executor loaded')
369
+ }
370
+
371
+ // 8.5 Web Web服务插件
372
+ if (shouldLoad('web')) {
373
+ const { WebPlugin } = require('./web-plugin')
374
+ await framework.loadPlugin(new WebPlugin())
375
+ framework._debug&&bootstrapLog.debug(' Web Plugin loaded')
376
+ }
377
+
378
+ // 9. Session 会话管理插件
379
+ if (shouldLoad('session')) {
380
+ const { SessionPlugin } = require('./session-plugin')
381
+ await framework.loadPlugin(new SessionPlugin())
382
+ framework._debug&&bootstrapLog.debug(' Session Plugin loaded')
383
+ }
384
+
385
+ // 10. Audit 审计日志插件
386
+ if (shouldLoad('audit')) {
387
+ const { AuditPlugin } = require('./audit-plugin')
388
+ await framework.loadPlugin(new AuditPlugin())
389
+ framework._debug&&bootstrapLog.debug(' Audit Plugin loaded')
390
+ }
391
+
392
+ // 10. Rules 规则引擎插件
393
+ if (shouldLoad('rules')) {
394
+ const { RulesPlugin } = require('./rules-plugin')
395
+ await framework.loadPlugin(new RulesPlugin())
396
+ framework._debug&&bootstrapLog.debug(' Rules Plugin loaded')
397
+ }
398
+
399
+ // 11. Scheduler 定时任务插件
400
+ if (shouldLoad('scheduler')) {
401
+ const { SchedulerPlugin } = require('./scheduler-plugin')
402
+ await framework.loadPlugin(new SchedulerPlugin())
403
+ framework._debug&&bootstrapLog.debug(' Scheduler Plugin loaded')
404
+ }
405
+
406
+ // 11. FileSystem 文件系统插件
407
+ if (shouldLoad('file-system')) {
408
+ const { FileSystemPlugin } = require('./file-system-plugin')
409
+ await framework.loadPlugin(new FileSystemPlugin())
410
+ framework._debug&&bootstrapLog.debug(' FileSystem Plugin loaded')
411
+ }
412
+
413
+ // 12. Think 主动思考插件
414
+ if (shouldLoad('think')) {
415
+ const { ThinkPlugin } = require('./think-plugin')
416
+ await framework.loadPlugin(new ThinkPlugin())
417
+ framework._debug&&bootstrapLog.debug(' Think Plugin loaded')
418
+ }
419
+
420
+ // 12.1 Ambient Agent 插件
421
+ if (shouldLoad('ambient')) {
422
+ const { AmbientAgentPlugin } = require('./ambient-agent')
423
+ await framework.loadPlugin(new AmbientAgentPlugin())
424
+ framework._debug&&bootstrapLog.debug(' Ambient Agent Plugin loaded')
425
+ }
426
+
427
+ // 12.5 Python 插件加载器
428
+ if (shouldLoad('python-plugin-loader')) {
429
+ const { PythonPluginLoader } = require('./python-plugin-loader')
430
+ await framework.loadPlugin(new PythonPluginLoader({ agentDir: agentConfig.agentDir }))
431
+ framework._debug&&bootstrapLog.debug(' Python Plugin Loader loaded')
432
+ }
433
+
434
+ // 12.6 Telegram 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
435
+ if (shouldLoad('telegram')) {
436
+ const { TelegramPlugin } = require('./telegram-plugin')
437
+ const telegramConfig = {
438
+ botToken: process.env.TELEGRAM_BOT_TOKEN || agentConfig.telegram?.botToken
439
+ }
440
+ await framework.loadPlugin(new TelegramPlugin(telegramConfig))
441
+ }
442
+
443
+ // 12.7 WeChat 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
444
+ if (shouldLoad('weixin')) {
445
+ try {
446
+ const { WeixinPlugin } = require('./weixin-plugin')
447
+ const weixinConfig = {
448
+ forceLogin: agentConfig.weixin?.forceLogin || false,
449
+ qrcodeTerminal: agentConfig.weixin?.qrcodeTerminal !== false,
450
+ allowedUsers: agentConfig.weixin?.allowedUsers || []
451
+ }
452
+ await framework.loadPlugin(new WeixinPlugin(weixinConfig))
453
+ } catch (err) {
454
+ bootstrapLog.warn(' WeChat Plugin not available:', err.message)
455
+ }
456
+ }
457
+
458
+ // 12.8 Email 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
459
+ if (shouldLoad('email')) {
460
+ try {
461
+ const { Plugin } = require('../src/core/plugin-base')
462
+ const { logger } = require('../src/utils/logger')
463
+ const log = logger.child('AgentConfig')
464
+ const bootstrapLog = logger.child('Bootstrap')
465
+ const { EmailPlugin } = require('./email')
466
+ // 支持两种格式:email.smtp 或 email.config.smtp
467
+ const emailData = agentConfig.email || {}
468
+ const emailCfg = emailData.config || {}
469
+ const emailConfig = {
470
+ smtp: emailCfg.smtp || emailData.smtp || {},
471
+ imap: emailCfg.imap || emailData.imap || {},
472
+ clientId: emailCfg.clientId || emailData.clientId
473
+ }
474
+ await framework.loadPlugin(new EmailPlugin(emailConfig))
475
+ } catch (err) {
476
+ bootstrapLog.warn(' Email Plugin not available:', err.message)
477
+ }
478
+ }
479
+
480
+ // 12.9 Feishu 插件(默认禁用,需要在 plugins.json 中设置 enabled: true)
481
+ if (shouldLoad('feishu')) {
482
+ try {
483
+ const { FeishuPlugin } = require('./feishu-plugin')
484
+ const feishuConfig = {
485
+ allowedUsers: agentConfig.feishu?.allowedUsers || []
486
+ }
487
+ await framework.loadPlugin(new FeishuPlugin(feishuConfig))
488
+ } catch (err) {
489
+ bootstrapLog.warn(' Feishu Plugin not available:', err.message)
490
+ }
491
+ }
492
+
493
+ // 12.10 SubAgent 管理器
494
+ if (shouldLoad('subagent-manager')) {
495
+ try {
496
+ // 先初始化 SubAgentConfigManager 并加载所有配置
497
+ const { SubAgentConfigManager } = require('../src/core/sub-agent-config')
498
+ const subAgentConfigManager = new SubAgentConfigManager(agentConfig.agentsDir)
499
+ subAgentConfigManager.loadAll()
500
+ framework._subAgentConfigManager = subAgentConfigManager
501
+
502
+ // 然后加载 SubAgentManagerPlugin
503
+ const { SubAgentManagerPlugin } = require('./subagent-plugin')
504
+ await framework.loadPlugin(new SubAgentManagerPlugin({ agentsDir: agentConfig.agentsDir }))
505
+ framework._debug&&bootstrapLog.debug(' SubAgent Manager loaded')
506
+ } catch (err) {
507
+ bootstrapLog.warn(' SubAgent Manager failed:', err.message)
508
+ }
509
+ }
510
+
511
+ // 13. 加载自定义插件
512
+ await loadCustomPlugins(framework, agentConfig)
513
+
514
+ framework._debug&&bootstrapLog.debug(' All plugins loaded')
515
+
516
+ // 统一启动所有插件(避免重复启动)
517
+ await framework.pluginManager.startAll()
518
+
519
+ // 清除 bootstrap 模式
520
+ framework.pluginManager.setBootstrapping(false)
521
+
522
+ framework._debug&&bootstrapLog.debug(' Loaded plugins ', framework.pluginManager.getAll().length)
523
+ framework._debug&&bootstrapLog.debug(' Loaded tools ', framework.getTools().length)
524
+ }
525
+
526
+ /**
527
+ * 解析插件路径
528
+ * 支持两种结构:
529
+ * 1. 文件夹结构: .agent/plugins/my-plugin/index.js
530
+ * 2. 单文件结构: .agent/plugins/my-plugin.js
531
+ * @param {string} pluginsDir - 插件目录
532
+ * @param {string} name - 插件名称
533
+ * @returns {{path: string, type: 'folder'|'file'}|null} 插件路径和类型
534
+ */
535
+ function resolvePluginPath(pluginsDir, name) {
536
+ const folderPath = path.join(pluginsDir, name)
537
+ const filePath = path.join(pluginsDir, `${name}.js`)
538
+
539
+ // 文件夹优先
540
+ if (fs.existsSync(folderPath) && fs.statSync(folderPath).isDirectory()) {
541
+ const pkgPath = path.join(folderPath, 'package.json')
542
+ if (fs.existsSync(pkgPath)) {
543
+ try {
544
+ const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf-8'))
545
+ const main = pkg.main || 'index.js'
546
+ const mainPath = path.join(folderPath, main)
547
+ if (fs.existsSync(mainPath)) {
548
+ return { path: mainPath, type: 'folder' }
549
+ }
550
+ } catch (err) {
551
+ log.warn(` Failed to parse package.json for ${name}:`, err.message)
552
+ }
553
+ }
554
+ // 默认加载 index.js
555
+ const indexPath = path.join(folderPath, 'index.js')
556
+ if (fs.existsSync(indexPath)) {
557
+ return { path: indexPath, type: 'folder' }
558
+ }
559
+ //log.warn(` No entry point found for plugin folder: ${name}`)
560
+ return null
561
+ }
562
+
563
+ // 单文件回退
564
+ if (fs.existsSync(filePath)) {
565
+ return { path: filePath, type: 'file' }
566
+ }
567
+
568
+ return null
569
+ }
570
+
571
+ /**
572
+ * 扫描插件目录,返回所有插件名称
573
+ * @param {string} pluginsDir - 插件目录
574
+ * @returns {string[]} 插件名称列表
575
+ */
576
+ function scanPluginNames(pluginsDir) {
577
+ if (!fs.existsSync(pluginsDir)) {
578
+ return []
579
+ }
580
+
581
+ const names = new Set()
582
+ const entries = fs.readdirSync(pluginsDir, { withFileTypes: true })
583
+
584
+ for (const entry of entries) {
585
+ if (entry.isDirectory()) {
586
+ // 文件夹插件
587
+ names.add(entry.name)
588
+ } else if (entry.isFile() && entry.name.endsWith('.js')) {
589
+ // 单文件插件(排除与文件夹同名的)
590
+ const baseName = entry.name.replace(/\.js$/, '')
591
+ if (!names.has(baseName)) {
592
+ names.add(baseName)
593
+ }
594
+ }
595
+ }
596
+
597
+ return Array.from(names)
598
+ }
599
+
600
+ async function loadCustomPlugins(framework, agentConfig) {
601
+ // 加载 .agent/plugins 目录下的自定义插件
602
+ const pluginsDir = path.resolve(process.cwd(), '.agent', 'plugins')
603
+
604
+ if (!fs.existsSync(pluginsDir)) {
605
+ return
606
+ }
607
+
608
+ // 扫描所有插件名称(支持文件夹和单文件)
609
+ const pluginNames = scanPluginNames(pluginsDir)
610
+
611
+ for (const pluginName of pluginNames) {
612
+ try {
613
+ const resolved = resolvePluginPath(pluginsDir, pluginName)
614
+ if (!resolved) {
615
+ //log.warn(` Cannot resolve plugin: ${pluginName}`)
616
+ continue
617
+ }
618
+
619
+ const { path: pluginPath, type } = resolved
620
+
621
+ // 清除缓存
622
+ delete require.cache[require.resolve(pluginPath)]
623
+ const pluginModule = require(pluginPath)
624
+
625
+ let plugin
626
+ if (typeof pluginModule === 'function') {
627
+ // 免引入写法:直接传递函数,让 loadPlugin 内部处理
628
+ plugin = pluginModule
629
+ } else if (pluginModule.default) {
630
+ plugin = pluginModule.default
631
+ } else {
632
+ plugin = pluginModule
633
+ }
634
+
635
+ // 获取插件名称
636
+ const tempPlugin = typeof plugin === 'function' && plugin.prototype?.name
637
+ ? new plugin() : plugin
638
+ const resolvedPluginName = tempPlugin.name || pluginName
639
+
640
+ // 如果插件已加载且已启动,跳过
641
+ if (framework.pluginManager.has(resolvedPluginName) &&
642
+ framework.pluginManager.get(resolvedPluginName)?._started) {
643
+ continue
644
+ }
645
+
646
+ //bootstrapLog.debug(` Loading custom plugin: ${pluginName} (${type})`)
647
+ // .agent/plugins 目录下的插件强制启用,不受 state 文件 enabled: false 影响
648
+ await framework.pluginManager.load(plugin, { forceEnabled: true })
649
+ } catch (err) {
650
+ log.error(` Failed to load plugin ${pluginName}:`, err.message)
651
+ }
652
+ }
653
+ }
654
+
655
+ module.exports = {
656
+ DefaultPlugins,
657
+ loadAgentConfig,
658
+ bootstrapDefaults,
659
+ loadCustomPlugins,
660
+ resolvePluginPath,
661
+ scanPluginNames
662
+ }