@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.
Files changed (101) hide show
  1. package/dist/index.js +36233 -32
  2. package/dist/interfaces/ui/terminal/cli/index.js +34318 -158
  3. package/dist/interfaces/ui/terminal/native/README.md +53 -0
  4. package/dist/interfaces/ui/terminal/native/claude_code_native.darwin-x64.node +0 -0
  5. package/dist/interfaces/ui/terminal/native/claude_code_native.dylib +0 -0
  6. package/dist/interfaces/ui/terminal/native/index.d.ts +0 -0
  7. package/dist/interfaces/ui/terminal/native/index.darwin-arm64.node +0 -0
  8. package/dist/interfaces/ui/terminal/native/index.js +43 -0
  9. package/dist/interfaces/ui/terminal/native/index.node +0 -0
  10. package/dist/interfaces/ui/terminal/native/package.json +34 -0
  11. package/dist/native/README.md +53 -0
  12. package/dist/native/claude_code_native.darwin-x64.node +0 -0
  13. package/dist/native/claude_code_native.dylib +0 -0
  14. package/dist/native/index.d.ts +0 -480
  15. package/dist/native/index.darwin-arm64.node +0 -0
  16. package/dist/native/index.js +43 -1625
  17. package/dist/native/index.node +0 -0
  18. package/dist/native/package.json +34 -0
  19. package/native/index.darwin-arm64.node +0 -0
  20. package/native/index.js +33 -19
  21. package/package.json +3 -2
  22. package/packages/src/core/agent-loop/__tests__/compaction.test.ts +17 -14
  23. package/packages/src/core/agent-loop/compaction.ts +6 -2
  24. package/packages/src/core/agent-loop/index.ts +2 -0
  25. package/packages/src/core/agent-loop/loop-state.ts +1 -1
  26. package/packages/src/core/agent-loop/turn-executor.ts +4 -0
  27. package/packages/src/core/agent-loop/types.ts +4 -0
  28. package/packages/src/core/api-client-impl.ts +377 -176
  29. package/packages/src/core/cognitive-security/hooks.ts +2 -1
  30. package/packages/src/core/config/todo +7 -0
  31. package/packages/src/core/context/__tests__/integration.test.ts +334 -0
  32. package/packages/src/core/context/compaction.ts +170 -0
  33. package/packages/src/core/context/constants.ts +58 -0
  34. package/packages/src/core/context/extraction.ts +85 -0
  35. package/packages/src/core/context/index.ts +66 -0
  36. package/packages/src/core/context/summarization.ts +251 -0
  37. package/packages/src/core/context/token-estimation.ts +98 -0
  38. package/packages/src/core/context/types.ts +59 -0
  39. package/packages/src/core/models.ts +81 -4
  40. package/packages/src/core/normalizers/todo +5 -1
  41. package/packages/src/core/providers/README.md +230 -0
  42. package/packages/src/core/providers/__tests__/providers.test.ts +135 -0
  43. package/packages/src/core/providers/index.ts +419 -0
  44. package/packages/src/core/providers/types.ts +132 -0
  45. package/packages/src/core/retry.ts +10 -0
  46. package/packages/src/ecosystem/tools/index.ts +174 -0
  47. package/packages/src/index.ts +23 -2
  48. package/packages/src/interfaces/ui/index.ts +17 -20
  49. package/packages/src/interfaces/ui/spinner.ts +2 -2
  50. package/packages/src/interfaces/ui/terminal/bridge/index.ts +370 -0
  51. package/packages/src/interfaces/ui/terminal/bridge/ipc.ts +829 -0
  52. package/packages/src/interfaces/ui/terminal/bridge/screen-export.ts +968 -0
  53. package/packages/src/interfaces/ui/terminal/bridge/types.ts +226 -0
  54. package/packages/src/interfaces/ui/terminal/bridge/useBridge.ts +210 -0
  55. package/packages/src/interfaces/ui/terminal/cli/bootstrap.ts +132 -0
  56. package/packages/src/interfaces/ui/terminal/cli/index.ts +200 -13
  57. package/packages/src/interfaces/ui/terminal/cli/interactive/index.ts +110 -0
  58. package/packages/src/interfaces/ui/terminal/cli/interactive/input-handler.ts +402 -0
  59. package/packages/src/interfaces/ui/terminal/cli/interactive/interactive-runner.ts +820 -0
  60. package/packages/src/interfaces/ui/terminal/cli/interactive/message-store.ts +299 -0
  61. package/packages/src/interfaces/ui/terminal/cli/interactive/types.ts +274 -0
  62. package/packages/src/interfaces/ui/terminal/shared/index.ts +13 -0
  63. package/packages/src/interfaces/ui/terminal/shared/query.ts +9 -3
  64. package/packages/src/interfaces/ui/terminal/shared/setup.ts +5 -1
  65. package/packages/src/interfaces/ui/terminal/shared/spinner-frames.ts +73 -0
  66. package/packages/src/interfaces/ui/terminal/shared/status-line.ts +10 -2
  67. package/packages/src/native/index.ts +404 -27
  68. package/packages/src/native/tui_v2_types.ts +39 -0
  69. package/packages/src/teammates/coordination.test.ts +279 -0
  70. package/packages/src/teammates/coordination.ts +646 -0
  71. package/packages/src/teammates/index.ts +95 -25
  72. package/packages/src/teammates/integration.test.ts +272 -0
  73. package/packages/src/teammates/runner.test.ts +235 -0
  74. package/packages/src/teammates/runner.ts +750 -0
  75. package/packages/src/teammates/schemas.ts +673 -0
  76. package/packages/src/types/index.ts +1 -0
  77. package/packages/src/core/context-compaction.ts +0 -578
  78. package/packages/src/interfaces/ui/Screenshot 2026-03-02 at 9.23.10/342/200/257PM.png +0 -0
  79. package/packages/src/interfaces/ui/Screenshot 2026-03-03 at 10.55.11/342/200/257AM.png +0 -0
  80. package/packages/src/interfaces/ui/terminal/tui/HelpPanel.tsx +0 -262
  81. package/packages/src/interfaces/ui/terminal/tui/InputContext.tsx +0 -232
  82. package/packages/src/interfaces/ui/terminal/tui/InputField.tsx +0 -62
  83. package/packages/src/interfaces/ui/terminal/tui/InteractiveTUI.tsx +0 -537
  84. package/packages/src/interfaces/ui/terminal/tui/MessageArea.tsx +0 -107
  85. package/packages/src/interfaces/ui/terminal/tui/MessageStore.tsx +0 -240
  86. package/packages/src/interfaces/ui/terminal/tui/StatusBar.tsx +0 -54
  87. package/packages/src/interfaces/ui/terminal/tui/commands.ts +0 -438
  88. package/packages/src/interfaces/ui/terminal/tui/components/InteractiveElements.tsx +0 -584
  89. package/packages/src/interfaces/ui/terminal/tui/components/MultilineInput.tsx +0 -614
  90. package/packages/src/interfaces/ui/terminal/tui/components/PaneManager.tsx +0 -333
  91. package/packages/src/interfaces/ui/terminal/tui/components/Sidebar.tsx +0 -604
  92. package/packages/src/interfaces/ui/terminal/tui/components/index.ts +0 -118
  93. package/packages/src/interfaces/ui/terminal/tui/console.ts +0 -49
  94. package/packages/src/interfaces/ui/terminal/tui/index.ts +0 -90
  95. package/packages/src/interfaces/ui/terminal/tui/run.tsx +0 -42
  96. package/packages/src/interfaces/ui/terminal/tui/spinner.ts +0 -69
  97. package/packages/src/interfaces/ui/terminal/tui/tui-app.tsx +0 -390
  98. package/packages/src/interfaces/ui/terminal/tui/tui-footer.ts +0 -422
  99. package/packages/src/interfaces/ui/terminal/tui/types.ts +0 -186
  100. package/packages/src/interfaces/ui/terminal/tui/useInputHandler.ts +0 -104
  101. 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
- }