mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-19 01:04:27 +00:00
191 lines
5.7 KiB
TypeScript
191 lines
5.7 KiB
TypeScript
/**
|
|
* Component for displaying bash command execution with streaming output.
|
|
*/
|
|
|
|
import { Container, Loader, Spacer, Text, type TUI } from "@mariozechner/pi-tui";
|
|
import stripAnsi from "strip-ansi";
|
|
import {
|
|
DEFAULT_MAX_BYTES,
|
|
DEFAULT_MAX_LINES,
|
|
type TruncationResult,
|
|
truncateTail,
|
|
} from "../../../core/tools/truncate.js";
|
|
import { theme } from "../theme/theme.js";
|
|
import { DynamicBorder } from "./dynamic-border.js";
|
|
|
|
// Preview line limit when not expanded (matches tool execution behavior)
|
|
const PREVIEW_LINES = 20;
|
|
|
|
export class BashExecutionComponent extends Container {
|
|
private command: string;
|
|
private outputLines: string[] = [];
|
|
private status: "running" | "complete" | "cancelled" | "error" = "running";
|
|
private exitCode: number | null = null;
|
|
private loader: Loader;
|
|
private truncationResult?: TruncationResult;
|
|
private fullOutputPath?: string;
|
|
private expanded = false;
|
|
private contentContainer: Container;
|
|
private ui: TUI;
|
|
|
|
constructor(command: string, ui: TUI) {
|
|
super();
|
|
this.command = command;
|
|
this.ui = ui;
|
|
|
|
const borderColor = (str: string) => theme.fg("bashMode", str);
|
|
|
|
// Add spacer
|
|
this.addChild(new Spacer(1));
|
|
|
|
// Top border
|
|
this.addChild(new DynamicBorder(borderColor));
|
|
|
|
// Content container (holds dynamic content between borders)
|
|
this.contentContainer = new Container();
|
|
this.addChild(this.contentContainer);
|
|
|
|
// Command header
|
|
const header = new Text(theme.fg("bashMode", theme.bold(`$ ${command}`)), 1, 0);
|
|
this.contentContainer.addChild(header);
|
|
|
|
// Loader
|
|
this.loader = new Loader(
|
|
ui,
|
|
(spinner) => theme.fg("bashMode", spinner),
|
|
(text) => theme.fg("muted", text),
|
|
"Running... (esc to cancel)",
|
|
);
|
|
this.contentContainer.addChild(this.loader);
|
|
|
|
// Bottom border
|
|
this.addChild(new DynamicBorder(borderColor));
|
|
}
|
|
|
|
/**
|
|
* Set whether the output is expanded (shows full output) or collapsed (preview only).
|
|
*/
|
|
setExpanded(expanded: boolean): void {
|
|
this.expanded = expanded;
|
|
this.updateDisplay();
|
|
}
|
|
|
|
appendOutput(chunk: string): void {
|
|
// Strip ANSI codes and normalize line endings
|
|
// Note: binary data is already sanitized in tui-renderer.ts executeBashCommand
|
|
const clean = stripAnsi(chunk).replace(/\r\n/g, "\n").replace(/\r/g, "\n");
|
|
|
|
// Append to output lines
|
|
const newLines = clean.split("\n");
|
|
if (this.outputLines.length > 0 && newLines.length > 0) {
|
|
// Append first chunk to last line (incomplete line continuation)
|
|
this.outputLines[this.outputLines.length - 1] += newLines[0];
|
|
this.outputLines.push(...newLines.slice(1));
|
|
} else {
|
|
this.outputLines.push(...newLines);
|
|
}
|
|
|
|
this.updateDisplay();
|
|
}
|
|
|
|
setComplete(
|
|
exitCode: number | null,
|
|
cancelled: boolean,
|
|
truncationResult?: TruncationResult,
|
|
fullOutputPath?: string,
|
|
): void {
|
|
this.exitCode = exitCode;
|
|
this.status = cancelled ? "cancelled" : exitCode !== 0 && exitCode !== null ? "error" : "complete";
|
|
this.truncationResult = truncationResult;
|
|
this.fullOutputPath = fullOutputPath;
|
|
|
|
// Stop loader
|
|
this.loader.stop();
|
|
|
|
this.updateDisplay();
|
|
}
|
|
|
|
private updateDisplay(): void {
|
|
// Apply truncation for LLM context limits (same limits as bash tool)
|
|
const fullOutput = this.outputLines.join("\n");
|
|
const contextTruncation = truncateTail(fullOutput, {
|
|
maxLines: DEFAULT_MAX_LINES,
|
|
maxBytes: DEFAULT_MAX_BYTES,
|
|
});
|
|
|
|
// Get the lines to potentially display (after context truncation)
|
|
const availableLines = contextTruncation.content ? contextTruncation.content.split("\n") : [];
|
|
|
|
// Apply preview truncation based on expanded state
|
|
const previewLogicalLines = availableLines.slice(-PREVIEW_LINES);
|
|
const hiddenLineCount = availableLines.length - previewLogicalLines.length;
|
|
|
|
// Rebuild content container
|
|
this.contentContainer.clear();
|
|
|
|
// Command header
|
|
const header = new Text(theme.fg("bashMode", theme.bold(`$ ${this.command}`)), 1, 0);
|
|
this.contentContainer.addChild(header);
|
|
|
|
// Output
|
|
if (availableLines.length > 0) {
|
|
if (this.expanded) {
|
|
// Show all lines
|
|
const displayText = availableLines.map((line) => theme.fg("muted", line)).join("\n");
|
|
this.contentContainer.addChild(new Text("\n" + displayText, 1, 0));
|
|
} else {
|
|
// Render preview lines, then cap at PREVIEW_LINES visual lines
|
|
const tempText = new Text(
|
|
"\n" + previewLogicalLines.map((line) => theme.fg("muted", line)).join("\n"),
|
|
1,
|
|
0,
|
|
);
|
|
const visualLines = tempText.render(this.ui.terminal.columns);
|
|
const truncatedVisualLines = visualLines.slice(-PREVIEW_LINES);
|
|
this.contentContainer.addChild({ render: () => truncatedVisualLines, invalidate: () => {} });
|
|
}
|
|
}
|
|
|
|
// Loader or status
|
|
if (this.status === "running") {
|
|
this.contentContainer.addChild(this.loader);
|
|
} else {
|
|
const statusParts: string[] = [];
|
|
|
|
// Show how many lines are hidden (collapsed preview)
|
|
if (hiddenLineCount > 0) {
|
|
statusParts.push(theme.fg("dim", `... ${hiddenLineCount} more lines (ctrl+o to expand)`));
|
|
}
|
|
|
|
if (this.status === "cancelled") {
|
|
statusParts.push(theme.fg("warning", "(cancelled)"));
|
|
} else if (this.status === "error") {
|
|
statusParts.push(theme.fg("error", `(exit ${this.exitCode})`));
|
|
}
|
|
|
|
// Add truncation warning (context truncation, not preview truncation)
|
|
const wasTruncated = this.truncationResult?.truncated || contextTruncation.truncated;
|
|
if (wasTruncated && this.fullOutputPath) {
|
|
statusParts.push(theme.fg("warning", `Output truncated. Full output: ${this.fullOutputPath}`));
|
|
}
|
|
|
|
if (statusParts.length > 0) {
|
|
this.contentContainer.addChild(new Text("\n" + statusParts.join("\n"), 1, 0));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the raw output for creating BashExecutionMessage.
|
|
*/
|
|
getOutput(): string {
|
|
return this.outputLines.join("\n");
|
|
}
|
|
|
|
/**
|
|
* Get the command that was executed.
|
|
*/
|
|
getCommand(): string {
|
|
return this.command;
|
|
}
|
|
}
|