@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 +21 -0
- package/README.md +117 -0
- package/dist/cli.d.ts +8 -0
- package/dist/cli.js +339 -0
- package/dist/commands/index.d.ts +9 -0
- package/dist/commands/index.js +467 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +320 -0
- package/dist/postinstall.d.ts +10 -0
- package/dist/postinstall.js +58 -0
- package/dist/utils/markdown.d.ts +10 -0
- package/dist/utils/markdown.js +69 -0
- package/package.json +69 -0
- package/scripts/postinstall.js +61 -0
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>;
|