Multi-agent audit + aggressive cleanup across server/web/coder/booterm, delivered behind a DEFER discipline so none of the in-flight files were touched. Removes dead code/deps/columns, dedups server + coder helpers, and splits the oversized modules (tools.ts, opencode-server.ts, sentinel-summaries, turn.ts, TerminalPane.tsx) behind stable contracts. Adds 78 parity/unit tests (server 587, coder 323); fixes two latent bugs (ChatPane queue keys, FileViewerOverlay blank-line parity). Intended tag: v2.7.12-audit-cleanup. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
179 lines
6.5 KiB
TypeScript
179 lines
6.5 KiB
TypeScript
import { useState } from 'react';
|
|
import { Check, ChevronRight, Loader2, X } from 'lucide-react';
|
|
import type { ToolCall, ToolResult } from '@/api/types';
|
|
import { linkifyPaths } from '@/lib/linkify-paths';
|
|
|
|
// v1.8.2: cap on the inline arg-summary length. Expanded view shows full
|
|
// args + full result, so this is purely a single-line render budget.
|
|
const ARG_SUMMARY_MAX = 60;
|
|
|
|
export interface ToolRun {
|
|
call: ToolCall;
|
|
// null while the call is in flight or the matching tool result hasn't
|
|
// arrived yet on the WS stream.
|
|
result: ToolResult | null;
|
|
}
|
|
|
|
function truncate(s: string, n: number): string {
|
|
return s.length > n ? s.slice(0, n - 1) + '…' : s;
|
|
}
|
|
|
|
// Per-tool argument summary mapping from the v1.8.2 spec. Goal is a single
|
|
// scannable line that surfaces the *what* (path / pattern) without
|
|
// overwhelming the chat with full JSON.
|
|
function formatToolArgs(name: string, args: Record<string, unknown>): string {
|
|
if (name === 'view_file') {
|
|
const path = String(args.path ?? '');
|
|
const start = args.start_line;
|
|
const end = args.end_line;
|
|
if (typeof start === 'number' && typeof end === 'number') {
|
|
return truncate(`${path}:${start}-${end}`, ARG_SUMMARY_MAX);
|
|
}
|
|
if (typeof start === 'number') {
|
|
return truncate(`${path}:${start}`, ARG_SUMMARY_MAX);
|
|
}
|
|
return truncate(path, ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'list_dir') {
|
|
return truncate(String(args.path ?? '.'), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'grep') {
|
|
const pattern = String(args.pattern ?? '');
|
|
const path = args.path ? ` ${String(args.path)}` : '';
|
|
return truncate(`"${pattern}"${path}`, ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'find_files') {
|
|
return truncate(String(args.pattern ?? ''), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'git_status') {
|
|
return '';
|
|
}
|
|
if (name === 'skill_use') {
|
|
// Schema (apps/server/src/services/tools.ts SkillUseInput) uses `name`;
|
|
// fall back to `skill_name` defensively in case a model emits that key.
|
|
return truncate(
|
|
String(args.name ?? (args as { skill_name?: unknown }).skill_name ?? '<unknown>'),
|
|
ARG_SUMMARY_MAX,
|
|
);
|
|
}
|
|
// v1.12 Track B.2: codecontext tool pills. Format is "most-identifying-arg",
|
|
// matching view_file/grep precedent — surface the path/symbol/query that
|
|
// makes the call meaningful at a glance.
|
|
if (name === 'get_codebase_overview') {
|
|
return '';
|
|
}
|
|
if (name === 'get_file_analysis') {
|
|
return truncate(String(args.file_path ?? ''), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'get_symbol_info') {
|
|
return truncate(String(args.symbol_name ?? ''), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'search_symbols') {
|
|
return truncate(`"${String(args.query ?? '')}"`, ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'get_dependencies') {
|
|
return truncate(String(args.file_path ?? '(project-wide)'), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'watch_changes') {
|
|
return args.enable ? 'enable' : 'disable';
|
|
}
|
|
if (name === 'get_semantic_neighborhoods') {
|
|
return truncate(String(args.file_path ?? '(project-wide)'), ARG_SUMMARY_MAX);
|
|
}
|
|
if (name === 'get_framework_analysis') {
|
|
return truncate(String(args.framework ?? '(auto-detect)'), ARG_SUMMARY_MAX);
|
|
}
|
|
// Unknown tool — surface first arg value or the literal {} so the user can
|
|
// see something happened. Forward-compatible with future tools.
|
|
const keys = Object.keys(args);
|
|
if (keys.length === 0) return '{}';
|
|
const first = keys[0]!;
|
|
return truncate(`${first}: ${String(args[first])}`, ARG_SUMMARY_MAX);
|
|
}
|
|
|
|
export function runStatus(run: ToolRun): 'pending' | 'success' | 'error' {
|
|
if (run.result === null) return 'pending';
|
|
if (run.result.error) return 'error';
|
|
return 'success';
|
|
}
|
|
|
|
// Path-shaped substrings in tool output get a click handler via the shared
|
|
// linkifyPaths util (lib/linkify-paths.tsx).
|
|
|
|
interface Props {
|
|
run: ToolRun;
|
|
// When rendered inside a ToolCallGroup the line is already nested under a
|
|
// shared header, so the leading arrow is dropped to avoid double indent.
|
|
insideGroup?: boolean;
|
|
}
|
|
|
|
export function ToolCallLine({ run, insideGroup }: Props) {
|
|
const [open, setOpen] = useState(false);
|
|
const status = runStatus(run);
|
|
const args = run.call.args ?? {};
|
|
const summary = formatToolArgs(run.call.name, args);
|
|
|
|
return (
|
|
<div className="text-xs">
|
|
<button
|
|
type="button"
|
|
onClick={() => setOpen((v) => !v)}
|
|
className="flex items-center gap-1.5 w-full text-left hover:bg-muted/40 rounded px-1 py-0.5 -mx-1"
|
|
>
|
|
{/* BooCode 2.0: glowing activity indicator (was ↳ / >_) */}
|
|
{!insideGroup && (
|
|
<span
|
|
className="size-1.5 rounded-full bg-primary shrink-0"
|
|
style={{ boxShadow: '0 0 6px var(--primary)' }}
|
|
aria-hidden
|
|
/>
|
|
)}
|
|
<ChevronRight
|
|
className={`size-3 text-muted-foreground/60 shrink-0 transition-transform ${open ? 'rotate-90' : ''}`}
|
|
/>
|
|
<span className="font-mono text-foreground/90 shrink-0">{run.call.name}</span>
|
|
{summary && (
|
|
<span className="font-mono text-muted-foreground truncate min-w-0 flex-1">
|
|
{summary}
|
|
</span>
|
|
)}
|
|
{!summary && <span className="flex-1" />}
|
|
<span className="shrink-0 ml-1">
|
|
{status === 'pending' && (
|
|
<Loader2 className="size-3 text-muted-foreground animate-spin" aria-label="running" />
|
|
)}
|
|
{status === 'success' && (
|
|
<Check className="size-3 text-emerald-500" aria-label="success" />
|
|
)}
|
|
{status === 'error' && (
|
|
<X className="size-3 text-destructive" aria-label="error" />
|
|
)}
|
|
</span>
|
|
</button>
|
|
{open && (
|
|
<div className="ml-5 mt-1 mb-1 space-y-1">
|
|
<pre className="text-[10px] text-muted-foreground font-mono whitespace-pre-wrap break-all bg-muted/30 rounded px-2 py-1">
|
|
{JSON.stringify(args, null, 2)}
|
|
</pre>
|
|
{run.result && (
|
|
<pre className="text-[11px] font-mono whitespace-pre-wrap bg-muted/30 rounded px-2 py-1 max-h-72 overflow-y-auto">
|
|
{run.result.error ? (
|
|
<span className="text-destructive">{run.result.error}</span>
|
|
) : (
|
|
linkifyPaths(
|
|
typeof run.result.output === 'string'
|
|
? run.result.output
|
|
: JSON.stringify(run.result.output, null, 2)
|
|
)
|
|
)}
|
|
{run.result.truncated && (
|
|
<div className="text-muted-foreground/60 mt-1">— output truncated —</div>
|
|
)}
|
|
</pre>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|