foliko 1.0.81 → 1.0.83

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/cli/bin/foliko.js +12 -12
  2. package/cli/src/commands/chat.js +143 -143
  3. package/cli/src/commands/list.js +93 -93
  4. package/cli/src/index.js +75 -75
  5. package/cli/src/ui/chat-ui.js +201 -201
  6. package/cli/src/utils/ansi.js +40 -40
  7. package/cli/src/utils/markdown.js +292 -292
  8. package/examples/ambient-example.js +194 -194
  9. package/examples/basic.js +115 -115
  10. package/examples/bootstrap.js +121 -121
  11. package/examples/mcp-example.js +56 -56
  12. package/examples/skill-example.js +49 -49
  13. package/examples/test-chat.js +137 -137
  14. package/examples/test-mcp.js +85 -85
  15. package/examples/test-reload.js +59 -59
  16. package/examples/test-telegram.js +50 -50
  17. package/examples/test-tg-bot.js +45 -45
  18. package/examples/test-tg-simple.js +47 -47
  19. package/examples/test-tg.js +62 -62
  20. package/examples/test-think.js +43 -43
  21. package/examples/test-web-plugin.js +103 -103
  22. package/examples/test-weixin-feishu.js +103 -103
  23. package/examples/workflow.js +158 -158
  24. package/package.json +83 -83
  25. package/plugins/ai-plugin.js +102 -102
  26. package/plugins/ambient-agent/EventWatcher.js +113 -113
  27. package/plugins/ambient-agent/ExplorerLoop.js +640 -640
  28. package/plugins/ambient-agent/GoalManager.js +197 -197
  29. package/plugins/ambient-agent/Reflector.js +95 -95
  30. package/plugins/ambient-agent/StateStore.js +90 -90
  31. package/plugins/ambient-agent/constants.js +101 -101
  32. package/plugins/ambient-agent/index.js +579 -579
  33. package/plugins/audit-plugin.js +187 -187
  34. package/plugins/default-plugins.js +548 -548
  35. package/plugins/email/constants.js +64 -64
  36. package/plugins/email/handlers.js +461 -461
  37. package/plugins/email/index.js +278 -278
  38. package/plugins/email/monitor.js +269 -269
  39. package/plugins/email/parser.js +138 -138
  40. package/plugins/email/reply.js +151 -151
  41. package/plugins/email/utils.js +124 -124
  42. package/plugins/extension-executor-plugin.js +326 -326
  43. package/plugins/feishu-plugin.js +481 -481
  44. package/plugins/file-system-plugin.js +920 -920
  45. package/plugins/gate-trading.js +747 -747
  46. package/plugins/install-plugin.js +199 -199
  47. package/plugins/python-executor-plugin.js +367 -367
  48. package/plugins/python-plugin-loader.js +651 -651
  49. package/plugins/rules-plugin.js +294 -294
  50. package/plugins/scheduler-plugin.js +691 -691
  51. package/plugins/session-plugin.js +494 -494
  52. package/plugins/shell-executor-plugin.js +197 -197
  53. package/plugins/storage-plugin.js +263 -263
  54. package/plugins/subagent-plugin.js +845 -845
  55. package/plugins/telegram-plugin.js +482 -482
  56. package/plugins/think-plugin.js +345 -345
  57. package/plugins/tools-plugin.js +196 -196
  58. package/plugins/web-plugin.js +637 -637
  59. package/plugins/weixin-plugin.js +545 -545
  60. package/src/capabilities/index.js +11 -11
  61. package/src/capabilities/skill-manager.js +609 -609
  62. package/src/capabilities/workflow-engine.js +1109 -1109
  63. package/src/core/agent-chat.js +1 -1
  64. package/src/core/agent.js +958 -958
  65. package/src/core/framework.js +465 -465
  66. package/src/core/index.js +19 -19
  67. package/src/core/plugin-base.js +262 -262
  68. package/src/core/plugin-manager.js +863 -863
  69. package/src/core/provider.js +114 -114
  70. package/src/core/sub-agent-config.js +264 -264
  71. package/src/core/system-prompt-builder.js +120 -120
  72. package/src/core/tool-registry.js +517 -517
  73. package/src/core/tool-router.js +297 -297
  74. package/src/executors/executor-base.js +58 -58
  75. package/src/executors/mcp-executor.js +845 -845
  76. package/src/index.js +25 -25
  77. package/src/utils/circuit-breaker.js +301 -301
  78. package/src/utils/error-boundary.js +363 -363
  79. package/src/utils/error.js +374 -374
  80. package/src/utils/event-emitter.js +97 -97
  81. package/src/utils/id.js +133 -133
  82. package/src/utils/index.js +217 -217
  83. package/src/utils/logger.js +181 -181
  84. package/src/utils/plugin-helpers.js +90 -90
  85. package/src/utils/retry.js +122 -122
  86. package/src/utils/sandbox.js +292 -292
  87. package/test/tool-registry-validation.test.js +218 -218
  88. package/website/script.js +136 -136
package/cli/bin/foliko.js CHANGED
@@ -1,12 +1,12 @@
1
- #!/usr/bin/env node
2
- /**
3
- * Foliko CLI 入口
4
- * Usage: foliko <command> [options]
5
- */
6
-
7
- // 加载 dotenv
8
- require('dotenv').config();
9
-
10
- const { cli } = require('../src');
11
-
12
- cli();
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Foliko CLI 入口
4
+ * Usage: foliko <command> [options]
5
+ */
6
+
7
+ // 加载 dotenv
8
+ require('dotenv').config();
9
+
10
+ const { cli } = require('../src');
11
+
12
+ cli();
@@ -1,143 +1,143 @@
1
- /**
2
- * Chat 命令实现
3
- */
4
-
5
- const path = require('path');
6
- const dotenv = require('dotenv');
7
- const { Framework } = require('../../../src');
8
- const { ChatUI } = require('../ui/chat-ui');
9
-
10
- // 加载 .env 文件
11
- dotenv.config();
12
-
13
- // 默认配置
14
- const DEFAULT_CONFIG = {
15
- model: 'MiniMax-M2.7',
16
- provider: 'minimax',
17
- baseURL: 'https://api.minimaxi.com/v1',
18
- apiKey: null,
19
- };
20
-
21
- // Provider 默认配置
22
- const PROVIDER_DEFAULTS = {
23
- minimax: {
24
- model: 'MiniMax-M2.7',
25
- baseURL: 'https://api.minimaxi.com/v1',
26
- },
27
- deepseek: {
28
- model: 'deepseek-chat',
29
- baseURL: 'https://api.deepseek.com/v1',
30
- },
31
- };
32
-
33
- /**
34
- * 获取环境变量配置
35
- */
36
- function getEnvConfig() {
37
- const provider = process.env.FOLIKO_PROVIDER || DEFAULT_CONFIG.provider;
38
- const providerDefaults = PROVIDER_DEFAULTS[provider] || PROVIDER_DEFAULTS.minimax;
39
-
40
- // 支持多种 API key 环境变量名
41
- let apiKey = process.env.FOLIKO_API_KEY || null;
42
- if (!apiKey) {
43
- // 根据 provider 查找对应的 API key
44
- const upperProvider = provider.toUpperCase().replace(/-/g, '_');
45
- apiKey = process.env[`${upperProvider}_API_KEY`] || null;
46
- }
47
-
48
- return {
49
- model: process.env.FOLIKO_MODEL || providerDefaults.model,
50
- provider: provider,
51
- baseURL: process.env.FOLIKO_BASE_URL || providerDefaults.baseURL,
52
- apiKey: apiKey,
53
- };
54
- }
55
-
56
- /**
57
- * 解析命令行参数(命令行参数优先于环境变量)
58
- */
59
- function parseArgs(args) {
60
- // 先获取环境变量配置(作为默认值)
61
- const envConfig = getEnvConfig();
62
- const options = { ...envConfig };
63
-
64
- for (let i = 0; i < args.length; i++) {
65
- const arg = args[i];
66
- if (arg === '--model' && args[i + 1]) {
67
- options.model = args[++i];
68
- } else if (arg === '--provider' && args[i + 1]) {
69
- options.provider = args[++i];
70
- } else if (arg === '--base-url' && args[i + 1]) {
71
- options.baseURL = args[++i];
72
- } else if (arg === '--api-key' && args[i + 1]) {
73
- options.apiKey = args[++i];
74
- }
75
- }
76
-
77
- return options;
78
- }
79
-
80
- /**
81
- * Chat 命令入口
82
- */
83
- async function chatCommand(args) {
84
- const options = parseArgs(args);
85
- console.log('=== Foliko 持续对话 ===\n');
86
- console.log('输入 exit 或 quit 退出\n');
87
- // 初始化框架
88
- const framework = new Framework({ debug: false });
89
- await framework.bootstrap({
90
- agentDir: process.cwd() + '/.agent',
91
- aiConfig: {
92
- provider: options.provider,
93
- model: options.model,
94
- baseURL: options.baseURL,
95
- apiKey: options.apiKey,
96
- providerOptions: {
97
- maxOutputTokens: 8192,
98
- },
99
- },
100
- });
101
-
102
- // 创建 Agent
103
- const agent = framework.createAgent({
104
- name: 'FolikoAgent',
105
- systemPrompt: `你是一个有帮助的助手,擅长回答问题和执行任务。
106
- **重要:** 子Agent 匹配规则必须遵守:
107
- - 根据【子 Agent 匹配表】,将任务委托给最匹配的子Agent处理
108
- - 使用 subagent_call 工具并指定 agentName 来委托任务
109
- - 只有当没有匹配的子Agent时,才直接调用工具
110
- **命令执行规范:**
111
- - 执行 npx skills add / npx skills remove / npx skills list 命令时,必须自动添加参数:-a openclaw -y
112
- - 禁止添加 -g 参数
113
- - 例如:npx skills add xxx 应该执行为 npx skills add xxx -a openclaw -y`,
114
- sharedPrompt: `工作目录: {{WORK_DIR}}`,
115
- metadata: {
116
- WORK_DIR: process.cwd(), // 覆盖内置的 WORK_DIR
117
- },
118
- });
119
-
120
- // 初始化 UI
121
- const ui = new ChatUI(agent);
122
-
123
- // 监听通知事件,在 CLI 中显示
124
- // 显示当前会话的通知,以及没有指定 sessionId 的广播通知
125
- framework.on('notification', (data) => {
126
- const { title, message, source, timestamp, sessionId } = data;
127
- // 如果通知指定了 sessionId 且不匹配 CLI 会话,跳过
128
- if (sessionId && sessionId !== ui.sessionId) {
129
- return;
130
- }
131
- const time = timestamp ? new Date(timestamp).toLocaleTimeString('zh-CN') : '';
132
- console.log('\n' + '='.repeat(50));
133
- console.log(`🔔 [${source}] ${title}`);
134
- console.log(message);
135
- if (time) console.log(`时间: ${time}`);
136
- console.log('='.repeat(50) + '\n');
137
- });
138
-
139
- // 启动聊天
140
- ui.start();
141
- }
142
-
143
- module.exports = { chatCommand };
1
+ /**
2
+ * Chat 命令实现
3
+ */
4
+
5
+ const path = require('path');
6
+ const dotenv = require('dotenv');
7
+ const { Framework } = require('../../../src');
8
+ const { ChatUI } = require('../ui/chat-ui');
9
+
10
+ // 加载 .env 文件
11
+ dotenv.config();
12
+
13
+ // 默认配置
14
+ const DEFAULT_CONFIG = {
15
+ model: 'MiniMax-M2.7',
16
+ provider: 'minimax',
17
+ baseURL: 'https://api.minimaxi.com/v1',
18
+ apiKey: null,
19
+ };
20
+
21
+ // Provider 默认配置
22
+ const PROVIDER_DEFAULTS = {
23
+ minimax: {
24
+ model: 'MiniMax-M2.7',
25
+ baseURL: 'https://api.minimaxi.com/v1',
26
+ },
27
+ deepseek: {
28
+ model: 'deepseek-chat',
29
+ baseURL: 'https://api.deepseek.com/v1',
30
+ },
31
+ };
32
+
33
+ /**
34
+ * 获取环境变量配置
35
+ */
36
+ function getEnvConfig() {
37
+ const provider = process.env.FOLIKO_PROVIDER || DEFAULT_CONFIG.provider;
38
+ const providerDefaults = PROVIDER_DEFAULTS[provider] || PROVIDER_DEFAULTS.minimax;
39
+
40
+ // 支持多种 API key 环境变量名
41
+ let apiKey = process.env.FOLIKO_API_KEY || null;
42
+ if (!apiKey) {
43
+ // 根据 provider 查找对应的 API key
44
+ const upperProvider = provider.toUpperCase().replace(/-/g, '_');
45
+ apiKey = process.env[`${upperProvider}_API_KEY`] || null;
46
+ }
47
+
48
+ return {
49
+ model: process.env.FOLIKO_MODEL || providerDefaults.model,
50
+ provider: provider,
51
+ baseURL: process.env.FOLIKO_BASE_URL || providerDefaults.baseURL,
52
+ apiKey: apiKey,
53
+ };
54
+ }
55
+
56
+ /**
57
+ * 解析命令行参数(命令行参数优先于环境变量)
58
+ */
59
+ function parseArgs(args) {
60
+ // 先获取环境变量配置(作为默认值)
61
+ const envConfig = getEnvConfig();
62
+ const options = { ...envConfig };
63
+
64
+ for (let i = 0; i < args.length; i++) {
65
+ const arg = args[i];
66
+ if (arg === '--model' && args[i + 1]) {
67
+ options.model = args[++i];
68
+ } else if (arg === '--provider' && args[i + 1]) {
69
+ options.provider = args[++i];
70
+ } else if (arg === '--base-url' && args[i + 1]) {
71
+ options.baseURL = args[++i];
72
+ } else if (arg === '--api-key' && args[i + 1]) {
73
+ options.apiKey = args[++i];
74
+ }
75
+ }
76
+
77
+ return options;
78
+ }
79
+
80
+ /**
81
+ * Chat 命令入口
82
+ */
83
+ async function chatCommand(args) {
84
+ const options = parseArgs(args);
85
+ console.log('=== Foliko 持续对话 ===\n');
86
+ console.log('输入 exit 或 quit 退出\n');
87
+ // 初始化框架
88
+ const framework = new Framework({ debug: false });
89
+ await framework.bootstrap({
90
+ agentDir: process.cwd() + '/.agent',
91
+ aiConfig: {
92
+ provider: options.provider,
93
+ model: options.model,
94
+ baseURL: options.baseURL,
95
+ apiKey: options.apiKey,
96
+ providerOptions: {
97
+ maxOutputTokens: 8192,
98
+ },
99
+ },
100
+ });
101
+
102
+ // 创建 Agent
103
+ const agent = framework.createAgent({
104
+ name: 'FolikoAgent',
105
+ systemPrompt: `你是一个有帮助的助手,擅长回答问题和执行任务。
106
+ **重要:** 子Agent 匹配规则必须遵守:
107
+ - 根据【子 Agent 匹配表】,将任务委托给最匹配的子Agent处理
108
+ - 使用 subagent_call 工具并指定 agentName 来委托任务
109
+ - 只有当没有匹配的子Agent时,才直接调用工具
110
+ **命令执行规范:**
111
+ - 执行 npx skills add / npx skills remove / npx skills list 命令时,必须自动添加参数:-a openclaw -y
112
+ - 禁止添加 -g 参数
113
+ - 例如:npx skills add xxx 应该执行为 npx skills add xxx -a openclaw -y`,
114
+ sharedPrompt: `工作目录: {{WORK_DIR}}`,
115
+ metadata: {
116
+ WORK_DIR: process.cwd(), // 覆盖内置的 WORK_DIR
117
+ },
118
+ });
119
+
120
+ // 初始化 UI
121
+ const ui = new ChatUI(agent);
122
+
123
+ // 监听通知事件,在 CLI 中显示
124
+ // 显示当前会话的通知,以及没有指定 sessionId 的广播通知
125
+ framework.on('notification', (data) => {
126
+ const { title, message, source, timestamp, sessionId } = data;
127
+ // 如果通知指定了 sessionId 且不匹配 CLI 会话,跳过
128
+ if (sessionId && sessionId !== ui.sessionId) {
129
+ return;
130
+ }
131
+ const time = timestamp ? new Date(timestamp).toLocaleTimeString('zh-CN') : '';
132
+ console.log('\n' + '='.repeat(50));
133
+ console.log(`🔔 [${source}] ${title}`);
134
+ console.log(message);
135
+ if (time) console.log(`时间: ${time}`);
136
+ console.log('='.repeat(50) + '\n');
137
+ });
138
+
139
+ // 启动聊天
140
+ ui.start();
141
+ }
142
+
143
+ module.exports = { chatCommand };
@@ -1,93 +1,93 @@
1
- /**
2
- * List 命令实现
3
- * 列出所有子Agent配置
4
- */
5
-
6
- const fs = require('fs');
7
- const path = require('path');
8
-
9
- /**
10
- * 列出 .agent/agents 目录下的所有 Agent 配置
11
- */
12
- async function listCommand() {
13
- const agentsDir = path.resolve(process.cwd(), '.agent', 'agents');
14
-
15
- if (!fs.existsSync(agentsDir)) {
16
- console.log('No .agent/agents directory found.');
17
- console.log('Create agents by adding files to: .agent/agents/');
18
- return;
19
- }
20
-
21
- const entries = fs.readdirSync(agentsDir, { withFileTypes: true });
22
- const agents = [];
23
-
24
- for (const entry of entries) {
25
- if (
26
- entry.isFile() &&
27
- (entry.name.endsWith('.js') || entry.name.endsWith('.json') || entry.name.endsWith('.md'))
28
- ) {
29
- const baseName = entry.name.replace(/\.(js|json|md)$/, '');
30
- const filePath = path.join(agentsDir, entry.name);
31
-
32
- try {
33
- let config = { name: baseName };
34
-
35
- if (entry.name.endsWith('.json')) {
36
- const content = fs.readFileSync(filePath, 'utf-8');
37
- config = { ...config, ...JSON.parse(content) };
38
- } else if (entry.name.endsWith('.md')) {
39
- // 尝试从 markdown 中提取配置
40
- const content = fs.readFileSync(filePath, 'utf-8');
41
- const jsonMatch = content.match(/```json\n([\s\S]*?)\n```/);
42
- if (jsonMatch) {
43
- config = { ...config, ...JSON.parse(jsonMatch[1]) };
44
- }
45
- // 提取 name 行
46
- const nameMatch = content.match(/^name:\s*(.+)$/m);
47
- if (nameMatch) {
48
- config.name = nameMatch[1].trim();
49
- }
50
- } else {
51
- // .js 文件,尝试加载
52
- delete require.cache[require.resolve(filePath)];
53
- const mod = require(filePath);
54
- config = typeof mod === 'function' ? mod() : mod;
55
- }
56
-
57
- agents.push({
58
- name: config.name || baseName,
59
- role: config.role || '-',
60
- description: config.description || '-',
61
- file: entry.name,
62
- });
63
- } catch (err) {
64
- agents.push({
65
- name: baseName,
66
- role: '-',
67
- description: `Error loading: ${err.message}`,
68
- file: entry.name,
69
- });
70
- }
71
- }
72
- }
73
-
74
- if (agents.length === 0) {
75
- console.log('No agents found in .agent/agents/');
76
- return;
77
- }
78
-
79
- console.log(`\nFound ${agents.length} agent(s) in .agent/agents/:\n`);
80
- console.log('Name Role Description');
81
- console.log('---------------- ------------------ ------------------------------------------');
82
-
83
- for (const agent of agents) {
84
- const name = (agent.name || '').padEnd(16).slice(0, 16);
85
- const role = (agent.role || '-').padEnd(18).slice(0, 18);
86
- const desc = (agent.description || '-').slice(0, 40);
87
- console.log(`${name} ${role} ${desc}`);
88
- }
89
-
90
- console.log('');
91
- }
92
-
93
- module.exports = { listCommand };
1
+ /**
2
+ * List 命令实现
3
+ * 列出所有子Agent配置
4
+ */
5
+
6
+ const fs = require('fs');
7
+ const path = require('path');
8
+
9
+ /**
10
+ * 列出 .agent/agents 目录下的所有 Agent 配置
11
+ */
12
+ async function listCommand() {
13
+ const agentsDir = path.resolve(process.cwd(), '.agent', 'agents');
14
+
15
+ if (!fs.existsSync(agentsDir)) {
16
+ console.log('No .agent/agents directory found.');
17
+ console.log('Create agents by adding files to: .agent/agents/');
18
+ return;
19
+ }
20
+
21
+ const entries = fs.readdirSync(agentsDir, { withFileTypes: true });
22
+ const agents = [];
23
+
24
+ for (const entry of entries) {
25
+ if (
26
+ entry.isFile() &&
27
+ (entry.name.endsWith('.js') || entry.name.endsWith('.json') || entry.name.endsWith('.md'))
28
+ ) {
29
+ const baseName = entry.name.replace(/\.(js|json|md)$/, '');
30
+ const filePath = path.join(agentsDir, entry.name);
31
+
32
+ try {
33
+ let config = { name: baseName };
34
+
35
+ if (entry.name.endsWith('.json')) {
36
+ const content = fs.readFileSync(filePath, 'utf-8');
37
+ config = { ...config, ...JSON.parse(content) };
38
+ } else if (entry.name.endsWith('.md')) {
39
+ // 尝试从 markdown 中提取配置
40
+ const content = fs.readFileSync(filePath, 'utf-8');
41
+ const jsonMatch = content.match(/```json\n([\s\S]*?)\n```/);
42
+ if (jsonMatch) {
43
+ config = { ...config, ...JSON.parse(jsonMatch[1]) };
44
+ }
45
+ // 提取 name 行
46
+ const nameMatch = content.match(/^name:\s*(.+)$/m);
47
+ if (nameMatch) {
48
+ config.name = nameMatch[1].trim();
49
+ }
50
+ } else {
51
+ // .js 文件,尝试加载
52
+ delete require.cache[require.resolve(filePath)];
53
+ const mod = require(filePath);
54
+ config = typeof mod === 'function' ? mod() : mod;
55
+ }
56
+
57
+ agents.push({
58
+ name: config.name || baseName,
59
+ role: config.role || '-',
60
+ description: config.description || '-',
61
+ file: entry.name,
62
+ });
63
+ } catch (err) {
64
+ agents.push({
65
+ name: baseName,
66
+ role: '-',
67
+ description: `Error loading: ${err.message}`,
68
+ file: entry.name,
69
+ });
70
+ }
71
+ }
72
+ }
73
+
74
+ if (agents.length === 0) {
75
+ console.log('No agents found in .agent/agents/');
76
+ return;
77
+ }
78
+
79
+ console.log(`\nFound ${agents.length} agent(s) in .agent/agents/:\n`);
80
+ console.log('Name Role Description');
81
+ console.log('---------------- ------------------ ------------------------------------------');
82
+
83
+ for (const agent of agents) {
84
+ const name = (agent.name || '').padEnd(16).slice(0, 16);
85
+ const role = (agent.role || '-').padEnd(18).slice(0, 18);
86
+ const desc = (agent.description || '-').slice(0, 40);
87
+ console.log(`${name} ${role} ${desc}`);
88
+ }
89
+
90
+ console.log('');
91
+ }
92
+
93
+ module.exports = { listCommand };
package/cli/src/index.js CHANGED
@@ -1,75 +1,75 @@
1
- /**
2
- * Foliko CLI 主逻辑
3
- */
4
-
5
- const { chatCommand } = require('./commands/chat');
6
- const { listCommand } = require('./commands/list');
7
- const fs = require('fs');
8
- const path = require('path');
9
- /**
10
- * CLI 主入口
11
- */
12
- async function cli() {
13
- const args = process.argv.slice(2);
14
- const command = args[0] || 'chat';
15
- const packageJsonPath = path.join(__dirname, '../../package.json');
16
-
17
- switch (command) {
18
- case 'chat':
19
- await chatCommand(args.slice(1));
20
- break;
21
-
22
- case 'list':
23
- case 'ls':
24
- await listCommand();
25
- break;
26
-
27
- case 'help':
28
- case '--help':
29
- case '-h':
30
- printHelp();
31
- break;
32
-
33
- case 'version':
34
- case '--version':
35
- case '-v':
36
- const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
37
- console.log(`${packageJson.name} v${packageJson.version}`);
38
- break;
39
-
40
- default:
41
- console.error(`Unknown command: ${command}`);
42
- console.error('Run "foliko help" for usage information');
43
- process.exit(1);
44
- }
45
- }
46
-
47
- /**
48
- * 打印帮助信息
49
- */
50
- function printHelp() {
51
- console.log(`
52
- Foliko CLI - Agent 框架的命令行工具
53
-
54
- Usage: foliko <command> [options]
55
-
56
- Commands:
57
- chat 启动持续对话聊天
58
- list 列出所有子Agent配置
59
- help 显示帮助信息
60
- version 显示版本号
61
-
62
- Chat Options:
63
- --model <name> 指定 AI 模型
64
- --provider <name> 指定 AI 提供商
65
- --base-url <url> 指定 API 基础地址
66
-
67
- Examples:
68
- foliko chat
69
- foliko chat --model MiniMax-M2.7
70
- foliko chat --provider minimax --base-url https://api.minimaxi.com/v1
71
- foliko list
72
- `);
73
- }
74
-
75
- module.exports = { cli };
1
+ /**
2
+ * Foliko CLI 主逻辑
3
+ */
4
+
5
+ const { chatCommand } = require('./commands/chat');
6
+ const { listCommand } = require('./commands/list');
7
+ const fs = require('fs');
8
+ const path = require('path');
9
+ /**
10
+ * CLI 主入口
11
+ */
12
+ async function cli() {
13
+ const args = process.argv.slice(2);
14
+ const command = args[0] || 'chat';
15
+ const packageJsonPath = path.join(__dirname, '../../package.json');
16
+
17
+ switch (command) {
18
+ case 'chat':
19
+ await chatCommand(args.slice(1));
20
+ break;
21
+
22
+ case 'list':
23
+ case 'ls':
24
+ await listCommand();
25
+ break;
26
+
27
+ case 'help':
28
+ case '--help':
29
+ case '-h':
30
+ printHelp();
31
+ break;
32
+
33
+ case 'version':
34
+ case '--version':
35
+ case '-v':
36
+ const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
37
+ console.log(`${packageJson.name} v${packageJson.version}`);
38
+ break;
39
+
40
+ default:
41
+ console.error(`Unknown command: ${command}`);
42
+ console.error('Run "foliko help" for usage information');
43
+ process.exit(1);
44
+ }
45
+ }
46
+
47
+ /**
48
+ * 打印帮助信息
49
+ */
50
+ function printHelp() {
51
+ console.log(`
52
+ Foliko CLI - Agent 框架的命令行工具
53
+
54
+ Usage: foliko <command> [options]
55
+
56
+ Commands:
57
+ chat 启动持续对话聊天
58
+ list 列出所有子Agent配置
59
+ help 显示帮助信息
60
+ version 显示版本号
61
+
62
+ Chat Options:
63
+ --model <name> 指定 AI 模型
64
+ --provider <name> 指定 AI 提供商
65
+ --base-url <url> 指定 API 基础地址
66
+
67
+ Examples:
68
+ foliko chat
69
+ foliko chat --model MiniMax-M2.7
70
+ foliko chat --provider minimax --base-url https://api.minimaxi.com/v1
71
+ foliko list
72
+ `);
73
+ }
74
+
75
+ module.exports = { cli };