mirror of
https://github.com/getcompanion-ai/co-mono.git
synced 2026-04-16 06:02:42 +00:00
- Removed Attachment from agent package (now in web-ui/coding-agent) - Agent.prompt now takes (text, images?: ImageContent[]) - Removed transports from web-ui (duplicate of agent package) - Updated coding-agent to use local message types - Updated mom package for new agent API Remaining: Fix AgentInterface.ts to compose UserMessageWithAttachments
159 lines
5.9 KiB
TypeScript
159 lines
5.9 KiB
TypeScript
import type { AgentTool } from "@mariozechner/pi-agent-core";
|
|
import type { ImageContent, TextContent } from "@mariozechner/pi-ai";
|
|
import { Type } from "@sinclair/typebox";
|
|
import { extname } from "path";
|
|
import type { Executor } from "../sandbox.js";
|
|
import { DEFAULT_MAX_BYTES, DEFAULT_MAX_LINES, formatSize, type TruncationResult, truncateHead } from "./truncate.js";
|
|
|
|
/**
|
|
* Map of file extensions to MIME types for common image formats
|
|
*/
|
|
const IMAGE_MIME_TYPES: Record<string, string> = {
|
|
".jpg": "image/jpeg",
|
|
".jpeg": "image/jpeg",
|
|
".png": "image/png",
|
|
".gif": "image/gif",
|
|
".webp": "image/webp",
|
|
};
|
|
|
|
/**
|
|
* Check if a file is an image based on its extension
|
|
*/
|
|
function isImageFile(filePath: string): string | null {
|
|
const ext = extname(filePath).toLowerCase();
|
|
return IMAGE_MIME_TYPES[ext] || null;
|
|
}
|
|
|
|
const readSchema = Type.Object({
|
|
label: Type.String({ description: "Brief description of what you're reading and why (shown to user)" }),
|
|
path: Type.String({ description: "Path to the file to read (relative or absolute)" }),
|
|
offset: Type.Optional(Type.Number({ description: "Line number to start reading from (1-indexed)" })),
|
|
limit: Type.Optional(Type.Number({ description: "Maximum number of lines to read" })),
|
|
});
|
|
|
|
interface ReadToolDetails {
|
|
truncation?: TruncationResult;
|
|
}
|
|
|
|
export function createReadTool(executor: Executor): AgentTool<typeof readSchema> {
|
|
return {
|
|
name: "read",
|
|
label: "read",
|
|
description: `Read the contents of a file. Supports text files and images (jpg, png, gif, webp). Images are sent as attachments. For text files, output is truncated to ${DEFAULT_MAX_LINES} lines or ${DEFAULT_MAX_BYTES / 1024}KB (whichever is hit first). Use offset/limit for large files.`,
|
|
parameters: readSchema,
|
|
execute: async (
|
|
_toolCallId: string,
|
|
{ path, offset, limit }: { label: string; path: string; offset?: number; limit?: number },
|
|
signal?: AbortSignal,
|
|
): Promise<{ content: (TextContent | ImageContent)[]; details: ReadToolDetails | undefined }> => {
|
|
const mimeType = isImageFile(path);
|
|
|
|
if (mimeType) {
|
|
// Read as image (binary) - use base64
|
|
const result = await executor.exec(`base64 < ${shellEscape(path)}`, { signal });
|
|
if (result.code !== 0) {
|
|
throw new Error(result.stderr || `Failed to read file: ${path}`);
|
|
}
|
|
const base64 = result.stdout.replace(/\s/g, ""); // Remove whitespace from base64
|
|
|
|
return {
|
|
content: [
|
|
{ type: "text", text: `Read image file [${mimeType}]` },
|
|
{ type: "image", data: base64, mimeType },
|
|
],
|
|
details: undefined,
|
|
};
|
|
}
|
|
|
|
// Get total line count first
|
|
const countResult = await executor.exec(`wc -l < ${shellEscape(path)}`, { signal });
|
|
if (countResult.code !== 0) {
|
|
throw new Error(countResult.stderr || `Failed to read file: ${path}`);
|
|
}
|
|
const totalFileLines = Number.parseInt(countResult.stdout.trim(), 10) + 1; // wc -l counts newlines, not lines
|
|
|
|
// Apply offset if specified (1-indexed)
|
|
const startLine = offset ? Math.max(1, offset) : 1;
|
|
const startLineDisplay = startLine;
|
|
|
|
// Check if offset is out of bounds
|
|
if (startLine > totalFileLines) {
|
|
throw new Error(`Offset ${offset} is beyond end of file (${totalFileLines} lines total)`);
|
|
}
|
|
|
|
// Read content with offset
|
|
let cmd: string;
|
|
if (startLine === 1) {
|
|
cmd = `cat ${shellEscape(path)}`;
|
|
} else {
|
|
cmd = `tail -n +${startLine} ${shellEscape(path)}`;
|
|
}
|
|
|
|
const result = await executor.exec(cmd, { signal });
|
|
if (result.code !== 0) {
|
|
throw new Error(result.stderr || `Failed to read file: ${path}`);
|
|
}
|
|
|
|
let selectedContent = result.stdout;
|
|
let userLimitedLines: number | undefined;
|
|
|
|
// Apply user limit if specified
|
|
if (limit !== undefined) {
|
|
const lines = selectedContent.split("\n");
|
|
const endLine = Math.min(limit, lines.length);
|
|
selectedContent = lines.slice(0, endLine).join("\n");
|
|
userLimitedLines = endLine;
|
|
}
|
|
|
|
// Apply truncation (respects both line and byte limits)
|
|
const truncation = truncateHead(selectedContent);
|
|
|
|
let outputText: string;
|
|
let details: ReadToolDetails | undefined;
|
|
|
|
if (truncation.firstLineExceedsLimit) {
|
|
// First line at offset exceeds 50KB - tell model to use bash
|
|
const firstLineSize = formatSize(Buffer.byteLength(selectedContent.split("\n")[0], "utf-8"));
|
|
outputText = `[Line ${startLineDisplay} is ${firstLineSize}, exceeds ${formatSize(DEFAULT_MAX_BYTES)} limit. Use bash: sed -n '${startLineDisplay}p' ${path} | head -c ${DEFAULT_MAX_BYTES}]`;
|
|
details = { truncation };
|
|
} else if (truncation.truncated) {
|
|
// Truncation occurred - build actionable notice
|
|
const endLineDisplay = startLineDisplay + truncation.outputLines - 1;
|
|
const nextOffset = endLineDisplay + 1;
|
|
|
|
outputText = truncation.content;
|
|
|
|
if (truncation.truncatedBy === "lines") {
|
|
outputText += `\n\n[Showing lines ${startLineDisplay}-${endLineDisplay} of ${totalFileLines}. Use offset=${nextOffset} to continue]`;
|
|
} else {
|
|
outputText += `\n\n[Showing lines ${startLineDisplay}-${endLineDisplay} of ${totalFileLines} (${formatSize(DEFAULT_MAX_BYTES)} limit). Use offset=${nextOffset} to continue]`;
|
|
}
|
|
details = { truncation };
|
|
} else if (userLimitedLines !== undefined) {
|
|
// User specified limit, check if there's more content
|
|
const linesFromStart = startLine - 1 + userLimitedLines;
|
|
if (linesFromStart < totalFileLines) {
|
|
const remaining = totalFileLines - linesFromStart;
|
|
const nextOffset = startLine + userLimitedLines;
|
|
|
|
outputText = truncation.content;
|
|
outputText += `\n\n[${remaining} more lines in file. Use offset=${nextOffset} to continue]`;
|
|
} else {
|
|
outputText = truncation.content;
|
|
}
|
|
} else {
|
|
// No truncation, no user limit exceeded
|
|
outputText = truncation.content;
|
|
}
|
|
|
|
return {
|
|
content: [{ type: "text", text: outputText }],
|
|
details,
|
|
};
|
|
},
|
|
};
|
|
}
|
|
|
|
function shellEscape(s: string): string {
|
|
return `'${s.replace(/'/g, "'\\''")}'`;
|
|
}
|