@oflow-ai/oflow-cli 0.1.0

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/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 oFlow Team
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,117 @@
1
+ # oFlow CLI
2
+
3
+ AI-powered CLI assistant for developers. Supports multiple AI providers including OpenAI and Chinese open-source models.
4
+
5
+ ## Installation
6
+
7
+ ### Mac/Linux/Ubuntu
8
+
9
+ ```bash
10
+ npm i -g @oflow-ai/oflow-cli
11
+ ```
12
+
13
+ ### Windows
14
+
15
+ 1. Restart your terminal (CMD or PowerShell)
16
+ 2. Run:
17
+ ```bash
18
+ npm install -g @oflow-ai/oflow-cli
19
+ ```
20
+
21
+ ## Quick Start
22
+
23
+ ```bash
24
+ # Start oFlow CLI
25
+ oflow
26
+
27
+ # Configure authentication
28
+ oflow auth
29
+
30
+ # List available models
31
+ oflow models
32
+ ```
33
+
34
+ ## Supported AI Providers
35
+
36
+ ### OpenAI
37
+ - gpt-4o
38
+ - gpt-4o-mini
39
+ - gpt-4-turbo
40
+ - o1-preview
41
+ - o1-mini
42
+
43
+ ### Chinese Open-Source Models
44
+
45
+ | Provider | Models | Features |
46
+ |----------|--------|----------|
47
+ | DeepSeek | deepseek-chat, deepseek-reasoner | Reasoning with thinking process |
48
+ | Qwen (通义千问) | qwen-turbo/plus/max/long | Chinese optimized |
49
+ | GLM (智谱) | glm-4/plus/flash/long | Free tier available |
50
+ | Baichuan (百川) | Baichuan4, Baichuan3-Turbo | Chinese optimized |
51
+ | Yi (零一万物) | yi-lightning/large/medium | Cost-effective |
52
+ | Moonshot (月之暗面) | moonshot-v1-8k/32k/128k | Long context |
53
+ | SiliconFlow | Multiple open-source models | One-stop API |
54
+
55
+ ## CLI Options
56
+
57
+ ```bash
58
+ oflow [options] [command] [prompt]
59
+
60
+ Options:
61
+ -V, --version output the version number
62
+ -m, --model <model> Specify the AI model to use
63
+ -d, --directory <dir> Working directory
64
+ --yolo Run in yolo mode (auto-approve all operations)
65
+ --thinking Show AI thinking/reasoning process
66
+ --no-thinking Hide AI thinking/reasoning process
67
+ -h, --help display help for command
68
+
69
+ Commands:
70
+ auth [options] Configure authentication
71
+ config [options] View or edit configuration
72
+ models List available AI models
73
+ ```
74
+
75
+ ## Slash Commands
76
+
77
+ Inside the interactive CLI:
78
+
79
+ | Command | Description |
80
+ |---------|-------------|
81
+ | `/help` | Show help message |
82
+ | `/exit` | Exit CLI |
83
+ | `/clear` | Clear conversation history |
84
+ | `/init` | Analyze current project |
85
+ | `/model [name]` | Show or set current model |
86
+ | `/config` | Show configuration |
87
+ | `/thinking` | Toggle thinking process display |
88
+ | `/tools` | List available tools |
89
+ | `/agents` | List available sub-agents |
90
+ | `/history` | Show conversation history |
91
+ | `/commit` | Generate commit message |
92
+ | `/review` | Review recent code changes |
93
+
94
+ ## Features
95
+
96
+ - **Multi-provider support**: OpenAI, DeepSeek, Qwen, GLM, and more
97
+ - **Tool calling**: File operations, shell commands, web search
98
+ - **Thinking process**: View AI reasoning for supported models (DeepSeek Reasoner)
99
+ - **Sub-agents**: Specialized agents for different tasks
100
+ - **MCP integration**: Extend with Model Context Protocol servers
101
+ - **Skill system**: Installable skill packages
102
+ - **Sandbox mode**: Isolated execution with Docker/Podman
103
+
104
+ ## Configuration
105
+
106
+ Configuration is stored in:
107
+ - macOS/Linux: `~/.config/oflow-nodejs/config.json`
108
+ - Windows: `%APPDATA%\oflow-nodejs\config.json`
109
+
110
+ ## Requirements
111
+
112
+ - Node.js 18.0.0 or higher
113
+ - npm 9.0.0 or higher
114
+
115
+ ## License
116
+
117
+ MIT
package/dist/cli.d.ts ADDED
@@ -0,0 +1,8 @@
1
+ export interface CLIOptions {
2
+ model?: string;
3
+ yolo?: boolean;
4
+ sandbox?: boolean;
5
+ showThinking?: boolean;
6
+ }
7
+ export declare function startInteractiveMode(options: CLIOptions): Promise<void>;
8
+ export declare function runSingleCommand(prompt: string, options: CLIOptions): Promise<void>;
package/dist/cli.js ADDED
@@ -0,0 +1,339 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.startInteractiveMode = startInteractiveMode;
40
+ exports.runSingleCommand = runSingleCommand;
41
+ const readline = __importStar(require("readline"));
42
+ const chalk_1 = __importDefault(require("chalk"));
43
+ const ora_1 = __importDefault(require("ora"));
44
+ const core_1 = require("@oflow-ai/core");
45
+ const commands_1 = require("./commands");
46
+ let sandboxExecutor = null;
47
+ let showThinking = false;
48
+ async function startInteractiveMode(options) {
49
+ const configManager = (0, core_1.getConfigManager)();
50
+ const auth = configManager.getAuth();
51
+ if (!auth) {
52
+ console.log(chalk_1.default.red('Not authenticated. Run `oflow auth` first.'));
53
+ return;
54
+ }
55
+ // 初始化沙箱执行器
56
+ if (options.sandbox || configManager.get('sandbox')) {
57
+ sandboxExecutor = await (0, core_1.createSandboxExecutor)();
58
+ console.log(chalk_1.default.gray(`Sandbox mode: ${sandboxExecutor ? 'enabled' : 'unavailable'}`));
59
+ }
60
+ // 获取思考过程显示设置
61
+ showThinking = options.showThinking ?? configManager.getShowThinking();
62
+ const model = options.model || configManager.getDefaultModel() || 'gpt-4o';
63
+ const provider = (0, core_1.createAIProvider)(auth, model);
64
+ const agentFactory = new core_1.AgentFactory(provider, process.cwd());
65
+ const conversation = new core_1.Conversation({
66
+ provider,
67
+ workingDirectory: process.cwd(),
68
+ maxTokens: configManager.get('maxTokens'),
69
+ temperature: configManager.get('temperature'),
70
+ showThinking: showThinking,
71
+ onContent: (content) => {
72
+ process.stdout.write(chalk_1.default.white(content));
73
+ },
74
+ onThinking: (thinking) => {
75
+ // 显示思考过程,使用不同的颜色
76
+ process.stdout.write(chalk_1.default.magenta(thinking));
77
+ },
78
+ onToolCall: (toolCall) => {
79
+ console.log(chalk_1.default.cyan(`\n🔧 Using tool: ${toolCall.function.name}`));
80
+ },
81
+ onToolResult: (result) => {
82
+ // Tool result is handled silently
83
+ }
84
+ });
85
+ // Set system prompt
86
+ conversation.setSystemPrompt(getSystemPrompt());
87
+ console.log(chalk_1.default.bold.blue('\n╔═══════════════════════════════════════╗'));
88
+ console.log(chalk_1.default.bold.blue('║ Welcome to oflow CLI ║'));
89
+ console.log(chalk_1.default.bold.blue('╚═══════════════════════════════════════╝'));
90
+ console.log('');
91
+ console.log(chalk_1.default.gray(`Model: ${model}`));
92
+ console.log(chalk_1.default.gray(`Working directory: ${process.cwd()}`));
93
+ if (sandboxExecutor) {
94
+ console.log(chalk_1.default.gray(`Sandbox: enabled`));
95
+ }
96
+ console.log(chalk_1.default.gray(`Thinking: ${showThinking ? 'ON' : 'OFF'} (use /thinking to toggle)`));
97
+ console.log('');
98
+ console.log(chalk_1.default.gray('Type your message and press Enter to chat.'));
99
+ console.log(chalk_1.default.gray('Type /help for available commands.'));
100
+ console.log(chalk_1.default.gray('Type $agent-name to call a sub-agent.'));
101
+ console.log(chalk_1.default.gray('Type /exit or press Ctrl+C to exit.'));
102
+ console.log('');
103
+ const rl = readline.createInterface({
104
+ input: process.stdin,
105
+ output: process.stdout,
106
+ prompt: chalk_1.default.green('> ')
107
+ });
108
+ rl.prompt();
109
+ rl.on('line', async (line) => {
110
+ const input = line.trim();
111
+ if (!input) {
112
+ rl.prompt();
113
+ return;
114
+ }
115
+ // Check for slash commands
116
+ if ((0, commands_1.isSlashCommand)(input)) {
117
+ const shouldContinue = await (0, commands_1.executeSlashCommand)(input, {
118
+ conversation,
119
+ rl,
120
+ configManager
121
+ });
122
+ if (!shouldContinue) {
123
+ rl.close();
124
+ return;
125
+ }
126
+ rl.prompt();
127
+ return;
128
+ }
129
+ // Check for agent calls (starting with $)
130
+ if (input.startsWith('$')) {
131
+ await handleAgentCall(input, agentFactory, options);
132
+ rl.prompt();
133
+ return;
134
+ }
135
+ // Check for shell commands (starting with !)
136
+ if (input.startsWith('!')) {
137
+ await executeShellCommand(input.slice(1), sandboxExecutor);
138
+ rl.prompt();
139
+ return;
140
+ }
141
+ // Check for file reference (starting with @)
142
+ const processedInput = await processFileReferences(input);
143
+ // Send to AI
144
+ const spinner = (0, ora_1.default)('Thinking...').start();
145
+ try {
146
+ const response = await conversation.sendMessage(processedInput);
147
+ spinner.stop();
148
+ console.log(''); // New line after response
149
+ rl.prompt();
150
+ }
151
+ catch (error) {
152
+ spinner.fail('Error');
153
+ console.log(chalk_1.default.red(error instanceof Error ? error.message : String(error)));
154
+ rl.prompt();
155
+ }
156
+ });
157
+ rl.on('close', () => {
158
+ console.log(chalk_1.default.bold('\nGoodbye! 👋\n'));
159
+ process.exit(0);
160
+ });
161
+ // Handle Ctrl+C
162
+ process.on('SIGINT', () => {
163
+ console.log(chalk_1.default.bold('\nGoodbye! 👋\n'));
164
+ process.exit(0);
165
+ });
166
+ }
167
+ async function handleAgentCall(input, agentFactory, options) {
168
+ // Parse agent call: $agent-type "task description"
169
+ const match = input.match(/^\$(\w+(?:-\w+)*)\s*(.*)$/);
170
+ if (!match) {
171
+ console.log(chalk_1.default.yellow('Invalid agent call format. Use: $agent-type "task"'));
172
+ console.log(chalk_1.default.gray('Available agents: general-purpose, plan-agent, explore-agent, code-reviewer, frontend-tester'));
173
+ return;
174
+ }
175
+ const agentType = match[1];
176
+ const task = match[2].replace(/^["']|["']$/g, '');
177
+ if (!task) {
178
+ console.log(chalk_1.default.yellow('Please provide a task for the agent.'));
179
+ return;
180
+ }
181
+ try {
182
+ const agent = agentFactory.createAgent(agentType);
183
+ console.log(chalk_1.default.cyan(`\n🤖 Starting ${agentType} agent...`));
184
+ console.log(chalk_1.default.gray(`Task: ${task}\n`));
185
+ const spinner = (0, ora_1.default)('Agent working...').start();
186
+ const result = await agent.execute(task, (update) => {
187
+ spinner.stop();
188
+ process.stdout.write(update);
189
+ spinner.start();
190
+ });
191
+ spinner.stop();
192
+ if (result.success) {
193
+ console.log(chalk_1.default.green('\n\n✓ Agent completed successfully'));
194
+ if (result.filesModified && result.filesModified.length > 0) {
195
+ console.log(chalk_1.default.gray('Files modified:'));
196
+ result.filesModified.forEach(f => console.log(chalk_1.default.gray(` - ${f}`)));
197
+ }
198
+ }
199
+ else {
200
+ console.log(chalk_1.default.red('\n\n✗ Agent encountered errors:'));
201
+ result.errors?.forEach(e => console.log(chalk_1.default.red(` - ${e}`)));
202
+ }
203
+ if (result.output) {
204
+ console.log(chalk_1.default.gray('\nOutput:'));
205
+ console.log(result.output);
206
+ }
207
+ }
208
+ catch (error) {
209
+ console.log(chalk_1.default.red(error instanceof Error ? error.message : String(error)));
210
+ }
211
+ }
212
+ async function runSingleCommand(prompt, options) {
213
+ const configManager = (0, core_1.getConfigManager)();
214
+ const auth = configManager.getAuth();
215
+ if (!auth) {
216
+ console.log(chalk_1.default.red('Not authenticated. Run `oflow auth` first.'));
217
+ return;
218
+ }
219
+ // 初始化沙箱执行器
220
+ if (options.sandbox || configManager.get('sandbox')) {
221
+ sandboxExecutor = await (0, core_1.createSandboxExecutor)();
222
+ }
223
+ const model = options.model || configManager.getDefaultModel() || 'gpt-4o';
224
+ const provider = (0, core_1.createAIProvider)(auth, model);
225
+ const conversation = new core_1.Conversation({
226
+ provider,
227
+ workingDirectory: process.cwd(),
228
+ maxTokens: configManager.get('maxTokens'),
229
+ temperature: configManager.get('temperature'),
230
+ onContent: (content) => {
231
+ process.stdout.write(chalk_1.default.white(content));
232
+ }
233
+ });
234
+ conversation.setSystemPrompt(getSystemPrompt());
235
+ const processedInput = await processFileReferences(prompt);
236
+ const spinner = (0, ora_1.default)('Processing...').start();
237
+ try {
238
+ await conversation.sendMessage(processedInput);
239
+ spinner.stop();
240
+ console.log('');
241
+ }
242
+ catch (error) {
243
+ spinner.fail('Error');
244
+ console.log(chalk_1.default.red(error instanceof Error ? error.message : String(error)));
245
+ }
246
+ }
247
+ async function executeShellCommand(command, sandbox) {
248
+ const { spawn } = require('child_modules');
249
+ if (sandbox && sandbox['config']?.backend !== 'none') {
250
+ // Use sandbox execution
251
+ const result = await sandbox.execute(command, process.cwd());
252
+ if (result.stdout)
253
+ console.log(result.stdout);
254
+ if (result.stderr)
255
+ console.error(result.stderr);
256
+ if (result.timedOut)
257
+ console.log(chalk_1.default.yellow('Command timed out'));
258
+ return;
259
+ }
260
+ return new Promise((resolve) => {
261
+ const isWindows = process.platform === 'win32';
262
+ const shell = isWindows ? 'powershell.exe' : '/bin/bash';
263
+ const args = isWindows ? ['-NoProfile', '-Command', command] : ['-c', command];
264
+ const proc = spawn(shell, args, {
265
+ stdio: 'inherit',
266
+ cwd: process.cwd()
267
+ });
268
+ proc.on('close', () => {
269
+ resolve();
270
+ });
271
+ });
272
+ }
273
+ async function processFileReferences(input) {
274
+ const fileRefRegex = /@([^\s]+)/g;
275
+ const matches = input.matchAll(fileRefRegex);
276
+ let result = input;
277
+ const fs = require('fs');
278
+ const path = require('path');
279
+ for (const match of matches) {
280
+ const filePath = match[1];
281
+ try {
282
+ const content = fs.readFileSync(filePath, 'utf-8');
283
+ result = result.replace(match[0], `\n[File: ${filePath}]\n${content}\n[End of file]\n`);
284
+ }
285
+ catch {
286
+ // File not found, keep original reference
287
+ }
288
+ }
289
+ return result;
290
+ }
291
+ function getSystemPrompt() {
292
+ return `You are oflow CLI, an interactive CLI agent specializing in software engineering tasks.
293
+
294
+ Your primary goal is to help users safely and efficiently complete programming tasks.
295
+
296
+ # Core Capabilities
297
+ - Read, write, and edit files
298
+ - Execute shell commands
299
+ - Search and analyze code
300
+ - Web search and fetch content
301
+ - Manage project structure
302
+ - Launch specialized sub-agents
303
+
304
+ # Tools Available
305
+ - read_file: Read file contents
306
+ - write_file: Write to files
307
+ - list_directory: List directory contents
308
+ - glob: Find files by pattern
309
+ - search_file_content: Search within files
310
+ - run_shell_command: Execute shell commands
311
+ - web_search: Search the web
312
+ - web_fetch: Fetch web content
313
+ - replace: Replace text in files
314
+ - image_read: Analyze images
315
+ - pdf_extract: Extract content from PDFs
316
+ - ask_user_question: Ask user for input
317
+ - task: Launch sub-agents
318
+ - save_memory: Save information for future sessions
319
+
320
+ # Sub-Agents (use $agent-name to call)
321
+ - general-purpose: Complex multi-step tasks
322
+ - plan-agent: Planning and analysis
323
+ - explore-agent: Codebase exploration
324
+ - code-reviewer: Code review
325
+ - frontend-tester: UI testing
326
+
327
+ # Guidelines
328
+ 1. Be concise and helpful
329
+ 2. Always explain what you're doing before using tools
330
+ 3. Ask for clarification when needed
331
+ 4. Prefer editing existing files over creating new ones
332
+ 5. Use appropriate tools for each task
333
+ 6. Verify changes after making them
334
+ 7. Provide actionable suggestions
335
+
336
+ Current working directory: ${process.cwd()}
337
+ `;
338
+ }
339
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["import * as readline from 'readline';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport {\n  getConfigManager,\n  createAIProvider,\n  Conversation,\n  Message,\n  AgentFactory,\n  Agent,\n  AgentType,\n  createSandboxExecutor,\n  SandboxExecutor\n} from '@oflow-ai/core';\nimport { isSlashCommand, executeSlashCommand } from './commands';\nimport { renderMarkdown } from './utils/markdown';\n\nexport interface CLIOptions {\n  model?: string;\n  yolo?: boolean;\n  sandbox?: boolean;\n  showThinking?: boolean;\n}\n\nlet sandboxExecutor: SandboxExecutor | null = null;\nlet showThinking = false;\n\nexport async function startInteractiveMode(options: CLIOptions): Promise<void> {\n  const configManager = getConfigManager();\n  const auth = configManager.getAuth();\n\n  if (!auth) {\n    console.log(chalk.red('Not authenticated. Run `oflow auth` first.'));\n    return;\n  }\n\n  // 初始化沙箱执行器\n  if (options.sandbox || configManager.get('sandbox')) {\n    sandboxExecutor = await createSandboxExecutor();\n    console.log(chalk.gray(`Sandbox mode: ${sandboxExecutor ? 'enabled' : 'unavailable'}`));\n  }\n\n  // 获取思考过程显示设置\n  showThinking = options.showThinking ?? configManager.getShowThinking();\n\n  const model = options.model || configManager.getDefaultModel() || 'gpt-4o';\n  const provider = createAIProvider(auth, model);\n  const agentFactory = new AgentFactory(provider, process.cwd());\n\n  const conversation = new Conversation({\n    provider,\n    workingDirectory: process.cwd(),\n    maxTokens: configManager.get('maxTokens'),\n    temperature: configManager.get('temperature'),\n    showThinking: showThinking,\n    onContent: (content) => {\n      process.stdout.write(chalk.white(content));\n    },\n    onThinking: (thinking) => {\n      // 显示思考过程，使用不同的颜色\n      process.stdout.write(chalk.magenta(thinking));\n    },\n    onToolCall: (toolCall) => {\n      console.log(chalk.cyan(`\\n🔧 Using tool: ${toolCall.function.name}`));\n    },\n    onToolResult: (result) => {\n      // Tool result is handled silently\n    }\n  });\n\n  // Set system prompt\n  conversation.setSystemPrompt(getSystemPrompt());\n\n  console.log(chalk.bold.blue('\\n╔═══════════════════════════════════════╗'));\n  console.log(chalk.bold.blue('║         Welcome to oflow CLI          ║'));\n  console.log(chalk.bold.blue('╚═══════════════════════════════════════╝'));\n  console.log('');\n  console.log(chalk.gray(`Model: ${model}`));\n  console.log(chalk.gray(`Working directory: ${process.cwd()}`));\n  if (sandboxExecutor) {\n    console.log(chalk.gray(`Sandbox: enabled`));\n  }\n  console.log(chalk.gray(`Thinking: ${showThinking ? 'ON' : 'OFF'} (use /thinking to toggle)`));\n  console.log('');\n  console.log(chalk.gray('Type your message and press Enter to chat.'));\n  console.log(chalk.gray('Type /help for available commands.'));\n  console.log(chalk.gray('Type $agent-name to call a sub-agent.'));\n  console.log(chalk.gray('Type /exit or press Ctrl+C to exit.'));\n  console.log('');\n\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n    prompt: chalk.green('> ')\n  });\n\n  rl.prompt();\n\n  rl.on('line', async (line) => {\n    const input = line.trim();\n    \n    if (!input) {\n      rl.prompt();\n      return;\n    }\n\n    // Check for slash commands\n    if (isSlashCommand(input)) {\n      const shouldContinue = await executeSlashCommand(input, {\n        conversation,\n        rl,\n        configManager\n      });\n      \n      if (!shouldContinue) {\n        rl.close();\n        return;\n      }\n      \n      rl.prompt();\n      return;\n    }\n\n    // Check for agent calls (starting with $)\n    if (input.startsWith('$')) {\n      await handleAgentCall(input, agentFactory, options);\n      rl.prompt();\n      return;\n    }\n\n    // Check for shell commands (starting with !)\n    if (input.startsWith('!')) {\n      await executeShellCommand(input.slice(1), sandboxExecutor);\n      rl.prompt();\n      return;\n    }\n\n    // Check for file reference (starting with @)\n    const processedInput = await processFileReferences(input);\n\n    // Send to AI\n    const spinner = ora('Thinking...').start();\n    \n    try {\n      const response = await conversation.sendMessage(processedInput);\n      spinner.stop();\n      \n      console.log(''); // New line after response\n      rl.prompt();\n    } catch (error) {\n      spinner.fail('Error');\n      console.log(chalk.red(error instanceof Error ? error.message : String(error)));\n      rl.prompt();\n    }\n  });\n\n  rl.on('close', () => {\n    console.log(chalk.bold('\\nGoodbye! 👋\\n'));\n    process.exit(0);\n  });\n\n  // Handle Ctrl+C\n  process.on('SIGINT', () => {\n    console.log(chalk.bold('\\nGoodbye! 👋\\n'));\n    process.exit(0);\n  });\n}\n\nasync function handleAgentCall(\n  input: string, \n  agentFactory: AgentFactory,\n  options: CLIOptions\n): Promise<void> {\n  // Parse agent call: $agent-type \"task description\"\n  const match = input.match(/^\\$(\\w+(?:-\\w+)*)\\s*(.*)$/);\n  \n  if (!match) {\n    console.log(chalk.yellow('Invalid agent call format. Use: $agent-type \"task\"'));\n    console.log(chalk.gray('Available agents: general-purpose, plan-agent, explore-agent, code-reviewer, frontend-tester'));\n    return;\n  }\n\n  const agentType = match[1] as AgentType;\n  const task = match[2].replace(/^[\"']|[\"']$/g, '');\n\n  if (!task) {\n    console.log(chalk.yellow('Please provide a task for the agent.'));\n    return;\n  }\n\n  try {\n    const agent = agentFactory.createAgent(agentType);\n    console.log(chalk.cyan(`\\n🤖 Starting ${agentType} agent...`));\n    console.log(chalk.gray(`Task: ${task}\\n`));\n\n    const spinner = ora('Agent working...').start();\n    \n    const result = await agent.execute(task, (update) => {\n      spinner.stop();\n      process.stdout.write(update);\n      spinner.start();\n    });\n\n    spinner.stop();\n    \n    if (result.success) {\n      console.log(chalk.green('\\n\\n✓ Agent completed successfully'));\n      if (result.filesModified && result.filesModified.length > 0) {\n        console.log(chalk.gray('Files modified:'));\n        result.filesModified.forEach(f => console.log(chalk.gray(`  - ${f}`)));\n      }\n    } else {\n      console.log(chalk.red('\\n\\n✗ Agent encountered errors:'));\n      result.errors?.forEach(e => console.log(chalk.red(`  - ${e}`)));\n    }\n\n    if (result.output) {\n      console.log(chalk.gray('\\nOutput:'));\n      console.log(result.output);\n    }\n\n  } catch (error) {\n    console.log(chalk.red(error instanceof Error ? error.message : String(error)));\n  }\n}\n\nexport async function runSingleCommand(prompt: string, options: CLIOptions): Promise<void> {\n  const configManager = getConfigManager();\n  const auth = configManager.getAuth();\n\n  if (!auth) {\n    console.log(chalk.red('Not authenticated. Run `oflow auth` first.'));\n    return;\n  }\n\n  // 初始化沙箱执行器\n  if (options.sandbox || configManager.get('sandbox')) {\n    sandboxExecutor = await createSandboxExecutor();\n  }\n\n  const model = options.model || configManager.getDefaultModel() || 'gpt-4o';\n  const provider = createAIProvider(auth, model);\n  const conversation = new Conversation({\n    provider,\n    workingDirectory: process.cwd(),\n    maxTokens: configManager.get('maxTokens'),\n    temperature: configManager.get('temperature'),\n    onContent: (content) => {\n      process.stdout.write(chalk.white(content));\n    }\n  });\n\n  conversation.setSystemPrompt(getSystemPrompt());\n\n  const processedInput = await processFileReferences(prompt);\n  const spinner = ora('Processing...').start();\n\n  try {\n    await conversation.sendMessage(processedInput);\n    spinner.stop();\n    console.log('');\n  } catch (error) {\n    spinner.fail('Error');\n    console.log(chalk.red(error instanceof Error ? error.message : String(error)));\n  }\n}\n\nasync function executeShellCommand(command: string, sandbox?: SandboxExecutor | null): Promise<void> {\n  const { spawn } = require('child_modules');\n  \n  if (sandbox && sandbox['config']?.backend !== 'none') {\n    // Use sandbox execution\n    const result = await sandbox.execute(command, process.cwd());\n    if (result.stdout) console.log(result.stdout);\n    if (result.stderr) console.error(result.stderr);\n    if (result.timedOut) console.log(chalk.yellow('Command timed out'));\n    return;\n  }\n  \n  return new Promise((resolve) => {\n    const isWindows = process.platform === 'win32';\n    const shell = isWindows ? 'powershell.exe' : '/bin/bash';\n    const args = isWindows ? ['-NoProfile', '-Command', command] : ['-c', command];\n\n    const proc = spawn(shell, args, {\n      stdio: 'inherit',\n      cwd: process.cwd()\n    });\n\n    proc.on('close', () => {\n      resolve();\n    });\n  });\n}\n\nasync function processFileReferences(input: string): Promise<string> {\n  const fileRefRegex = /@([^\\s]+)/g;\n  const matches = input.matchAll(fileRefRegex);\n  \n  let result = input;\n  const fs = require('fs');\n  const path = require('path');\n\n  for (const match of matches) {\n    const filePath = match[1];\n    try {\n      const content = fs.readFileSync(filePath, 'utf-8');\n      result = result.replace(match[0], `\\n[File: ${filePath}]\\n${content}\\n[End of file]\\n`);\n    } catch {\n      // File not found, keep original reference\n    }\n  }\n\n  return result;\n}\n\nfunction getSystemPrompt(): string {\n  return `You are oflow CLI, an interactive CLI agent specializing in software engineering tasks.\n\nYour primary goal is to help users safely and efficiently complete programming tasks.\n\n# Core Capabilities\n- Read, write, and edit files\n- Execute shell commands\n- Search and analyze code\n- Web search and fetch content\n- Manage project structure\n- Launch specialized sub-agents\n\n# Tools Available\n- read_file: Read file contents\n- write_file: Write to files\n- list_directory: List directory contents\n- glob: Find files by pattern\n- search_file_content: Search within files\n- run_shell_command: Execute shell commands\n- web_search: Search the web\n- web_fetch: Fetch web content\n- replace: Replace text in files\n- image_read: Analyze images\n- pdf_extract: Extract content from PDFs\n- ask_user_question: Ask user for input\n- task: Launch sub-agents\n- save_memory: Save information for future sessions\n\n# Sub-Agents (use $agent-name to call)\n- general-purpose: Complex multi-step tasks\n- plan-agent: Planning and analysis\n- explore-agent: Codebase exploration\n- code-reviewer: Code review\n- frontend-tester: UI testing\n\n# Guidelines\n1. Be concise and helpful\n2. Always explain what you're doing before using tools\n3. Ask for clarification when needed\n4. Prefer editing existing files over creating new ones\n5. Use appropriate tools for each task\n6. Verify changes after making them\n7. Provide actionable suggestions\n\nCurrent working directory: ${process.cwd()}\n`;\n}"]}
@@ -0,0 +1,9 @@
1
+ import { Conversation, ConfigManager } from '@oflow-ai/core';
2
+ import * as readline from 'readline';
3
+ export interface CommandContext {
4
+ conversation: Conversation;
5
+ rl: readline.Interface;
6
+ configManager: ConfigManager;
7
+ }
8
+ export declare function isSlashCommand(input: string): boolean;
9
+ export declare function executeSlashCommand(input: string, context: CommandContext): Promise<boolean>;