foliko 1.0.74 → 1.0.76
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.agent/.shared/ui-ux-pro-max/data/charts.csv +26 -0
- package/.agent/.shared/ui-ux-pro-max/data/colors.csv +97 -0
- package/.agent/.shared/ui-ux-pro-max/data/icons.csv +101 -0
- package/.agent/.shared/ui-ux-pro-max/data/landing.csv +31 -0
- package/.agent/.shared/ui-ux-pro-max/data/products.csv +97 -0
- package/.agent/.shared/ui-ux-pro-max/data/prompts.csv +24 -0
- package/.agent/.shared/ui-ux-pro-max/data/react-performance.csv +45 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/flutter.csv +53 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/jetpack-compose.csv +53 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/nextjs.csv +53 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/react-native.csv +52 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/react.csv +54 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/shadcn.csv +61 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/svelte.csv +54 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/swiftui.csv +51 -0
- package/.agent/.shared/ui-ux-pro-max/data/stacks/vue.csv +50 -0
- package/.agent/.shared/ui-ux-pro-max/data/styles.csv +59 -0
- package/.agent/.shared/ui-ux-pro-max/data/typography.csv +58 -0
- package/.agent/.shared/ui-ux-pro-max/data/ui-reasoning.csv +101 -0
- package/.agent/.shared/ui-ux-pro-max/data/ux-guidelines.csv +100 -0
- package/.agent/.shared/ui-ux-pro-max/data/web-interface.csv +31 -0
- package/.agent/.shared/ui-ux-pro-max/scripts/__pycache__/core.cpython-313.pyc +0 -0
- package/.agent/.shared/ui-ux-pro-max/scripts/__pycache__/design_system.cpython-313.pyc +0 -0
- package/.agent/.shared/ui-ux-pro-max/scripts/core.py +258 -0
- package/.agent/.shared/ui-ux-pro-max/scripts/design_system.py +1067 -0
- package/.agent/.shared/ui-ux-pro-max/scripts/search.py +106 -0
- package/.agent/ARCHITECTURE.md +288 -0
- package/.agent/agents/ambient-agent.md +57 -0
- package/.agent/agents/debugger.md +55 -0
- package/.agent/agents/email-assistant.md +49 -0
- package/.agent/agents/file-manager.md +42 -0
- package/.agent/agents/python-developer.md +60 -0
- package/.agent/agents/scheduler.md +59 -0
- package/.agent/agents/web-developer.md +45 -0
- package/.agent/data/default.json +29 -0
- package/.agent/data/plugins-state.json +255 -0
- package/.agent/mcp_config.json +4 -0
- package/.agent/mcp_config_updated.json +12 -0
- package/.agent/plugins.json +5 -0
- package/.agent/rules/GEMINI.md +273 -0
- package/.agent/rules/allow-rule.md +77 -0
- package/.agent/rules/log-rule.md +83 -0
- package/.agent/rules/security-rule.md +93 -0
- package/.agent/scripts/auto_preview.py +148 -0
- package/.agent/scripts/checklist.py +217 -0
- package/.agent/scripts/session_manager.py +120 -0
- package/.agent/scripts/verify_all.py +327 -0
- package/.agent/skills/api-patterns/SKILL.md +81 -0
- package/.agent/skills/api-patterns/api-style.md +42 -0
- package/.agent/skills/api-patterns/auth.md +24 -0
- package/.agent/skills/api-patterns/documentation.md +26 -0
- package/.agent/skills/api-patterns/graphql.md +41 -0
- package/.agent/skills/api-patterns/rate-limiting.md +31 -0
- package/.agent/skills/api-patterns/response.md +37 -0
- package/.agent/skills/api-patterns/rest.md +40 -0
- package/.agent/skills/api-patterns/scripts/api_validator.py +211 -0
- package/.agent/skills/api-patterns/security-testing.md +122 -0
- package/.agent/skills/api-patterns/trpc.md +41 -0
- package/.agent/skills/api-patterns/versioning.md +22 -0
- package/.agent/skills/app-builder/SKILL.md +75 -0
- package/.agent/skills/app-builder/agent-coordination.md +71 -0
- package/.agent/skills/app-builder/feature-building.md +53 -0
- package/.agent/skills/app-builder/project-detection.md +34 -0
- package/.agent/skills/app-builder/scaffolding.md +118 -0
- package/.agent/skills/app-builder/tech-stack.md +40 -0
- package/.agent/skills/app-builder/templates/SKILL.md +39 -0
- package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -0
- package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -0
- package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -0
- package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -0
- package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -0
- package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -0
- package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -0
- package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +122 -0
- package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +122 -0
- package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +169 -0
- package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +134 -0
- package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -0
- package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +119 -0
- package/.agent/skills/architecture/SKILL.md +55 -0
- package/.agent/skills/architecture/context-discovery.md +43 -0
- package/.agent/skills/architecture/examples.md +94 -0
- package/.agent/skills/architecture/pattern-selection.md +68 -0
- package/.agent/skills/architecture/patterns-reference.md +50 -0
- package/.agent/skills/architecture/trade-off-analysis.md +77 -0
- package/.agent/skills/clean-code/SKILL.md +201 -0
- package/.agent/skills/doc.md +177 -0
- package/.agent/skills/frontend-design/SKILL.md +418 -0
- package/.agent/skills/frontend-design/animation-guide.md +331 -0
- package/.agent/skills/frontend-design/color-system.md +311 -0
- package/.agent/skills/frontend-design/decision-trees.md +418 -0
- package/.agent/skills/frontend-design/motion-graphics.md +306 -0
- package/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -0
- package/.agent/skills/frontend-design/scripts/ux_audit.py +722 -0
- package/.agent/skills/frontend-design/typography-system.md +345 -0
- package/.agent/skills/frontend-design/ux-psychology.md +1116 -0
- package/.agent/skills/frontend-design/visual-effects.md +383 -0
- package/.agent/skills/i18n-localization/SKILL.md +154 -0
- package/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -0
- package/.agent/skills/mcp-builder/SKILL.md +176 -0
- package/.agent/skills/web-design-guidelines/SKILL.md +57 -0
- package/.agent/workflows/brainstorm.md +113 -0
- package/.agent/workflows/create.md +59 -0
- package/.agent/workflows/debug.md +103 -0
- package/.agent/workflows/deploy.md +176 -0
- package/.agent/workflows/enhance.md +63 -0
- package/.agent/workflows/orchestrate.md +237 -0
- package/.agent/workflows/plan.md +89 -0
- package/.agent/workflows/preview.md +81 -0
- package/.agent/workflows/simple-test.md +42 -0
- package/.agent/workflows/status.md +86 -0
- package/.agent/workflows/structured-orchestrate.md +180 -0
- package/.agent/workflows/test.md +144 -0
- package/.agent/workflows/ui-ux-pro-max.md +296 -0
- package/.claude/settings.local.json +11 -1
- package/.editorconfig +56 -0
- package/.husky/pre-commit +4 -0
- package/.lintstagedrc +7 -0
- package/.prettierignore +29 -0
- package/.prettierrc +11 -0
- package/CLAUDE.md +2 -0
- package/README.md +64 -55
- package/SPEC.md +102 -61
- package/cli/bin/foliko.js +11 -11
- package/cli/src/commands/chat.js +143 -141
- package/cli/src/commands/list.js +93 -90
- package/cli/src/index.js +75 -75
- package/cli/src/ui/chat-ui.js +201 -199
- package/cli/src/utils/ansi.js +40 -40
- package/cli/src/utils/markdown.js +292 -296
- package/docker-compose.yml +1 -1
- package/docs/ai-sdk-optimization.md +655 -643
- package/docs/features.md +80 -80
- package/docs/quick-reference.md +49 -46
- package/docs/user-manual.md +411 -380
- package/examples/ambient-example.js +194 -196
- package/examples/basic.js +50 -45
- package/examples/bootstrap.js +121 -112
- package/examples/mcp-example.js +19 -16
- package/examples/skill-example.js +20 -20
- package/examples/test-chat.js +137 -135
- package/examples/test-mcp.js +85 -79
- package/examples/test-reload.js +59 -61
- package/examples/test-telegram.js +50 -50
- package/examples/test-tg-bot.js +45 -42
- package/examples/test-tg-simple.js +47 -46
- package/examples/test-tg.js +62 -62
- package/examples/test-think.js +43 -37
- package/examples/test-web-plugin.js +103 -98
- package/examples/test-weixin-feishu.js +103 -100
- package/examples/workflow.js +158 -158
- package/package.json +37 -3
- package/plugins/ai-plugin.js +102 -100
- package/plugins/ambient-agent/EventWatcher.js +113 -0
- package/plugins/ambient-agent/ExplorerLoop.js +640 -0
- package/plugins/ambient-agent/GoalManager.js +197 -0
- package/plugins/ambient-agent/Reflector.js +95 -0
- package/plugins/ambient-agent/StateStore.js +90 -0
- package/plugins/ambient-agent/constants.js +101 -0
- package/plugins/ambient-agent/index.js +579 -0
- package/plugins/audit-plugin.js +187 -187
- package/plugins/default-plugins.js +662 -649
- package/plugins/email/constants.js +64 -0
- package/plugins/email/handlers.js +461 -0
- package/plugins/email/index.js +278 -0
- package/plugins/email/monitor.js +269 -0
- package/plugins/email/parser.js +138 -0
- package/plugins/email/reply.js +151 -0
- package/plugins/email/utils.js +124 -0
- package/plugins/feishu-plugin.js +481 -477
- package/plugins/file-system-plugin.js +826 -476
- package/plugins/install-plugin.js +199 -197
- package/plugins/python-executor-plugin.js +367 -365
- package/plugins/python-plugin-loader.js +481 -479
- package/plugins/rules-plugin.js +294 -292
- package/plugins/scheduler-plugin.js +691 -689
- package/plugins/session-plugin.js +369 -367
- package/plugins/shell-executor-plugin.js +197 -197
- package/plugins/storage-plugin.js +240 -238
- package/plugins/subagent-plugin.js +845 -785
- package/plugins/telegram-plugin.js +482 -475
- package/plugins/think-plugin.js +345 -343
- package/plugins/tools-plugin.js +196 -194
- package/plugins/web-plugin.js +606 -604
- package/plugins/weixin-plugin.js +545 -538
- package/reports/system-health-report-20260401.md +79 -0
- package/skills/ambient-agent/SKILL.md +49 -39
- package/skills/foliko-dev/AGENTS.md +64 -61
- package/skills/foliko-dev/SKILL.md +125 -119
- package/skills/mcp-usage/SKILL.md +19 -17
- package/skills/python-plugin-dev/SKILL.md +16 -15
- package/skills/skill-guide/SKILL.md +12 -12
- package/skills/subagent-guide/SKILL.md +237 -0
- package/skills/workflow-guide/SKILL.md +90 -45
- package/skills/workflow-troubleshooting/DEBUGGING.md +36 -21
- package/skills/workflow-troubleshooting/SKILL.md +156 -79
- package/src/capabilities/index.js +11 -11
- package/src/capabilities/skill-manager.js +609 -595
- package/src/capabilities/workflow-engine.js +1109 -1195
- package/src/core/agent-chat.js +882 -735
- package/src/core/agent.js +892 -688
- package/src/core/framework.js +465 -431
- package/src/core/index.js +19 -19
- package/src/core/plugin-base.js +219 -219
- package/src/core/plugin-manager.js +863 -767
- package/src/core/provider.js +114 -111
- package/src/core/sub-agent-config.js +264 -0
- package/src/core/system-prompt-builder.js +120 -0
- package/src/core/tool-registry.js +517 -134
- package/src/core/tool-router.js +297 -216
- package/src/executors/executor-base.js +12 -12
- package/src/executors/mcp-executor.js +741 -729
- package/src/index.js +25 -37
- package/src/utils/circuit-breaker.js +301 -0
- package/src/utils/error-boundary.js +363 -0
- package/src/utils/error.js +374 -0
- package/src/utils/event-emitter.js +97 -97
- package/src/utils/id.js +133 -0
- package/src/utils/index.js +217 -3
- package/src/utils/logger.js +181 -0
- package/src/utils/plugin-helpers.js +90 -0
- package/src/utils/retry.js +122 -0
- package/src/utils/sandbox.js +292 -0
- package/test/tool-registry-validation.test.js +218 -0
- package/test_report.md +70 -0
- package/website/docs/api.html +169 -107
- package/website/docs/configuration.html +296 -144
- package/website/docs/plugin-development.html +154 -85
- package/website/docs/project-structure.html +110 -109
- package/website/docs/skill-development.html +117 -61
- package/website/index.html +209 -205
- package/website/script.js +136 -133
- package/website/styles.css +1 -1
- package/plugins/ambient-agent-plugin.js +0 -1565
- package/plugins/email.js +0 -1142
|
@@ -1,649 +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
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
const
|
|
50
|
-
if (
|
|
51
|
-
|
|
52
|
-
const
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
//
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
this.
|
|
173
|
-
this.
|
|
174
|
-
|
|
175
|
-
this.
|
|
176
|
-
this.
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
//
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
//
|
|
227
|
-
const
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
'
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
if (
|
|
250
|
-
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
//
|
|
275
|
-
|
|
276
|
-
const
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
}
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
const
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
}
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
if (
|
|
565
|
-
return
|
|
566
|
-
}
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
}
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
plugin
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
const
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
}
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
}
|
|
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
|
+
}
|