@ebowwa/coder 0.7.64 → 0.7.66
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/dist/index.js +36233 -32
- package/dist/interfaces/ui/terminal/cli/index.js +34318 -158
- package/dist/interfaces/ui/terminal/native/README.md +53 -0
- package/dist/interfaces/ui/terminal/native/claude_code_native.darwin-x64.node +0 -0
- package/dist/interfaces/ui/terminal/native/claude_code_native.dylib +0 -0
- package/dist/interfaces/ui/terminal/native/index.d.ts +0 -0
- package/dist/interfaces/ui/terminal/native/index.darwin-arm64.node +0 -0
- package/dist/interfaces/ui/terminal/native/index.js +43 -0
- package/dist/interfaces/ui/terminal/native/index.node +0 -0
- package/dist/interfaces/ui/terminal/native/package.json +34 -0
- package/dist/native/README.md +53 -0
- package/dist/native/claude_code_native.darwin-x64.node +0 -0
- package/dist/native/claude_code_native.dylib +0 -0
- package/dist/native/index.d.ts +0 -480
- package/dist/native/index.darwin-arm64.node +0 -0
- package/dist/native/index.js +43 -1625
- package/dist/native/index.node +0 -0
- package/dist/native/package.json +34 -0
- package/native/index.darwin-arm64.node +0 -0
- package/native/index.js +33 -19
- package/package.json +3 -2
- package/packages/src/core/agent-loop/__tests__/compaction.test.ts +17 -14
- package/packages/src/core/agent-loop/compaction.ts +6 -2
- package/packages/src/core/agent-loop/index.ts +2 -0
- package/packages/src/core/agent-loop/loop-state.ts +1 -1
- package/packages/src/core/agent-loop/turn-executor.ts +4 -0
- package/packages/src/core/agent-loop/types.ts +4 -0
- package/packages/src/core/api-client-impl.ts +377 -176
- package/packages/src/core/cognitive-security/hooks.ts +2 -1
- package/packages/src/core/config/todo +7 -0
- package/packages/src/core/context/__tests__/integration.test.ts +334 -0
- package/packages/src/core/context/compaction.ts +170 -0
- package/packages/src/core/context/constants.ts +58 -0
- package/packages/src/core/context/extraction.ts +85 -0
- package/packages/src/core/context/index.ts +66 -0
- package/packages/src/core/context/summarization.ts +251 -0
- package/packages/src/core/context/token-estimation.ts +98 -0
- package/packages/src/core/context/types.ts +59 -0
- package/packages/src/core/models.ts +81 -4
- package/packages/src/core/normalizers/todo +5 -1
- package/packages/src/core/providers/README.md +230 -0
- package/packages/src/core/providers/__tests__/providers.test.ts +135 -0
- package/packages/src/core/providers/index.ts +419 -0
- package/packages/src/core/providers/types.ts +132 -0
- package/packages/src/core/retry.ts +10 -0
- package/packages/src/ecosystem/tools/index.ts +174 -0
- package/packages/src/index.ts +23 -2
- package/packages/src/interfaces/ui/index.ts +17 -20
- package/packages/src/interfaces/ui/spinner.ts +2 -2
- package/packages/src/interfaces/ui/terminal/bridge/index.ts +370 -0
- package/packages/src/interfaces/ui/terminal/bridge/ipc.ts +829 -0
- package/packages/src/interfaces/ui/terminal/bridge/screen-export.ts +968 -0
- package/packages/src/interfaces/ui/terminal/bridge/types.ts +226 -0
- package/packages/src/interfaces/ui/terminal/bridge/useBridge.ts +210 -0
- package/packages/src/interfaces/ui/terminal/cli/bootstrap.ts +132 -0
- package/packages/src/interfaces/ui/terminal/cli/index.ts +200 -13
- package/packages/src/interfaces/ui/terminal/cli/interactive/index.ts +110 -0
- package/packages/src/interfaces/ui/terminal/cli/interactive/input-handler.ts +402 -0
- package/packages/src/interfaces/ui/terminal/cli/interactive/interactive-runner.ts +820 -0
- package/packages/src/interfaces/ui/terminal/cli/interactive/message-store.ts +299 -0
- package/packages/src/interfaces/ui/terminal/cli/interactive/types.ts +274 -0
- package/packages/src/interfaces/ui/terminal/shared/index.ts +13 -0
- package/packages/src/interfaces/ui/terminal/shared/query.ts +9 -3
- package/packages/src/interfaces/ui/terminal/shared/setup.ts +5 -1
- package/packages/src/interfaces/ui/terminal/shared/spinner-frames.ts +73 -0
- package/packages/src/interfaces/ui/terminal/shared/status-line.ts +10 -2
- package/packages/src/native/index.ts +404 -27
- package/packages/src/native/tui_v2_types.ts +39 -0
- package/packages/src/teammates/coordination.test.ts +279 -0
- package/packages/src/teammates/coordination.ts +646 -0
- package/packages/src/teammates/index.ts +95 -25
- package/packages/src/teammates/integration.test.ts +272 -0
- package/packages/src/teammates/runner.test.ts +235 -0
- package/packages/src/teammates/runner.ts +750 -0
- package/packages/src/teammates/schemas.ts +673 -0
- package/packages/src/types/index.ts +1 -0
- package/packages/src/core/context-compaction.ts +0 -578
- package/packages/src/interfaces/ui/Screenshot 2026-03-02 at 9.23.10/342/200/257PM.png +0 -0
- package/packages/src/interfaces/ui/Screenshot 2026-03-03 at 10.55.11/342/200/257AM.png +0 -0
- package/packages/src/interfaces/ui/terminal/tui/HelpPanel.tsx +0 -262
- package/packages/src/interfaces/ui/terminal/tui/InputContext.tsx +0 -232
- package/packages/src/interfaces/ui/terminal/tui/InputField.tsx +0 -62
- package/packages/src/interfaces/ui/terminal/tui/InteractiveTUI.tsx +0 -537
- package/packages/src/interfaces/ui/terminal/tui/MessageArea.tsx +0 -107
- package/packages/src/interfaces/ui/terminal/tui/MessageStore.tsx +0 -240
- package/packages/src/interfaces/ui/terminal/tui/StatusBar.tsx +0 -54
- package/packages/src/interfaces/ui/terminal/tui/commands.ts +0 -438
- package/packages/src/interfaces/ui/terminal/tui/components/InteractiveElements.tsx +0 -584
- package/packages/src/interfaces/ui/terminal/tui/components/MultilineInput.tsx +0 -614
- package/packages/src/interfaces/ui/terminal/tui/components/PaneManager.tsx +0 -333
- package/packages/src/interfaces/ui/terminal/tui/components/Sidebar.tsx +0 -604
- package/packages/src/interfaces/ui/terminal/tui/components/index.ts +0 -118
- package/packages/src/interfaces/ui/terminal/tui/console.ts +0 -49
- package/packages/src/interfaces/ui/terminal/tui/index.ts +0 -90
- package/packages/src/interfaces/ui/terminal/tui/run.tsx +0 -42
- package/packages/src/interfaces/ui/terminal/tui/spinner.ts +0 -69
- package/packages/src/interfaces/ui/terminal/tui/tui-app.tsx +0 -390
- package/packages/src/interfaces/ui/terminal/tui/tui-footer.ts +0 -422
- package/packages/src/interfaces/ui/terminal/tui/types.ts +0 -186
- package/packages/src/interfaces/ui/terminal/tui/useInputHandler.ts +0 -104
- package/packages/src/interfaces/ui/terminal/tui/useNativeInput.ts +0 -239
|
@@ -1,438 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* TUI Commands
|
|
3
|
-
* Command handlers for slash commands in the TUI
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import type { ClaudeModel, Message as ApiMessage } from "../../../../types/index.js";
|
|
7
|
-
import {
|
|
8
|
-
createCheckpoint,
|
|
9
|
-
registerCheckpoint,
|
|
10
|
-
restoreCheckpoint,
|
|
11
|
-
applyCheckpoint,
|
|
12
|
-
listCheckpoints,
|
|
13
|
-
undoCheckpoint,
|
|
14
|
-
redoCheckpoint,
|
|
15
|
-
getNavigationStatus,
|
|
16
|
-
getCheckpointSummary,
|
|
17
|
-
formatCheckpoint,
|
|
18
|
-
} from "../../../../core/checkpoints.js";
|
|
19
|
-
import { formatCost } from "../../../../core/api-client.js";
|
|
20
|
-
import type { CommandContext, UIMessage } from "./types.js";
|
|
21
|
-
import { VERSION as CODER_VERSION, getContextWindow } from "../shared/status-line.js";
|
|
22
|
-
import {
|
|
23
|
-
AVAILABLE_MODELS,
|
|
24
|
-
getContextWindow as getModelContextWindow,
|
|
25
|
-
getMaxOutput,
|
|
26
|
-
} from "../../../../core/models.js";
|
|
27
|
-
import { getHelpText as getSectionedHelpText, getCompactHelpText, type HelpSection } from "./HelpPanel.js";
|
|
28
|
-
|
|
29
|
-
// Debug mode state (module-level)
|
|
30
|
-
let debugMode = false;
|
|
31
|
-
|
|
32
|
-
/**
|
|
33
|
-
* Get help text for commands
|
|
34
|
-
* @param section - Optional section to display
|
|
35
|
-
*/
|
|
36
|
-
export function getHelpText(section?: string): string {
|
|
37
|
-
return getSectionedHelpText(section);
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* Get all help sections
|
|
42
|
-
*/
|
|
43
|
-
export function getHelpSections(): HelpSection[] {
|
|
44
|
-
return [
|
|
45
|
-
{ id: "general", title: "general", content: "" },
|
|
46
|
-
{ id: "commands", title: "commands", content: "" },
|
|
47
|
-
{ id: "checkpoints", title: "checkpoints", content: "" },
|
|
48
|
-
{ id: "custom", title: "custom-commands", content: "" },
|
|
49
|
-
{ id: "keybindings", title: "keybindings", content: "" },
|
|
50
|
-
];
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
/**
|
|
54
|
-
* Handle a slash command
|
|
55
|
-
* Returns true if the command was handled
|
|
56
|
-
*/
|
|
57
|
-
export async function handleCommand(cmd: string, context: CommandContext): Promise<boolean> {
|
|
58
|
-
const parts = cmd.slice(1).split(" ");
|
|
59
|
-
const command = parts[0]?.toLowerCase();
|
|
60
|
-
const rest = parts.slice(1).join(" ");
|
|
61
|
-
|
|
62
|
-
const {
|
|
63
|
-
sessionId,
|
|
64
|
-
setSessionId,
|
|
65
|
-
model,
|
|
66
|
-
setModel,
|
|
67
|
-
apiMessages,
|
|
68
|
-
setApiMessages,
|
|
69
|
-
setMessages,
|
|
70
|
-
processedCountRef,
|
|
71
|
-
totalCost,
|
|
72
|
-
setTotalCost,
|
|
73
|
-
totalTokens,
|
|
74
|
-
setTotalTokens,
|
|
75
|
-
permissionMode,
|
|
76
|
-
tools,
|
|
77
|
-
workingDirectory,
|
|
78
|
-
sessionStore,
|
|
79
|
-
addSystemMessage,
|
|
80
|
-
messagesLength,
|
|
81
|
-
onExit,
|
|
82
|
-
exit,
|
|
83
|
-
setSessionSelectMode,
|
|
84
|
-
setSelectableSessions,
|
|
85
|
-
helpMode,
|
|
86
|
-
setHelpMode,
|
|
87
|
-
helpSection,
|
|
88
|
-
setHelpSection,
|
|
89
|
-
} = context;
|
|
90
|
-
|
|
91
|
-
switch (command) {
|
|
92
|
-
case "exit":
|
|
93
|
-
case "quit":
|
|
94
|
-
case "q":
|
|
95
|
-
onExit();
|
|
96
|
-
exit();
|
|
97
|
-
return true;
|
|
98
|
-
|
|
99
|
-
case "help":
|
|
100
|
-
case "?":
|
|
101
|
-
// If section specified, show that section; otherwise enter help mode
|
|
102
|
-
if (rest) {
|
|
103
|
-
addSystemMessage(getHelpText(rest));
|
|
104
|
-
} else {
|
|
105
|
-
// Enter interactive help mode
|
|
106
|
-
setHelpMode(true);
|
|
107
|
-
setHelpSection(0);
|
|
108
|
-
}
|
|
109
|
-
return true;
|
|
110
|
-
|
|
111
|
-
case "keybindings":
|
|
112
|
-
case "keys":
|
|
113
|
-
addSystemMessage(getHelpText("keybindings"));
|
|
114
|
-
return true;
|
|
115
|
-
|
|
116
|
-
case "clear":
|
|
117
|
-
setApiMessages([]);
|
|
118
|
-
setMessages([]);
|
|
119
|
-
processedCountRef.current = 0;
|
|
120
|
-
addSystemMessage("Conversation cleared.");
|
|
121
|
-
return true;
|
|
122
|
-
|
|
123
|
-
case "new":
|
|
124
|
-
try {
|
|
125
|
-
// Create a new session
|
|
126
|
-
const newSessionId = await sessionStore.createSession({
|
|
127
|
-
model,
|
|
128
|
-
workingDirectory,
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
// Clear all state
|
|
132
|
-
setSessionId(newSessionId);
|
|
133
|
-
setApiMessages([]);
|
|
134
|
-
setMessages([]);
|
|
135
|
-
processedCountRef.current = 0;
|
|
136
|
-
setTotalCost(0);
|
|
137
|
-
setTotalTokens(0);
|
|
138
|
-
|
|
139
|
-
addSystemMessage(`Started new session: ${newSessionId}`);
|
|
140
|
-
} catch (error) {
|
|
141
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
142
|
-
addSystemMessage(`Failed to create new session: ${errorMessage}`);
|
|
143
|
-
}
|
|
144
|
-
return true;
|
|
145
|
-
|
|
146
|
-
case "compact":
|
|
147
|
-
addSystemMessage("Forcing context compaction...");
|
|
148
|
-
return true;
|
|
149
|
-
|
|
150
|
-
case "model":
|
|
151
|
-
if (rest) {
|
|
152
|
-
setModel(rest as ClaudeModel);
|
|
153
|
-
addSystemMessage(`Switched to model: ${rest}`);
|
|
154
|
-
} else {
|
|
155
|
-
addSystemMessage(`Current model: ${model}`);
|
|
156
|
-
}
|
|
157
|
-
return true;
|
|
158
|
-
|
|
159
|
-
case "tools": {
|
|
160
|
-
const toolList = tools.map((t) => ` ${t.name}: ${t.description.split(".")[0]}`).join("\n");
|
|
161
|
-
addSystemMessage(`Available tools:\n${toolList}`);
|
|
162
|
-
return true;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
case "cost":
|
|
166
|
-
addSystemMessage(`Total cost: ${formatCost(totalCost)}`);
|
|
167
|
-
return true;
|
|
168
|
-
|
|
169
|
-
case "status":
|
|
170
|
-
case "session":
|
|
171
|
-
addSystemMessage(`Session Status:
|
|
172
|
-
ID: ${sessionId}
|
|
173
|
-
Model: ${model}
|
|
174
|
-
Messages: ${apiMessages.length}
|
|
175
|
-
Total cost: ${formatCost(totalCost)}
|
|
176
|
-
Permission mode: ${permissionMode}
|
|
177
|
-
Tokens: ${totalTokens}
|
|
178
|
-
Debug mode: ${debugMode ? "on" : "off"}`);
|
|
179
|
-
return true;
|
|
180
|
-
|
|
181
|
-
case "version":
|
|
182
|
-
case "ver":
|
|
183
|
-
addSystemMessage(`Coder v${CODER_VERSION}
|
|
184
|
-
Model: ${model}
|
|
185
|
-
Context: ${getContextWindow(model).toLocaleString()} tokens
|
|
186
|
-
Node: ${process.version}
|
|
187
|
-
Platform: ${process.platform} ${process.arch}`);
|
|
188
|
-
return true;
|
|
189
|
-
|
|
190
|
-
case "models":
|
|
191
|
-
const modelList = AVAILABLE_MODELS.map(
|
|
192
|
-
(m) => {
|
|
193
|
-
const contextStr = `${(m.contextWindow / 1000).toFixed(0)}k context`;
|
|
194
|
-
const outputStr = m.maxOutput ? `, ${(m.maxOutput / 1000).toFixed(0)}k output` : "";
|
|
195
|
-
return ` ${m.id.padEnd(22)} ${m.fullName} (${contextStr}${outputStr})`;
|
|
196
|
-
}
|
|
197
|
-
).join("\n");
|
|
198
|
-
addSystemMessage(`Available models:\n${modelList}\n\nUse /model <id> to switch`);
|
|
199
|
-
return true;
|
|
200
|
-
|
|
201
|
-
case "debug":
|
|
202
|
-
debugMode = !debugMode;
|
|
203
|
-
addSystemMessage(`Debug mode: ${debugMode ? "ON" : "OFF"}`);
|
|
204
|
-
return true;
|
|
205
|
-
|
|
206
|
-
case "export":
|
|
207
|
-
try {
|
|
208
|
-
const format = (rest || "markdown") as "jsonl" | "json" | "markdown";
|
|
209
|
-
const outputPath = await sessionStore.exportSession(sessionId, format);
|
|
210
|
-
addSystemMessage(`Session exported to: ${outputPath}`);
|
|
211
|
-
} catch (error) {
|
|
212
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
213
|
-
addSystemMessage(`Export failed: ${errorMessage}`);
|
|
214
|
-
}
|
|
215
|
-
return true;
|
|
216
|
-
|
|
217
|
-
case "checkpoint":
|
|
218
|
-
case "cp":
|
|
219
|
-
if (!rest) {
|
|
220
|
-
addSystemMessage("Usage: /checkpoint <label>\nExample: /checkpoint before-refactor");
|
|
221
|
-
return true;
|
|
222
|
-
}
|
|
223
|
-
try {
|
|
224
|
-
const checkpoint = await createCheckpoint(sessionId, apiMessages, {
|
|
225
|
-
label: rest,
|
|
226
|
-
model,
|
|
227
|
-
workingDirectory,
|
|
228
|
-
totalCost,
|
|
229
|
-
trackFiles: true,
|
|
230
|
-
});
|
|
231
|
-
|
|
232
|
-
await registerCheckpoint(sessionId, checkpoint.id);
|
|
233
|
-
const summary = getCheckpointSummary(checkpoint);
|
|
234
|
-
|
|
235
|
-
addSystemMessage(`✓ Checkpoint saved: ${formatCheckpoint(checkpoint)}${summary ? `\n ${summary}` : ""}`);
|
|
236
|
-
} catch (error) {
|
|
237
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
238
|
-
addSystemMessage(`Failed to create checkpoint: ${errorMessage}`);
|
|
239
|
-
}
|
|
240
|
-
return true;
|
|
241
|
-
|
|
242
|
-
case "checkpoints":
|
|
243
|
-
case "cps":
|
|
244
|
-
try {
|
|
245
|
-
const checkpoints = await listCheckpoints(sessionId);
|
|
246
|
-
if (checkpoints.length === 0) {
|
|
247
|
-
addSystemMessage("No checkpoints saved.");
|
|
248
|
-
} else {
|
|
249
|
-
const list = checkpoints.map((cp) => ` ${formatCheckpoint(cp)}`).join("\n");
|
|
250
|
-
addSystemMessage(`Checkpoints:\n${list}`);
|
|
251
|
-
}
|
|
252
|
-
} catch (error) {
|
|
253
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
254
|
-
addSystemMessage(`Failed to list checkpoints: ${errorMessage}`);
|
|
255
|
-
}
|
|
256
|
-
return true;
|
|
257
|
-
|
|
258
|
-
case "restore":
|
|
259
|
-
case "rollback":
|
|
260
|
-
case "rewind":
|
|
261
|
-
if (!rest) {
|
|
262
|
-
addSystemMessage("Usage: /restore <checkpoint-id>\nUse /checkpoints to see available checkpoints");
|
|
263
|
-
return true;
|
|
264
|
-
}
|
|
265
|
-
try {
|
|
266
|
-
const checkpointId = rest.trim();
|
|
267
|
-
const checkpoint = await restoreCheckpoint(sessionId, checkpointId);
|
|
268
|
-
|
|
269
|
-
if (!checkpoint) {
|
|
270
|
-
addSystemMessage(`Checkpoint not found: ${checkpointId}`);
|
|
271
|
-
return true;
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
// Apply checkpoint without files (for TUI simplicity)
|
|
275
|
-
const result = await applyCheckpoint(checkpoint, {
|
|
276
|
-
restoreFiles: false,
|
|
277
|
-
restoreMessages: true,
|
|
278
|
-
});
|
|
279
|
-
|
|
280
|
-
setApiMessages(result.messages);
|
|
281
|
-
processedCountRef.current = result.messages.length;
|
|
282
|
-
setTotalCost(checkpoint.metadata.totalCost);
|
|
283
|
-
|
|
284
|
-
addSystemMessage(`✓ Checkpoint restored: ${result.messages.length} messages (no files changed)`);
|
|
285
|
-
} catch (error) {
|
|
286
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
287
|
-
addSystemMessage(`Failed to restore checkpoint: ${errorMessage}`);
|
|
288
|
-
}
|
|
289
|
-
return true;
|
|
290
|
-
|
|
291
|
-
case "restore-chat":
|
|
292
|
-
if (!rest) {
|
|
293
|
-
addSystemMessage("Usage: /restore-chat <checkpoint-id>");
|
|
294
|
-
return true;
|
|
295
|
-
}
|
|
296
|
-
try {
|
|
297
|
-
const checkpointId = rest.trim();
|
|
298
|
-
const checkpoint = await restoreCheckpoint(sessionId, checkpointId);
|
|
299
|
-
|
|
300
|
-
if (!checkpoint) {
|
|
301
|
-
addSystemMessage(`Checkpoint not found: ${checkpointId}`);
|
|
302
|
-
return true;
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
const result = await applyCheckpoint(checkpoint, {
|
|
306
|
-
restoreFiles: false,
|
|
307
|
-
restoreMessages: true,
|
|
308
|
-
});
|
|
309
|
-
|
|
310
|
-
setApiMessages(result.messages);
|
|
311
|
-
processedCountRef.current = result.messages.length;
|
|
312
|
-
setTotalCost(checkpoint.metadata.totalCost);
|
|
313
|
-
|
|
314
|
-
addSystemMessage(`✓ Chat restored: ${result.messages.length} messages`);
|
|
315
|
-
} catch (error) {
|
|
316
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
317
|
-
addSystemMessage(`Failed to restore chat: ${errorMessage}`);
|
|
318
|
-
}
|
|
319
|
-
return true;
|
|
320
|
-
|
|
321
|
-
case "undo":
|
|
322
|
-
try {
|
|
323
|
-
const result = await undoCheckpoint(sessionId);
|
|
324
|
-
|
|
325
|
-
if (!result.checkpoint) {
|
|
326
|
-
addSystemMessage("Nothing to undo");
|
|
327
|
-
return true;
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
const applyResult = await applyCheckpoint(result.checkpoint, {
|
|
331
|
-
restoreFiles: false,
|
|
332
|
-
restoreMessages: true,
|
|
333
|
-
});
|
|
334
|
-
|
|
335
|
-
setApiMessages(applyResult.messages);
|
|
336
|
-
processedCountRef.current = applyResult.messages.length;
|
|
337
|
-
setTotalCost(result.checkpoint.metadata.totalCost);
|
|
338
|
-
|
|
339
|
-
addSystemMessage(`✓ Undone to: ${formatCheckpoint(result.checkpoint)}${result.canRedo ? "\nUse /redo to go forward" : ""}`);
|
|
340
|
-
} catch (error) {
|
|
341
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
342
|
-
addSystemMessage(`Undo failed: ${errorMessage}`);
|
|
343
|
-
}
|
|
344
|
-
return true;
|
|
345
|
-
|
|
346
|
-
case "redo":
|
|
347
|
-
try {
|
|
348
|
-
const result = await redoCheckpoint(sessionId);
|
|
349
|
-
|
|
350
|
-
if (!result.checkpoint) {
|
|
351
|
-
addSystemMessage("Nothing to redo");
|
|
352
|
-
return true;
|
|
353
|
-
}
|
|
354
|
-
|
|
355
|
-
const applyResult = await applyCheckpoint(result.checkpoint, {
|
|
356
|
-
restoreFiles: false,
|
|
357
|
-
restoreMessages: true,
|
|
358
|
-
});
|
|
359
|
-
|
|
360
|
-
setApiMessages(applyResult.messages);
|
|
361
|
-
processedCountRef.current = applyResult.messages.length;
|
|
362
|
-
setTotalCost(result.checkpoint.metadata.totalCost);
|
|
363
|
-
|
|
364
|
-
addSystemMessage(`✓ Redone to: ${formatCheckpoint(result.checkpoint)}`);
|
|
365
|
-
} catch (error) {
|
|
366
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
367
|
-
addSystemMessage(`Redo failed: ${errorMessage}`);
|
|
368
|
-
}
|
|
369
|
-
return true;
|
|
370
|
-
|
|
371
|
-
case "checkpoint-status":
|
|
372
|
-
case "cps-status":
|
|
373
|
-
try {
|
|
374
|
-
const status = await getNavigationStatus(sessionId);
|
|
375
|
-
addSystemMessage(`Checkpoint Navigation:
|
|
376
|
-
Position: ${status.current}/${status.total}
|
|
377
|
-
Can undo: ${status.canUndo ? "yes" : "no"}
|
|
378
|
-
Can redo: ${status.canRedo ? "yes" : "no"}
|
|
379
|
-
Current: ${status.currentId || "none"}`);
|
|
380
|
-
} catch (error) {
|
|
381
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
382
|
-
addSystemMessage(`Failed to get status: ${errorMessage}`);
|
|
383
|
-
}
|
|
384
|
-
return true;
|
|
385
|
-
|
|
386
|
-
case "sessions":
|
|
387
|
-
case "resume": {
|
|
388
|
-
try {
|
|
389
|
-
// If ID provided directly, try to resume
|
|
390
|
-
if (rest) {
|
|
391
|
-
const loaded = await sessionStore.resumeSession(rest.trim());
|
|
392
|
-
if (loaded) {
|
|
393
|
-
setApiMessages(loaded.messages);
|
|
394
|
-
processedCountRef.current = loaded.messages.length;
|
|
395
|
-
setMessages([]); // Clear UI messages, will be rebuilt from apiMessages
|
|
396
|
-
const loadedModel = loaded.metadata?.model as ClaudeModel | undefined;
|
|
397
|
-
if (loadedModel) setModel(loadedModel);
|
|
398
|
-
const loadedCost = loaded.metadata?.totalCost as number | undefined;
|
|
399
|
-
if (typeof loadedCost === "number") setTotalCost(loadedCost);
|
|
400
|
-
const loadedTokens = loaded.metadata?.totalTokens as number | undefined;
|
|
401
|
-
if (typeof loadedTokens === "number") setTotalTokens(loadedTokens);
|
|
402
|
-
addSystemMessage(`Resumed session: ${rest.trim()} (${loaded.messages.length} messages)`);
|
|
403
|
-
} else {
|
|
404
|
-
addSystemMessage(`Session not found: ${rest.trim()}`);
|
|
405
|
-
}
|
|
406
|
-
return true;
|
|
407
|
-
}
|
|
408
|
-
|
|
409
|
-
// Otherwise, list sessions with numbers for selection
|
|
410
|
-
const sessions = await sessionStore.listSessions(15);
|
|
411
|
-
if (sessions.length === 0) {
|
|
412
|
-
addSystemMessage("No recent sessions found.");
|
|
413
|
-
} else {
|
|
414
|
-
// Store sessions for selection
|
|
415
|
-
setSelectableSessions(sessions);
|
|
416
|
-
setSessionSelectMode(true);
|
|
417
|
-
|
|
418
|
-
const list = sessions.map((s, i) => {
|
|
419
|
-
const id = s.id || "unknown";
|
|
420
|
-
const model = (s.metadata as Record<string, unknown>)?.model || "unknown";
|
|
421
|
-
const msgs = s.messageCount || 0;
|
|
422
|
-
const date = s.lastActivity ? new Date(s.lastActivity).toLocaleDateString() : "unknown";
|
|
423
|
-
return ` ${String(i + 1).padStart(2)}. ${String(id).slice(0, 8)} ${String(model).padEnd(20)} ${String(msgs).padStart(3)} msgs ${date}`;
|
|
424
|
-
}).join("\n");
|
|
425
|
-
addSystemMessage(`Select a session to resume:\n${list}\n\nType the number (1-${sessions.length}) to resume, or any other key to cancel.`);
|
|
426
|
-
}
|
|
427
|
-
} catch (error) {
|
|
428
|
-
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
429
|
-
addSystemMessage(`Failed to list sessions: ${errorMessage}`);
|
|
430
|
-
}
|
|
431
|
-
return true;
|
|
432
|
-
}
|
|
433
|
-
|
|
434
|
-
default:
|
|
435
|
-
addSystemMessage(`Unknown command: /${command}\nType /help for available commands.`);
|
|
436
|
-
return true;
|
|
437
|
-
}
|
|
438
|
-
}
|