/**
* Process @file CLI arguments into text content and image attachments
*/
import * as fs from "node:fs";
import * as path from "node:path";
import type { ImageContent } from "@oh-my-pi/pi-ai";
import { getProjectDir, isEnoent, readImageMetadata } from "@oh-my-pi/pi-utils";
import chalk from "chalk";
import { resolveReadPath } from "../tools/path-utils";
import { formatBytes } from "../tools/render-utils";
import { formatDimensionNote, resizeImage } from "../utils/image-resize";
// Keep CLI startup responsive and avoid OOM when users pass huge files.
// If a file exceeds these limits, we include it as a path-only block.
const MAX_CLI_TEXT_BYTES = 5 * 1024 * 1024; // 5MB
const MAX_CLI_IMAGE_BYTES = 25 * 1024 * 1024; // 25MB
export interface ProcessedFiles {
text: string;
images: ImageContent[];
}
export interface ProcessFileOptions {
/** Whether to auto-resize images to 2000x2000 max. Default: true */
autoResizeImages?: boolean;
}
/** Process @file arguments into text content and image attachments */
export async function processFileArguments(fileArgs: string[], options?: ProcessFileOptions): Promise {
const autoResizeImages = options?.autoResizeImages ?? true;
let text = "";
const images: ImageContent[] = [];
for (const fileArg of fileArgs) {
// Expand and resolve path (handles ~ expansion and macOS screenshot Unicode spaces)
const absolutePath = path.resolve(resolveReadPath(fileArg, getProjectDir()));
const stat = fs.statSync(absolutePath, { throwIfNoEntry: false });
if (!stat) {
console.error(chalk.red(`Error: File not found: ${absolutePath}`));
process.exit(1);
}
const imageMetadata = await readImageMetadata(absolutePath);
const mimeType = imageMetadata?.mimeType;
const maxBytes = mimeType ? MAX_CLI_IMAGE_BYTES : MAX_CLI_TEXT_BYTES;
if (stat.size > maxBytes) {
console.error(
chalk.yellow(`Warning: Skipping file contents (too large: ${formatBytes(stat.size)}): ${absolutePath}`),
);
text += `(skipped: too large, ${formatBytes(stat.size)})\n`;
continue;
}
// Read file, handling not-found gracefully
let buffer: Uint8Array;
try {
buffer = await Bun.file(absolutePath).bytes();
} catch (err) {
if (isEnoent(err)) {
console.error(chalk.red(`Error: File not found: ${absolutePath}`));
process.exit(1);
}
throw err;
}
if (buffer.length === 0) {
continue;
}
if (mimeType) {
// Handle image file
const base64Content = buffer.toBase64();
let attachment: ImageContent;
let dimensionNote: string | undefined;
if (autoResizeImages) {
try {
const resized = await resizeImage({ type: "image", data: base64Content, mimeType });
dimensionNote = formatDimensionNote(resized);
attachment = {
type: "image",
mimeType: resized.mimeType,
data: resized.data,
};
} catch {
// Fall back to original image on resize failure
attachment = {
type: "image",
mimeType,
data: base64Content,
};
}
} else {
attachment = {
type: "image",
mimeType,
data: base64Content,
};
}
images.push(attachment);
// Add text reference to image with optional dimension note
if (dimensionNote) {
text += `${dimensionNote}\n`;
} else {
text += `\n`;
}
} else {
// Handle text file
try {
const content = new TextDecoder().decode(buffer);
text += `\n${content}\n\n`;
} catch (error: unknown) {
const message = error instanceof Error ? error.message : String(error);
console.error(chalk.red(`Error: Could not read file ${absolutePath}: ${message}`));
process.exit(1);
}
}
}
return { text, images };
}