mirror of
https://github.com/earendil-works/pi.git
synced 2026-06-18 15:54:04 +08:00
65 lines
2.0 KiB
TypeScript
65 lines
2.0 KiB
TypeScript
import * as os from "node:os";
|
|
import type { ImageContent, TextContent } from "@earendil-works/pi-ai";
|
|
import { getCapabilities, getImageDimensions, imageFallback } from "@earendil-works/pi-tui";
|
|
import { stripAnsi } from "../../utils/ansi.js";
|
|
import { sanitizeBinaryOutput } from "../../utils/shell.js";
|
|
|
|
export function shortenPath(path: unknown): string {
|
|
if (typeof path !== "string") return "";
|
|
const home = os.homedir();
|
|
if (path.startsWith(home)) {
|
|
return `~${path.slice(home.length)}`;
|
|
}
|
|
return path;
|
|
}
|
|
|
|
export function str(value: unknown): string | null {
|
|
if (typeof value === "string") return value;
|
|
if (value == null) return "";
|
|
return null;
|
|
}
|
|
|
|
export function replaceTabs(text: string): string {
|
|
return text.replace(/\t/g, " ");
|
|
}
|
|
|
|
export function normalizeDisplayText(text: string): string {
|
|
return text.replace(/\r/g, "");
|
|
}
|
|
|
|
export function getTextOutput(
|
|
result: { content: Array<{ type: string; text?: string; data?: string; mimeType?: string }> } | undefined,
|
|
showImages: boolean,
|
|
): string {
|
|
if (!result) return "";
|
|
|
|
const textBlocks = result.content.filter((c) => c.type === "text");
|
|
const imageBlocks = result.content.filter((c) => c.type === "image");
|
|
|
|
let output = textBlocks.map((c) => sanitizeBinaryOutput(stripAnsi(c.text || "")).replace(/\r/g, "")).join("\n");
|
|
|
|
const caps = getCapabilities();
|
|
if (imageBlocks.length > 0 && (!caps.images || !showImages)) {
|
|
const imageIndicators = imageBlocks
|
|
.map((img) => {
|
|
const mimeType = img.mimeType ?? "image/unknown";
|
|
const dims =
|
|
img.data && img.mimeType ? (getImageDimensions(img.data, img.mimeType) ?? undefined) : undefined;
|
|
return imageFallback(mimeType, dims);
|
|
})
|
|
.join("\n");
|
|
output = output ? `${output}\n${imageIndicators}` : imageIndicators;
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
export type ToolRenderResultLike<TDetails> = {
|
|
content: (TextContent | ImageContent)[];
|
|
details: TDetails;
|
|
};
|
|
|
|
export function invalidArgText(theme: { fg: (name: any, text: string) => string }): string {
|
|
return theme.fg("error", "[invalid arg]");
|
|
}
|