@serjm/deepseek-code 0.3.1
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/CONTRIBUTING.md +73 -0
- package/README.md +194 -0
- package/README.ru.md +194 -0
- package/dist/api/index.d.ts +77 -0
- package/dist/api/index.d.ts.map +1 -0
- package/dist/api/index.js +263 -0
- package/dist/api/index.js.map +1 -0
- package/dist/cli/headless.d.ts +22 -0
- package/dist/cli/headless.d.ts.map +1 -0
- package/dist/cli/headless.js +122 -0
- package/dist/cli/headless.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +90 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/interactive.d.ts +18 -0
- package/dist/cli/interactive.d.ts.map +1 -0
- package/dist/cli/interactive.js +75 -0
- package/dist/cli/interactive.js.map +1 -0
- package/dist/commands/index.d.ts +30 -0
- package/dist/commands/index.d.ts.map +1 -0
- package/dist/commands/index.js +964 -0
- package/dist/commands/index.js.map +1 -0
- package/dist/config/defaults.d.ts +37 -0
- package/dist/config/defaults.d.ts.map +1 -0
- package/dist/config/defaults.js +39 -0
- package/dist/config/defaults.js.map +1 -0
- package/dist/config/loader.d.ts +4 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +76 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/core/agent-loop.d.ts +111 -0
- package/dist/core/agent-loop.d.ts.map +1 -0
- package/dist/core/agent-loop.js +485 -0
- package/dist/core/agent-loop.js.map +1 -0
- package/dist/core/checkpoint.d.ts +10 -0
- package/dist/core/checkpoint.d.ts.map +1 -0
- package/dist/core/checkpoint.js +83 -0
- package/dist/core/checkpoint.js.map +1 -0
- package/dist/core/extensions.d.ts +55 -0
- package/dist/core/extensions.d.ts.map +1 -0
- package/dist/core/extensions.js +113 -0
- package/dist/core/extensions.js.map +1 -0
- package/dist/core/git.d.ts +68 -0
- package/dist/core/git.d.ts.map +1 -0
- package/dist/core/git.js +148 -0
- package/dist/core/git.js.map +1 -0
- package/dist/core/hooks.d.ts +37 -0
- package/dist/core/hooks.d.ts.map +1 -0
- package/dist/core/hooks.js +77 -0
- package/dist/core/hooks.js.map +1 -0
- package/dist/core/i18n.d.ts +90 -0
- package/dist/core/i18n.d.ts.map +1 -0
- package/dist/core/i18n.js +253 -0
- package/dist/core/i18n.js.map +1 -0
- package/dist/core/lsp.d.ts +74 -0
- package/dist/core/lsp.d.ts.map +1 -0
- package/dist/core/lsp.js +239 -0
- package/dist/core/lsp.js.map +1 -0
- package/dist/core/mcp.d.ts +49 -0
- package/dist/core/mcp.d.ts.map +1 -0
- package/dist/core/mcp.js +195 -0
- package/dist/core/mcp.js.map +1 -0
- package/dist/core/memory.d.ts +38 -0
- package/dist/core/memory.d.ts.map +1 -0
- package/dist/core/memory.js +231 -0
- package/dist/core/memory.js.map +1 -0
- package/dist/core/metrics.d.ts +36 -0
- package/dist/core/metrics.d.ts.map +1 -0
- package/dist/core/metrics.js +111 -0
- package/dist/core/metrics.js.map +1 -0
- package/dist/core/review.d.ts +27 -0
- package/dist/core/review.d.ts.map +1 -0
- package/dist/core/review.js +201 -0
- package/dist/core/review.js.map +1 -0
- package/dist/core/sandbox.d.ts +52 -0
- package/dist/core/sandbox.d.ts.map +1 -0
- package/dist/core/sandbox.js +140 -0
- package/dist/core/sandbox.js.map +1 -0
- package/dist/core/scheduler.d.ts +56 -0
- package/dist/core/scheduler.d.ts.map +1 -0
- package/dist/core/scheduler.js +167 -0
- package/dist/core/scheduler.js.map +1 -0
- package/dist/core/session.d.ts +49 -0
- package/dist/core/session.d.ts.map +1 -0
- package/dist/core/session.js +127 -0
- package/dist/core/session.js.map +1 -0
- package/dist/core/skills.d.ts +36 -0
- package/dist/core/skills.d.ts.map +1 -0
- package/dist/core/skills.js +90 -0
- package/dist/core/skills.js.map +1 -0
- package/dist/core/subagent.d.ts +45 -0
- package/dist/core/subagent.d.ts.map +1 -0
- package/dist/core/subagent.js +130 -0
- package/dist/core/subagent.js.map +1 -0
- package/dist/core/themes.d.ts +35 -0
- package/dist/core/themes.d.ts.map +1 -0
- package/dist/core/themes.js +188 -0
- package/dist/core/themes.js.map +1 -0
- package/dist/tools/bash.d.ts +3 -0
- package/dist/tools/bash.d.ts.map +1 -0
- package/dist/tools/bash.js +92 -0
- package/dist/tools/bash.js.map +1 -0
- package/dist/tools/chrome-manager.d.ts +35 -0
- package/dist/tools/chrome-manager.d.ts.map +1 -0
- package/dist/tools/chrome-manager.js +163 -0
- package/dist/tools/chrome-manager.js.map +1 -0
- package/dist/tools/chrome.d.ts +78 -0
- package/dist/tools/chrome.d.ts.map +1 -0
- package/dist/tools/chrome.js +1058 -0
- package/dist/tools/chrome.js.map +1 -0
- package/dist/tools/edit.d.ts +3 -0
- package/dist/tools/edit.d.ts.map +1 -0
- package/dist/tools/edit.js +81 -0
- package/dist/tools/edit.js.map +1 -0
- package/dist/tools/glob.d.ts +3 -0
- package/dist/tools/glob.d.ts.map +1 -0
- package/dist/tools/glob.js +41 -0
- package/dist/tools/glob.js.map +1 -0
- package/dist/tools/grep.d.ts +3 -0
- package/dist/tools/grep.d.ts.map +1 -0
- package/dist/tools/grep.js +74 -0
- package/dist/tools/grep.js.map +1 -0
- package/dist/tools/path-safety.d.ts +3 -0
- package/dist/tools/path-safety.d.ts.map +1 -0
- package/dist/tools/path-safety.js +19 -0
- package/dist/tools/path-safety.js.map +1 -0
- package/dist/tools/read.d.ts +3 -0
- package/dist/tools/read.d.ts.map +1 -0
- package/dist/tools/read.js +58 -0
- package/dist/tools/read.js.map +1 -0
- package/dist/tools/registry.d.ts +4 -0
- package/dist/tools/registry.d.ts.map +1 -0
- package/dist/tools/registry.js +43 -0
- package/dist/tools/registry.js.map +1 -0
- package/dist/tools/types.d.ts +47 -0
- package/dist/tools/types.d.ts.map +1 -0
- package/dist/tools/types.js +90 -0
- package/dist/tools/types.js.map +1 -0
- package/dist/tools/write.d.ts +3 -0
- package/dist/tools/write.d.ts.map +1 -0
- package/dist/tools/write.js +51 -0
- package/dist/tools/write.js.map +1 -0
- package/dist/ui/activity-cards.d.ts +50 -0
- package/dist/ui/activity-cards.d.ts.map +1 -0
- package/dist/ui/activity-cards.js +185 -0
- package/dist/ui/activity-cards.js.map +1 -0
- package/dist/ui/app.d.ts +9 -0
- package/dist/ui/app.d.ts.map +1 -0
- package/dist/ui/app.js +852 -0
- package/dist/ui/app.js.map +1 -0
- package/dist/ui/chat-view.d.ts +10 -0
- package/dist/ui/chat-view.d.ts.map +1 -0
- package/dist/ui/chat-view.js +94 -0
- package/dist/ui/chat-view.js.map +1 -0
- package/dist/ui/error-boundary.d.ts +13 -0
- package/dist/ui/error-boundary.d.ts.map +1 -0
- package/dist/ui/error-boundary.js +16 -0
- package/dist/ui/error-boundary.js.map +1 -0
- package/dist/ui/fade-in.d.ts +8 -0
- package/dist/ui/fade-in.d.ts.map +1 -0
- package/dist/ui/fade-in.js +14 -0
- package/dist/ui/fade-in.js.map +1 -0
- package/dist/ui/input-bar.d.ts +16 -0
- package/dist/ui/input-bar.d.ts.map +1 -0
- package/dist/ui/input-bar.js +269 -0
- package/dist/ui/input-bar.js.map +1 -0
- package/dist/ui/markdown-view.d.ts +9 -0
- package/dist/ui/markdown-view.d.ts.map +1 -0
- package/dist/ui/markdown-view.js +240 -0
- package/dist/ui/markdown-view.js.map +1 -0
- package/dist/ui/matrix-rain.d.ts +2 -0
- package/dist/ui/matrix-rain.d.ts.map +1 -0
- package/dist/ui/matrix-rain.js +134 -0
- package/dist/ui/matrix-rain.js.map +1 -0
- package/dist/ui/reasoning-view.d.ts +12 -0
- package/dist/ui/reasoning-view.d.ts.map +1 -0
- package/dist/ui/reasoning-view.js +34 -0
- package/dist/ui/reasoning-view.js.map +1 -0
- package/dist/ui/results-panel.d.ts +11 -0
- package/dist/ui/results-panel.d.ts.map +1 -0
- package/dist/ui/results-panel.js +17 -0
- package/dist/ui/results-panel.js.map +1 -0
- package/dist/ui/setup-wizard.d.ts +30 -0
- package/dist/ui/setup-wizard.d.ts.map +1 -0
- package/dist/ui/setup-wizard.js +166 -0
- package/dist/ui/setup-wizard.js.map +1 -0
- package/dist/ui/status-bar.d.ts +14 -0
- package/dist/ui/status-bar.d.ts.map +1 -0
- package/dist/ui/status-bar.js +63 -0
- package/dist/ui/status-bar.js.map +1 -0
- package/dist/ui/tool-activity-card.d.ts +9 -0
- package/dist/ui/tool-activity-card.d.ts.map +1 -0
- package/dist/ui/tool-activity-card.js +172 -0
- package/dist/ui/tool-activity-card.js.map +1 -0
- package/dist/ui/tool-call-view.d.ts +9 -0
- package/dist/ui/tool-call-view.d.ts.map +1 -0
- package/dist/ui/tool-call-view.js +149 -0
- package/dist/ui/tool-call-view.js.map +1 -0
- package/dist/utils/clipboard.d.ts +6 -0
- package/dist/utils/clipboard.d.ts.map +1 -0
- package/dist/utils/clipboard.js +56 -0
- package/dist/utils/clipboard.js.map +1 -0
- package/dist/utils/ignore.d.ts +6 -0
- package/dist/utils/ignore.d.ts.map +1 -0
- package/dist/utils/ignore.js +40 -0
- package/dist/utils/ignore.js.map +1 -0
- package/dist/utils/logger.d.ts +4 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +13 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/string-width.d.ts +6 -0
- package/dist/utils/string-width.d.ts.map +1 -0
- package/dist/utils/string-width.js +37 -0
- package/dist/utils/string-width.js.map +1 -0
- package/package.json +68 -0
|
@@ -0,0 +1,263 @@
|
|
|
1
|
+
import OpenAI from 'openai';
|
|
2
|
+
// ─── Retry helpers ───────────────────────────────────────────────────────────
|
|
3
|
+
const MAX_RETRY_ATTEMPTS = 3;
|
|
4
|
+
const STREAM_CHUNK_TIMEOUT_MS = 60_000;
|
|
5
|
+
function sleep(ms) {
|
|
6
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
7
|
+
}
|
|
8
|
+
function isRetryable(err) {
|
|
9
|
+
const msg = String(err?.message ?? err);
|
|
10
|
+
return /429|5[0-9]{2}|ECONNRESET|ETIMEDOUT|ENOTFOUND|ECONNREFUSED/.test(msg);
|
|
11
|
+
}
|
|
12
|
+
async function withRetry(fn) {
|
|
13
|
+
let lastErr;
|
|
14
|
+
for (let attempt = 1; attempt <= MAX_RETRY_ATTEMPTS; attempt++) {
|
|
15
|
+
try {
|
|
16
|
+
return await fn();
|
|
17
|
+
}
|
|
18
|
+
catch (err) {
|
|
19
|
+
lastErr = err;
|
|
20
|
+
if (attempt === MAX_RETRY_ATTEMPTS || !isRetryable(err))
|
|
21
|
+
throw err;
|
|
22
|
+
await sleep(Math.min(1000 * 2 ** (attempt - 1), 30_000));
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
throw lastErr;
|
|
26
|
+
}
|
|
27
|
+
export class DeepSeekAPI {
|
|
28
|
+
client;
|
|
29
|
+
model;
|
|
30
|
+
systemPrompt;
|
|
31
|
+
constructor(config) {
|
|
32
|
+
this.client = new OpenAI({
|
|
33
|
+
apiKey: config.apiKey ?? process.env.DEEPSEEK_API_KEY ?? '',
|
|
34
|
+
baseURL: config.baseUrl,
|
|
35
|
+
});
|
|
36
|
+
this.model = config.model;
|
|
37
|
+
this.systemPrompt = config.systemPrompt ?? '';
|
|
38
|
+
}
|
|
39
|
+
async *streamChat(messages, tools) {
|
|
40
|
+
// Only prepend system prompt if messages don't already contain one
|
|
41
|
+
const hasSystem = messages.some(m => m.role === 'system');
|
|
42
|
+
const fullMessages = hasSystem
|
|
43
|
+
? buildMessages(messages)
|
|
44
|
+
: [
|
|
45
|
+
{ role: 'system', content: this.systemPrompt },
|
|
46
|
+
...buildMessages(messages),
|
|
47
|
+
];
|
|
48
|
+
const timeoutController = new AbortController();
|
|
49
|
+
let chunkTimer;
|
|
50
|
+
const resetChunkTimer = () => {
|
|
51
|
+
clearTimeout(chunkTimer);
|
|
52
|
+
chunkTimer = setTimeout(() => {
|
|
53
|
+
timeoutController.abort(new Error(`Stream timeout: no data received for ${STREAM_CHUNK_TIMEOUT_MS / 1000}s`));
|
|
54
|
+
}, STREAM_CHUNK_TIMEOUT_MS);
|
|
55
|
+
};
|
|
56
|
+
const stream = await withRetry(() => this.client.chat.completions.create({
|
|
57
|
+
model: this.model,
|
|
58
|
+
messages: fullMessages,
|
|
59
|
+
stream: true,
|
|
60
|
+
stream_options: { include_usage: true },
|
|
61
|
+
...(tools && tools.length > 0 ? { tools, tool_choice: 'auto' } : {}),
|
|
62
|
+
}, { signal: timeoutController.signal }));
|
|
63
|
+
let currentToolCallId = '';
|
|
64
|
+
let currentToolName = '';
|
|
65
|
+
let currentToolArgs = '';
|
|
66
|
+
resetChunkTimer();
|
|
67
|
+
try {
|
|
68
|
+
for await (const chunk of stream) {
|
|
69
|
+
resetChunkTimer();
|
|
70
|
+
// Usage chunk (last chunk with stream_options.include_usage)
|
|
71
|
+
if (chunk.usage) {
|
|
72
|
+
yield {
|
|
73
|
+
type: 'usage',
|
|
74
|
+
content: '',
|
|
75
|
+
usage: {
|
|
76
|
+
input: chunk.usage.prompt_tokens ?? 0,
|
|
77
|
+
output: chunk.usage.completion_tokens ?? 0,
|
|
78
|
+
total: chunk.usage.total_tokens ?? 0,
|
|
79
|
+
},
|
|
80
|
+
};
|
|
81
|
+
continue;
|
|
82
|
+
}
|
|
83
|
+
const delta = chunk.choices[0]?.delta;
|
|
84
|
+
if (!delta)
|
|
85
|
+
continue;
|
|
86
|
+
// Reasoning content (DeepSeek-specific, from deepseek-reasoner model)
|
|
87
|
+
const reasoningContent = delta.reasoning_content;
|
|
88
|
+
if (reasoningContent) {
|
|
89
|
+
yield {
|
|
90
|
+
type: 'reasoning',
|
|
91
|
+
content: reasoningContent,
|
|
92
|
+
};
|
|
93
|
+
continue;
|
|
94
|
+
}
|
|
95
|
+
// Tool call delta
|
|
96
|
+
if (delta.tool_calls) {
|
|
97
|
+
for (const tc of delta.tool_calls) {
|
|
98
|
+
if (tc.id) {
|
|
99
|
+
// Flush previous tool call if any
|
|
100
|
+
if (currentToolCallId && currentToolName) {
|
|
101
|
+
yield {
|
|
102
|
+
type: 'tool_use',
|
|
103
|
+
content: '',
|
|
104
|
+
toolName: currentToolName,
|
|
105
|
+
toolInput: safeParseJSON(currentToolArgs),
|
|
106
|
+
toolCallId: currentToolCallId,
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
currentToolCallId = tc.id;
|
|
110
|
+
currentToolName = tc.function?.name ?? '';
|
|
111
|
+
currentToolArgs = tc.function?.arguments ?? '';
|
|
112
|
+
}
|
|
113
|
+
else if (tc.function?.arguments) {
|
|
114
|
+
currentToolArgs += tc.function.arguments;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
continue;
|
|
118
|
+
}
|
|
119
|
+
// Flush pending tool call before text
|
|
120
|
+
if (currentToolCallId && currentToolName) {
|
|
121
|
+
yield {
|
|
122
|
+
type: 'tool_use',
|
|
123
|
+
content: '',
|
|
124
|
+
toolName: currentToolName,
|
|
125
|
+
toolInput: safeParseJSON(currentToolArgs),
|
|
126
|
+
toolCallId: currentToolCallId,
|
|
127
|
+
};
|
|
128
|
+
currentToolCallId = '';
|
|
129
|
+
currentToolName = '';
|
|
130
|
+
currentToolArgs = '';
|
|
131
|
+
}
|
|
132
|
+
if (delta?.content) {
|
|
133
|
+
yield {
|
|
134
|
+
type: 'text',
|
|
135
|
+
content: delta.content,
|
|
136
|
+
};
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
// Flush remaining tool call at end of stream
|
|
140
|
+
if (currentToolCallId && currentToolName) {
|
|
141
|
+
yield {
|
|
142
|
+
type: 'tool_use',
|
|
143
|
+
content: '',
|
|
144
|
+
toolName: currentToolName,
|
|
145
|
+
toolInput: safeParseJSON(currentToolArgs),
|
|
146
|
+
toolCallId: currentToolCallId,
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
finally {
|
|
151
|
+
clearTimeout(chunkTimer);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Non-streaming chat with optional tools support.
|
|
156
|
+
* Returns either text content or tool_calls.
|
|
157
|
+
*/
|
|
158
|
+
async chat(messages, tools) {
|
|
159
|
+
const fullMessages = [
|
|
160
|
+
{ role: 'system', content: this.systemPrompt },
|
|
161
|
+
...buildMessages(messages),
|
|
162
|
+
];
|
|
163
|
+
const response = await withRetry(() => this.client.chat.completions.create({
|
|
164
|
+
model: this.model,
|
|
165
|
+
messages: fullMessages,
|
|
166
|
+
...(tools && tools.length > 0 ? { tools, tool_choice: 'auto' } : {}),
|
|
167
|
+
}));
|
|
168
|
+
const message = response.choices[0]?.message;
|
|
169
|
+
if (message?.tool_calls && message.tool_calls.length > 0) {
|
|
170
|
+
return {
|
|
171
|
+
content: message.content ?? '',
|
|
172
|
+
toolCalls: message.tool_calls.map(tc => ({
|
|
173
|
+
id: tc.id,
|
|
174
|
+
type: 'function',
|
|
175
|
+
function: {
|
|
176
|
+
name: tc.function.name,
|
|
177
|
+
arguments: tc.function.arguments,
|
|
178
|
+
},
|
|
179
|
+
})),
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
return {
|
|
183
|
+
content: message?.content ?? '',
|
|
184
|
+
usage: response.usage
|
|
185
|
+
? { input: response.usage.prompt_tokens ?? 0, output: response.usage.completion_tokens ?? 0 }
|
|
186
|
+
: undefined,
|
|
187
|
+
};
|
|
188
|
+
}
|
|
189
|
+
/** Validate API key by making a minimal models list request */
|
|
190
|
+
async validateKey() {
|
|
191
|
+
try {
|
|
192
|
+
await this.client.models.list();
|
|
193
|
+
return { valid: true };
|
|
194
|
+
}
|
|
195
|
+
catch (err) {
|
|
196
|
+
const msg = err?.message ?? String(err);
|
|
197
|
+
// DeepSeek returns 401 or 403 for invalid keys
|
|
198
|
+
if (typeof msg === 'string' && (msg.includes('401') || msg.includes('403') || msg.includes('Incorrect API key') || msg.includes('invalid') || msg.includes('Unauthorized'))) {
|
|
199
|
+
return { valid: false, error: 'Invalid API key. Please check your key at https://platform.deepseek.com/api_keys' };
|
|
200
|
+
}
|
|
201
|
+
if (typeof msg === 'string' && msg.includes('429')) {
|
|
202
|
+
return { valid: false, error: 'Rate limited. Please wait and try again.' };
|
|
203
|
+
}
|
|
204
|
+
return { valid: false, error: `API error: ${msg}` };
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Get embedding vector for a text string using DeepSeek's embeddings API.
|
|
209
|
+
*/
|
|
210
|
+
async getEmbedding(text) {
|
|
211
|
+
const response = await this.client.embeddings.create({
|
|
212
|
+
model: 'deepseek-embedding',
|
|
213
|
+
input: text,
|
|
214
|
+
});
|
|
215
|
+
return response.data[0].embedding;
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
// ─── Helper functions ────────────────────────────────────────────────────────
|
|
219
|
+
/**
|
|
220
|
+
* Build OpenAI-compatible message array from our internal ChatMessage format.
|
|
221
|
+
*/
|
|
222
|
+
function buildMessages(messages) {
|
|
223
|
+
return messages.map(m => {
|
|
224
|
+
if (m.role === 'tool') {
|
|
225
|
+
return {
|
|
226
|
+
role: 'tool',
|
|
227
|
+
content: typeof m.content === 'string' ? m.content : '',
|
|
228
|
+
tool_call_id: m.tool_call_id ?? '',
|
|
229
|
+
};
|
|
230
|
+
}
|
|
231
|
+
if (m.role === 'assistant' && m.tool_calls) {
|
|
232
|
+
return {
|
|
233
|
+
role: 'assistant',
|
|
234
|
+
content: (typeof m.content === 'string' ? m.content : null) || null,
|
|
235
|
+
tool_calls: m.tool_calls.map(tc => ({
|
|
236
|
+
id: tc.id,
|
|
237
|
+
type: 'function',
|
|
238
|
+
function: {
|
|
239
|
+
name: tc.function.name,
|
|
240
|
+
arguments: tc.function.arguments,
|
|
241
|
+
},
|
|
242
|
+
})),
|
|
243
|
+
};
|
|
244
|
+
}
|
|
245
|
+
// Content can be string or ContentBlock[] (for vision/multimodal messages)
|
|
246
|
+
return {
|
|
247
|
+
role: m.role,
|
|
248
|
+
content: m.content,
|
|
249
|
+
};
|
|
250
|
+
});
|
|
251
|
+
}
|
|
252
|
+
/**
|
|
253
|
+
* Safely parse JSON string, returning empty object on failure.
|
|
254
|
+
*/
|
|
255
|
+
function safeParseJSON(str) {
|
|
256
|
+
try {
|
|
257
|
+
return JSON.parse(str);
|
|
258
|
+
}
|
|
259
|
+
catch {
|
|
260
|
+
return {};
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/api/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import type { SessionOptions } from './interactive.js';
|
|
2
|
+
export interface HeadlessToolCallSummary {
|
|
3
|
+
id: string;
|
|
4
|
+
name: string;
|
|
5
|
+
status: string;
|
|
6
|
+
durationMs?: number;
|
|
7
|
+
error?: string;
|
|
8
|
+
}
|
|
9
|
+
export interface HeadlessResult {
|
|
10
|
+
response: string;
|
|
11
|
+
exitCode: number;
|
|
12
|
+
durationMs: number;
|
|
13
|
+
messageCount: number;
|
|
14
|
+
toolCallCount: number;
|
|
15
|
+
sessionId?: string;
|
|
16
|
+
handoffFile?: string;
|
|
17
|
+
bundleFile?: string;
|
|
18
|
+
toolCalls?: HeadlessToolCallSummary[];
|
|
19
|
+
error?: string;
|
|
20
|
+
}
|
|
21
|
+
export declare function headlessMode(prompt: string, options: SessionOptions): Promise<HeadlessResult>;
|
|
22
|
+
//# sourceMappingURL=headless.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"headless.d.ts","sourceRoot":"","sources":["../../src/cli/headless.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAA;AAEtD,MAAM,WAAW,uBAAuB;IACtC,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,cAAc;IAC7B,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,aAAa,EAAE,MAAM,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS,CAAC,EAAE,uBAAuB,EAAE,CAAC;IACtC,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,wBAAsB,YAAY,CAChC,MAAM,EAAE,MAAM,EACd,OAAO,EAAE,cAAc,GACtB,OAAO,CAAC,cAAc,CAAC,CA4HzB"}
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
import { loadConfig } from '../config/loader.js';
|
|
2
|
+
import { AgentLoop } from '../core/agent-loop.js';
|
|
3
|
+
import { saveSession, writeExecutionBundle, writeSessionHandoff } from '../core/session.js';
|
|
4
|
+
export async function headlessMode(prompt, options) {
|
|
5
|
+
const startTime = Date.now();
|
|
6
|
+
const config = await loadConfig();
|
|
7
|
+
if (options.model) {
|
|
8
|
+
config.model = options.model;
|
|
9
|
+
}
|
|
10
|
+
const approvalMode = (options.approvalMode ?? config.approvalMode ?? 'turbo');
|
|
11
|
+
const sessionId = await saveSession({ approvalMode, lastPrompt: prompt });
|
|
12
|
+
const agent = new AgentLoop(config, {
|
|
13
|
+
approvalMode,
|
|
14
|
+
cwd: process.cwd(),
|
|
15
|
+
onApprovalRequest: async (toolName) => {
|
|
16
|
+
if (approvalMode === 'plan') {
|
|
17
|
+
// In plan mode, only reject tools that need approval (write/edit/bash)
|
|
18
|
+
// Read-only tools (read_file, glob, grep_search) have approval='never'
|
|
19
|
+
// and won't reach this callback
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
return true;
|
|
23
|
+
},
|
|
24
|
+
});
|
|
25
|
+
try {
|
|
26
|
+
const response = await agent.run(prompt);
|
|
27
|
+
const duration = Date.now() - startTime;
|
|
28
|
+
const toolCalls = agent.getToolCallHistory();
|
|
29
|
+
const bundleFile = await writeExecutionBundle({
|
|
30
|
+
sessionId,
|
|
31
|
+
prompt,
|
|
32
|
+
response,
|
|
33
|
+
approvalMode,
|
|
34
|
+
toolCalls: toolCalls.map(toolCall => ({
|
|
35
|
+
id: toolCall.id,
|
|
36
|
+
name: toolCall.name,
|
|
37
|
+
status: toolCall.status,
|
|
38
|
+
durationMs: toolCall.durationMs,
|
|
39
|
+
error: toolCall.error,
|
|
40
|
+
result: toolCall.result,
|
|
41
|
+
})),
|
|
42
|
+
});
|
|
43
|
+
const handoffFile = await writeSessionHandoff({
|
|
44
|
+
sessionId,
|
|
45
|
+
prompt,
|
|
46
|
+
response,
|
|
47
|
+
approvalMode,
|
|
48
|
+
toolCalls: toolCalls.map(toolCall => ({
|
|
49
|
+
name: toolCall.name,
|
|
50
|
+
status: toolCall.status,
|
|
51
|
+
durationMs: toolCall.durationMs,
|
|
52
|
+
error: toolCall.error,
|
|
53
|
+
})),
|
|
54
|
+
});
|
|
55
|
+
await saveSession({
|
|
56
|
+
id: sessionId,
|
|
57
|
+
approvalMode,
|
|
58
|
+
messageCount: agent.getMessages().length,
|
|
59
|
+
toolCallCount: toolCalls.length,
|
|
60
|
+
lastPrompt: prompt,
|
|
61
|
+
lastResponse: response,
|
|
62
|
+
summary: response,
|
|
63
|
+
handoffFile,
|
|
64
|
+
bundleFile,
|
|
65
|
+
});
|
|
66
|
+
return {
|
|
67
|
+
response,
|
|
68
|
+
exitCode: 0,
|
|
69
|
+
durationMs: duration,
|
|
70
|
+
messageCount: agent.getMessages().length,
|
|
71
|
+
toolCallCount: toolCalls.length,
|
|
72
|
+
sessionId,
|
|
73
|
+
handoffFile,
|
|
74
|
+
bundleFile,
|
|
75
|
+
toolCalls: toolCalls.map(toolCall => ({
|
|
76
|
+
id: toolCall.id,
|
|
77
|
+
name: toolCall.name,
|
|
78
|
+
status: toolCall.status,
|
|
79
|
+
durationMs: toolCall.durationMs,
|
|
80
|
+
error: toolCall.error,
|
|
81
|
+
})),
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
catch (err) {
|
|
85
|
+
const error = err.message;
|
|
86
|
+
const bundleFile = await writeExecutionBundle({
|
|
87
|
+
sessionId,
|
|
88
|
+
prompt,
|
|
89
|
+
error,
|
|
90
|
+
approvalMode,
|
|
91
|
+
});
|
|
92
|
+
const handoffFile = await writeSessionHandoff({
|
|
93
|
+
sessionId,
|
|
94
|
+
prompt,
|
|
95
|
+
error,
|
|
96
|
+
approvalMode,
|
|
97
|
+
});
|
|
98
|
+
await saveSession({
|
|
99
|
+
id: sessionId,
|
|
100
|
+
approvalMode,
|
|
101
|
+
messageCount: 0,
|
|
102
|
+
toolCallCount: 0,
|
|
103
|
+
lastPrompt: prompt,
|
|
104
|
+
lastError: error,
|
|
105
|
+
summary: error,
|
|
106
|
+
handoffFile,
|
|
107
|
+
bundleFile,
|
|
108
|
+
});
|
|
109
|
+
return {
|
|
110
|
+
response: '',
|
|
111
|
+
exitCode: 1,
|
|
112
|
+
durationMs: Date.now() - startTime,
|
|
113
|
+
messageCount: 0,
|
|
114
|
+
toolCallCount: 0,
|
|
115
|
+
sessionId,
|
|
116
|
+
handoffFile,
|
|
117
|
+
bundleFile,
|
|
118
|
+
error,
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
//# sourceMappingURL=headless.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"headless.js","sourceRoot":"","sources":["../../src/cli/headless.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":";AAgFA,wBAAsB,GAAG,CAAE,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAaxD"}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { Command } from 'commander';
|
|
3
|
+
import { readFileSync } from 'node:fs';
|
|
4
|
+
import { fileURLToPath } from 'node:url';
|
|
5
|
+
import { dirname, join } from 'node:path';
|
|
6
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
7
|
+
const __dirname = dirname(__filename);
|
|
8
|
+
// Read version from package.json
|
|
9
|
+
const pkgPath = join(__dirname, '..', '..', 'package.json');
|
|
10
|
+
const pkg = JSON.parse(readFileSync(pkgPath, 'utf-8'));
|
|
11
|
+
const program = new Command();
|
|
12
|
+
program
|
|
13
|
+
.name('deepseek-code')
|
|
14
|
+
.description('AI-powered CLI agent for software development')
|
|
15
|
+
.version(pkg.version)
|
|
16
|
+
.argument('[query...]', 'Optional query to run in non-interactive mode')
|
|
17
|
+
.option('-p, --prompt <text>', 'Run a single prompt and exit (non-interactive)')
|
|
18
|
+
.option('-i, --prompt-interactive <text>', 'Run a prompt then continue in interactive mode')
|
|
19
|
+
.option('-m, --model <model>', 'Model to use (e.g. deepseek-chat)')
|
|
20
|
+
.option('-t, --turbo', 'Enable Turbo mode (auto-approve all actions)')
|
|
21
|
+
.option('--approval-mode <mode>', 'Set approval mode: plan, default, auto-edit, turbo')
|
|
22
|
+
.option('--debug', 'Enable debug logging')
|
|
23
|
+
.option('-c, --continue', 'Continue last session')
|
|
24
|
+
.option('-r, --resume [sessionId]', 'Resume a specific session')
|
|
25
|
+
.option('--json', 'JSON output mode (for CI/CD integration)')
|
|
26
|
+
.option('--headless', 'Headless mode (no TUI, pipe-friendly)')
|
|
27
|
+
.option('--theme <name>', 'Set color theme')
|
|
28
|
+
.option('--lang <code>', 'Set language (en, ru, zh)')
|
|
29
|
+
.helpOption(false); // Disable built-in --help, we handle it ourselves
|
|
30
|
+
program.hook('preAction', (thisCommand) => {
|
|
31
|
+
const opts = thisCommand.optsWithGlobals();
|
|
32
|
+
if (opts.debug) {
|
|
33
|
+
process.env.DEEPSEEK_CODE_DEBUG = '1';
|
|
34
|
+
}
|
|
35
|
+
});
|
|
36
|
+
program.action(async (query, opts) => {
|
|
37
|
+
const options = {
|
|
38
|
+
query: query?.join(' ') ?? undefined,
|
|
39
|
+
prompt: opts.prompt,
|
|
40
|
+
promptInteractive: opts.promptInteractive,
|
|
41
|
+
model: opts.model,
|
|
42
|
+
turbo: !!opts.turbo,
|
|
43
|
+
approvalMode: opts.approvalMode,
|
|
44
|
+
debug: !!opts.debug,
|
|
45
|
+
continue_: !!opts.continue,
|
|
46
|
+
resume: opts.resume,
|
|
47
|
+
json: !!opts.json,
|
|
48
|
+
headless: !!opts.headless,
|
|
49
|
+
theme: opts.theme,
|
|
50
|
+
lang: opts.lang,
|
|
51
|
+
};
|
|
52
|
+
// Headless/JSON mode for CI/CD
|
|
53
|
+
if (options.headless || options.json) {
|
|
54
|
+
const { headlessMode } = await import('./headless.js');
|
|
55
|
+
const prompt = options.prompt ?? options.query;
|
|
56
|
+
if (prompt) {
|
|
57
|
+
const result = await headlessMode(prompt, options);
|
|
58
|
+
if (options.json) {
|
|
59
|
+
console.log(JSON.stringify(result));
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
console.log(result.response);
|
|
63
|
+
}
|
|
64
|
+
process.exit(result.exitCode ?? 0);
|
|
65
|
+
}
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
const { startInteractiveSession } = await import('./interactive.js');
|
|
69
|
+
await startInteractiveSession(options);
|
|
70
|
+
});
|
|
71
|
+
export async function run(args) {
|
|
72
|
+
// Handle --help and --version before loading Ink
|
|
73
|
+
if (args.includes('--help') || args.includes('-h')) {
|
|
74
|
+
program.outputHelp();
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
if (args.includes('--version') || args.includes('-V')) {
|
|
78
|
+
console.log(pkg.version);
|
|
79
|
+
return;
|
|
80
|
+
}
|
|
81
|
+
// Handle --help passed via commander's parse
|
|
82
|
+
// Commander may intercept --help before action, but we also handle it here
|
|
83
|
+
await program.parseAsync(args, { from: 'user' });
|
|
84
|
+
}
|
|
85
|
+
// Always run when this module is loaded
|
|
86
|
+
run(process.argv).catch((err) => {
|
|
87
|
+
console.error('Fatal error:', err);
|
|
88
|
+
process.exit(1);
|
|
89
|
+
});
|
|
90
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
export interface CliOptions {
|
|
2
|
+
query?: string;
|
|
3
|
+
prompt?: string;
|
|
4
|
+
promptInteractive?: string;
|
|
5
|
+
model?: string;
|
|
6
|
+
turbo?: boolean;
|
|
7
|
+
approvalMode?: string;
|
|
8
|
+
debug?: boolean;
|
|
9
|
+
continue_?: boolean;
|
|
10
|
+
resume?: string;
|
|
11
|
+
json?: boolean;
|
|
12
|
+
headless?: boolean;
|
|
13
|
+
theme?: string;
|
|
14
|
+
lang?: string;
|
|
15
|
+
}
|
|
16
|
+
export type SessionOptions = CliOptions;
|
|
17
|
+
export declare function startInteractiveSession(options: SessionOptions): Promise<void>;
|
|
18
|
+
//# sourceMappingURL=interactive.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"interactive.d.ts","sourceRoot":"","sources":["../../src/cli/interactive.ts"],"names":[],"mappings":"AAQA,MAAM,WAAW,UAAU;IACzB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,MAAM,MAAM,cAAc,GAAG,UAAU,CAAA;AAEvC,wBAAsB,uBAAuB,CAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC,CAkFrF"}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { render } from 'ink';
|
|
2
|
+
import React from 'react';
|
|
3
|
+
import { App } from '../ui/app.js';
|
|
4
|
+
import { ErrorBoundary } from '../ui/error-boundary.js';
|
|
5
|
+
import { loadConfig } from '../config/loader.js';
|
|
6
|
+
import { themeManager } from '../core/themes.js';
|
|
7
|
+
import { i18n } from '../core/i18n.js';
|
|
8
|
+
export async function startInteractiveSession(options) {
|
|
9
|
+
const config = await loadConfig();
|
|
10
|
+
// Apply CLI theme/lang overrides
|
|
11
|
+
if (options.theme) {
|
|
12
|
+
themeManager.setTheme(options.theme);
|
|
13
|
+
}
|
|
14
|
+
if (options.lang) {
|
|
15
|
+
i18n.setLocale(options.lang);
|
|
16
|
+
}
|
|
17
|
+
let cleanup = null;
|
|
18
|
+
// Double Ctrl+C guard: first Ctrl+C in Ready shows hint, second within 2s exits.
|
|
19
|
+
// App registers process.__agentSoftCancel while isProcessing=true.
|
|
20
|
+
let pendingExitTimer = null;
|
|
21
|
+
const onSIGINT = () => {
|
|
22
|
+
const proc = process;
|
|
23
|
+
// 1) Agent is running — soft cancel, never exit
|
|
24
|
+
if (proc.__agentSoftCancel) {
|
|
25
|
+
proc.__agentSoftCancel();
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
// 2) Agent just finished (pending exit flag from app.tsx) — exit immediately
|
|
29
|
+
if (proc.__pendingExit) {
|
|
30
|
+
proc.__pendingExit = false;
|
|
31
|
+
if (cleanup)
|
|
32
|
+
cleanup();
|
|
33
|
+
process.exit(0);
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
// 3) Ready state — double Ctrl+C guard
|
|
37
|
+
if (pendingExitTimer) {
|
|
38
|
+
// Second Ctrl+C within 2s — exit
|
|
39
|
+
clearTimeout(pendingExitTimer);
|
|
40
|
+
pendingExitTimer = null;
|
|
41
|
+
if (cleanup)
|
|
42
|
+
cleanup();
|
|
43
|
+
process.exit(0);
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
// First Ctrl+C — show hint, start timer
|
|
47
|
+
pendingExitTimer = setTimeout(() => {
|
|
48
|
+
pendingExitTimer = null;
|
|
49
|
+
}, 2000);
|
|
50
|
+
// Write hint to stderr so it appears even if TUI is rendering
|
|
51
|
+
process.stderr.write(`\n\x1b[33m⚠ ${i18n.t('ctrlCHint')}\x1b[0m\n`);
|
|
52
|
+
};
|
|
53
|
+
// SIGTERM: always exit gracefully regardless of agent state
|
|
54
|
+
const onSIGTERM = () => {
|
|
55
|
+
if (cleanup)
|
|
56
|
+
cleanup();
|
|
57
|
+
process.exit(0);
|
|
58
|
+
};
|
|
59
|
+
process.on('SIGINT', onSIGINT);
|
|
60
|
+
process.on('SIGTERM', onSIGTERM);
|
|
61
|
+
const { waitUntilExit, clear } = render(React.createElement(ErrorBoundary, null, React.createElement(App, { config, options })), { exitOnCtrlC: false } // App owns Ctrl+C: useInput handles raw-mode, onSIGINT handles signal
|
|
62
|
+
);
|
|
63
|
+
cleanup = () => {
|
|
64
|
+
clear();
|
|
65
|
+
if (pendingExitTimer) {
|
|
66
|
+
clearTimeout(pendingExitTimer);
|
|
67
|
+
pendingExitTimer = null;
|
|
68
|
+
}
|
|
69
|
+
process.removeListener('SIGINT', onSIGINT);
|
|
70
|
+
process.removeListener('SIGTERM', onSIGTERM);
|
|
71
|
+
};
|
|
72
|
+
await waitUntilExit();
|
|
73
|
+
cleanup();
|
|
74
|
+
}
|
|
75
|
+
//# sourceMappingURL=interactive.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"interactive.js","sourceRoot":"","sources":["../../src/cli/interactive.ts"],"names":[],"mappings":"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"}
|