* Print mode (single-shot): Send prompts, output result, exit.
*
* Used for:
* - `omp -p "prompt"` - text output
* - `omp --mode json "prompt"` - JSON event stream
*/
import type { AssistantMessage, ImageContent } from "@oh-my-pi/pi-ai";
import { logger, sanitizeText } from "@oh-my-pi/pi-utils";
import type { AgentSession } from "../session/agent-session";
import { isSilentAbort } from "../session/messages";
import { initializeExtensions } from "./runtime-init";
* Options for print mode.
*/
export interface PrintModeOptions {
mode: "text" | "json";
messages?: string[];
initialMessage?: string;
initialImages?: ImageContent[];
}
* Run in print (single-shot) mode.
* Sends prompts to the agent and outputs the result.
*/
export async function runPrintMode(session: AgentSession, options: PrintModeOptions): Promise<void> {
const { mode, messages = [], initialMessage, initialImages } = options;
if (mode === "json") {
const header = session.sessionManager.getHeader();
if (header) {
process.stdout.write(`${JSON.stringify(header)}\n`);
}
}
await initializeExtensions(session, {
reportSendError: (action, err) => {
process.stderr.write(
`Extension ${action === "extension_send" ? "sendMessage" : "sendUserMessage"} failed: ${err.message}\n`,
);
},
reportRuntimeError: err => {
process.stderr.write(`Extension error (${err.extensionPath}): ${err.error}\n`);
},
});
session.subscribe(event => {
if (mode === "json") {
process.stdout.write(`${JSON.stringify(event)}\n`);
}
});
if (initialMessage !== undefined) {
await logger.time("print:prompt:initial", () => session.prompt(initialMessage, { images: initialImages }));
}
for (const message of messages) {
await logger.time("print:prompt:next", () => session.prompt(message));
}
if (mode === "text") {
const state = session.state;
const lastMessage = state.messages[state.messages.length - 1];
if (lastMessage?.role === "assistant") {
const assistantMsg = lastMessage as AssistantMessage;
if (
(assistantMsg.stopReason === "error" || assistantMsg.stopReason === "aborted") &&
!isSilentAbort(assistantMsg.errorMessage)
) {
const errorLine = sanitizeText(assistantMsg.errorMessage || `Request ${assistantMsg.stopReason}`);
const flushed = process.stderr.write(`${errorLine}\n`);
if (flushed) {
process.exit(1);
} else {
process.stderr.once("drain", () => process.exit(1));
}
}
if (
assistantMsg.errorMessage &&
assistantMsg.stopReason !== "error" &&
assistantMsg.stopReason !== "aborted"
) {
process.stderr.write(`${sanitizeText(assistantMsg.errorMessage)}\n`);
}
for (const content of assistantMsg.content) {
if (content.type === "text") {
process.stdout.write(`${sanitizeText(content.text)}\n`);
}
}
}
}
await new Promise<void>((resolve, reject) => {
process.stdout.write("", err => {
if (err) reject(err);
else resolve();
});
});
await session.dispose();
}