* Main entry point for the coding agent CLI.
*
* This file handles CLI argument parsing and translates them into
* createAgentSession() options. The SDK does the heavy lifting.
*/
import * as fs from "node:fs/promises";
import * as os from "node:os";
import * as path from "node:path";
import { createInterface } from "node:readline/promises";
import { EventLoopKeepalive } from "@oh-my-pi/pi-agent-core";
import type { ImageContent } from "@oh-my-pi/pi-ai";
import {
$env,
getProjectDir,
logger,
normalizePathForComparison,
postmortem,
setProjectDir,
VERSION,
} from "@oh-my-pi/pi-utils";
import chalk from "chalk";
import type { Args } from "./cli/args";
import { applyExtensionFlags, type ExtensionFlagSink } from "./cli/extension-flags";
import { processFileArguments } from "./cli/file-processor";
import { buildInitialMessage } from "./cli/initial-message";
import { runListModelsCommand } from "./cli/list-models";
import { selectSession } from "./cli/session-picker";
import { findConfigFile } from "./config";
import { ModelRegistry, ModelsConfigFile } from "./config/model-registry";
import { resolveCliModel, resolveModelRoleValue, resolveModelScope, type ScopedModel } from "./config/model-resolver";
import { getDefault, type SettingPath, Settings, settings } from "./config/settings";
import { initializeWithSettings } from "./discovery";
import {
clearPluginRootsAndCaches,
injectPluginDirRoots,
preloadPluginRoots,
resolveActiveProjectRegistryPath,
} from "./discovery/helpers";
import { injectOmpExtensionCliRoots } from "./discovery/omp-extension-roots";
import { exportFromFile } from "./export/html";
import { ExtensionRunner } from "./extensibility/extensions/runner";
import type { ExtensionUIContext } from "./extensibility/extensions/types";
import {
getInstalledPluginsRegistryPath,
getMarketplacesCacheDir,
getMarketplacesRegistryPath,
getPluginsCacheDir,
MarketplaceManager,
} from "./extensibility/plugins/marketplace";
import type { MCPManager } from "./mcp";
import { InteractiveMode, runAcpMode, runPrintMode, runRpcMode } from "./modes";
import { ALL_SCENES, runSetupWizard, selectSetupScenes } from "./modes/setup-wizard";
import { initTheme, stopThemeWatcher } from "./modes/theme/theme";
import type { SubmittedUserInput } from "./modes/types";
import {
type CreateAgentSessionOptions,
type CreateAgentSessionResult,
createAgentSession,
discoverAuthStorage,
loadSessionExtensions,
} from "./sdk";
import type { AgentSession } from "./session/agent-session";
import type { AuthStorage } from "./session/auth-storage";
import { resolveResumableSession, type SessionInfo, SessionManager } from "./session/session-manager";
import { resolvePromptInput } from "./system-prompt";
import { AUTO_THINKING } from "./thinking";
import type { LspStartupServerInfo } from "./tools";
import { getChangelogPath, getNewEntries, parseChangelog } from "./utils/changelog";
import { EventBus } from "./utils/event-bus";
async function checkForNewVersion(currentVersion: string): Promise<string | undefined> {
if (!settings.get("startup.checkUpdate")) {
return;
}
try {
const response = await fetch("https://registry.npmjs.org/@oh-my-pi/pi-coding-agent/latest");
if (!response.ok) return undefined;
const data = (await response.json()) as { version?: string };
const latestVersion = data.version;
if (latestVersion && Bun.semver.order(latestVersion, currentVersion) > 0) {
return latestVersion;
}
return undefined;
} catch {
return undefined;
}
}
const RPC_DEFAULTED_SETTING_PATHS: SettingPath[] = [
"todo.enabled",
"todo.reminders",
"todo.reminders.max",
"todo.eager",
"async.enabled",
"async.maxJobs",
"bash.autoBackground.enabled",
"bash.autoBackground.thresholdMs",
"task.isolation.mode",
"task.isolation.merge",
"task.isolation.commits",
"task.eager",
"task.simple",
"task.maxConcurrency",
"task.maxRecursionDepth",
"task.disabledAgents",
"task.agentModelOverrides",
"memory.backend",
"memories.enabled",
];
function applyRpcDefaultSettingOverrides(targetSettings: Settings = settings): void {
for (const settingPath of RPC_DEFAULTED_SETTING_PATHS) {
targetSettings.override(settingPath, getDefault(settingPath));
}
}
async function readPipedInput(): Promise<string | undefined> {
if (process.stdin.isTTY !== false) return undefined;
try {
const text = await Bun.stdin.text();
if (text.trim().length === 0) return undefined;
return text;
} catch {
return undefined;
}
}
export interface InteractiveModeNotify {
kind: "warn" | "error" | "info";
message: string;
}
export async function submitInteractiveInput(
mode: Pick<
InteractiveMode,
"markPendingSubmissionStarted" | "finishPendingSubmission" | "showError" | "checkShutdownRequested"
>,
session: Pick<AgentSession, "prompt" | "promptCustomMessage">,
input: SubmittedUserInput,
): Promise<void> {
if (input.cancelled) {
return;
}
try {
using _keepalive = new EventLoopKeepalive();
if (!input.started && !mode.markPendingSubmissionStarted(input)) {
return;
}
if (input.customType) {
await session.promptCustomMessage({
customType: input.customType,
content: input.text,
display: input.display ?? false,
attribution: "agent",
});
} else {
await session.prompt(input.text, { images: input.images });
}
} catch (error: unknown) {
const errorMessage = error instanceof Error ? error.message : "Unknown error occurred";
mode.showError(errorMessage);
} finally {
mode.finishPendingSubmission(input);
await mode.checkShutdownRequested();
}
}
type AcpSessionFactory = (cwd: string) => Promise<AgentSession>;
export interface AcpSessionFactoryOptions {
baseOptions: CreateAgentSessionOptions;
settings: Settings;
sessionDir?: string;
authStorage: AuthStorage;
modelRegistry: ModelRegistry;
parsedArgs: Pick<Args, "apiKey">;
rawArgs: string[];
createSession: (options: CreateAgentSessionOptions) => Promise<CreateAgentSessionResult>;
}
* Build the per-`session/new` factory used by ACP mode.
*
* MCP servers in ACP sessions are owned exclusively by the ACP client, which
* supplies them through `session/new.mcpServers` and re-applies them via
* {@link AcpAgent#configureMcpServers}. We therefore force `enableMCP: false`
* on every session created here so {@link createAgentSession} skips the on-disk
* `.mcp.json` discovery path — otherwise host MCP tools land in the session's
* tool registry and shadow the client-supplied servers (issue #1234).
*/
export function createAcpSessionFactory(args: AcpSessionFactoryOptions): AcpSessionFactory {
return async cwd => {
const nextSettings = await args.settings.cloneForCwd(cwd);
const nextSessionManager = SessionManager.create(cwd, args.sessionDir);
const agentId = `acp:${nextSessionManager.getSessionId()}`;
const { session: nextSession } = await args.createSession({
...args.baseOptions,
cwd,
sessionManager: nextSessionManager,
settings: nextSettings,
authStorage: args.authStorage,
modelRegistry: args.modelRegistry,
agentId,
hasUI: false,
enableMCP: false,
});
if (args.parsedArgs.apiKey && !args.baseOptions.model && nextSession.model) {
args.authStorage.setRuntimeApiKey(nextSession.model.provider, args.parsedArgs.apiKey);
}
applyExtensionFlags(nextSession.extensionRunner, args.rawArgs);
return nextSession;
};
}
async function runInteractiveMode(
session: AgentSession,
version: string,
changelogMarkdown: string | undefined,
notifs: (InteractiveModeNotify | null)[],
versionCheckPromise: Promise<string | undefined>,
initialMessages: string[],
setExtensionUIContext: (uiContext: ExtensionUIContext, hasUI: boolean) => void,
lspServers: LspStartupServerInfo[] | undefined,
mcpManager: MCPManager | undefined,
resuming: boolean,
forceSetupWizard: boolean,
eventBus?: EventBus,
initialMessage?: string,
initialImages?: ImageContent[],
): Promise<void> {
const mode = new InteractiveMode(
session,
version,
changelogMarkdown,
setExtensionUIContext,
lspServers,
mcpManager,
eventBus,
);
const setupScenes = await selectSetupScenes(settings.get("setupVersion"), ALL_SCENES, mode, {
resuming,
isTTY: process.stdin.isTTY && process.stdout.isTTY,
setupWizardEnabled: settings.get("startup.setupWizard"),
force: forceSetupWizard,
});
await mode.init({ suppressWelcomeIntro: setupScenes.length > 0 });
if (setupScenes.length > 0) {
await runSetupWizard(mode, setupScenes);
}
versionCheckPromise
.then(newVersion => {
if (!settings.get("startup.checkUpdate")) {
return;
}
if (newVersion) {
mode.showNewVersionNotification(newVersion);
}
})
.catch(() => {});
mode.renderInitialMessages(undefined, { preserveExistingChat: true });
for (const notify of notifs) {
if (!notify) {
continue;
}
if (notify.kind === "warn") {
mode.showWarning(notify.message);
} else if (notify.kind === "error") {
mode.showError(notify.message);
} else if (notify.kind === "info") {
mode.showStatus(notify.message);
}
}
if (initialMessage !== undefined) {
try {
using _keepalive = new EventLoopKeepalive();
await session.prompt(initialMessage, { images: initialImages });
} catch (error: unknown) {
const errorMessage = error instanceof Error ? error.message : "Unknown error occurred";
mode.showError(errorMessage);
}
}
for (const message of initialMessages) {
try {
using _keepalive = new EventLoopKeepalive();
await session.prompt(message);
} catch (error: unknown) {
const errorMessage = error instanceof Error ? error.message : "Unknown error occurred";
mode.showError(errorMessage);
}
}
while (true) {
const input = await mode.getUserInput();
await submitInteractiveInput(mode, session, input);
}
}
async function promptForkSession(session: SessionInfo): Promise<boolean> {
if (!process.stdin.isTTY) {
return false;
}
const message = `Session found in different project: ${session.cwd}. Fork into current directory? [y/N] `;
const rl = createInterface({ input: process.stdin, output: process.stdout });
try {
const answer = (await rl.question(message)).trim().toLowerCase();
return answer === "y" || answer === "yes";
} finally {
rl.close();
}
}
async function getChangelogForDisplay(parsed: Args): Promise<string | undefined> {
if (parsed.continue || parsed.resume) {
return undefined;
}
const lastVersion = settings.get("lastChangelogVersion");
if (lastVersion === VERSION) {
return undefined;
}
const changelogPath = getChangelogPath();
const entries = await parseChangelog(changelogPath);
if (!lastVersion) {
if (entries.length > 0) {
settings.set("lastChangelogVersion", VERSION);
await flushChangelogVersion();
return entries.map(e => e.content).join("\n\n");
}
} else {
const newEntries = getNewEntries(entries, lastVersion);
if (newEntries.length > 0) {
settings.set("lastChangelogVersion", VERSION);
await flushChangelogVersion();
return newEntries.map(e => e.content).join("\n\n");
}
}
return undefined;
}
async function flushChangelogVersion(): Promise<void> {
try {
await settings.flush();
} catch (error: unknown) {
logger.warn("Failed to persist lastChangelogVersion", { error });
}
}
async function createSessionManager(
parsed: Args,
cwd: string,
activeSettings: Settings = settings,
): Promise<SessionManager | undefined> {
if (parsed.fork) {
if (parsed.noSession) {
throw new Error("--fork requires session persistence");
}
const forkSource = parsed.fork;
if (forkSource.includes("/") || forkSource.includes("\\") || forkSource.endsWith(".jsonl")) {
return await SessionManager.forkFrom(forkSource, cwd, parsed.sessionDir);
}
const match = await resolveResumableSession(forkSource, cwd, parsed.sessionDir);
if (!match) {
throw new Error(`Session "${forkSource}" not found.`);
}
return await SessionManager.forkFrom(match.session.path, cwd, parsed.sessionDir);
}
if (parsed.noSession) {
return SessionManager.inMemory();
}
if (typeof parsed.resume === "string") {
const sessionArg = parsed.resume;
if (sessionArg.includes("/") || sessionArg.includes("\\") || sessionArg.endsWith(".jsonl")) {
return await SessionManager.open(sessionArg, parsed.sessionDir);
}
const match = await resolveResumableSession(sessionArg, cwd, parsed.sessionDir);
if (!match) {
throw new Error(`Session "${sessionArg}" not found.`);
}
if (match.scope === "global") {
const normalizedCwd = normalizePathForComparison(cwd);
const normalizedMatchCwd = normalizePathForComparison(match.session.cwd || cwd);
if (normalizedCwd !== normalizedMatchCwd) {
const shouldFork = await promptForkSession(match.session);
if (!shouldFork) {
throw new Error(`Session "${sessionArg}" is in another project (${match.session.cwd}).`);
}
return await SessionManager.forkFrom(match.session.path, cwd, parsed.sessionDir);
}
}
return await SessionManager.open(match.session.path, parsed.sessionDir);
}
if (parsed.continue) {
return await SessionManager.continueRecent(cwd, parsed.sessionDir);
}
if (parsed.sessionDir) {
return SessionManager.create(cwd, parsed.sessionDir);
}
if (activeSettings.get("autoResume")) {
const manager = await SessionManager.continueRecent(cwd, parsed.sessionDir);
if (manager.getEntries().length > 0) {
parsed.continue = true;
}
return manager;
}
return undefined;
}
async function maybeAutoChdir(parsed: Args): Promise<void> {
if (parsed.allowHome || parsed.cwd) {
return;
}
const home = os.homedir();
if (!home) {
return;
}
const normalizePath = normalizePathForComparison;
const cwd = normalizePath(getProjectDir());
const normalizedHome = normalizePath(home);
if (cwd !== normalizedHome) {
return;
}
const isDirectory = async (p: string) => {
try {
const s = await fs.stat(p);
return s.isDirectory();
} catch {
return false;
}
};
const candidates = [path.join(home, "tmp"), "/tmp", "/var/tmp"];
for (const candidate of candidates) {
try {
if (!(await isDirectory(candidate))) {
continue;
}
setProjectDir(candidate);
return;
} catch {
}
}
try {
const fallback = os.tmpdir();
if (fallback && normalizePath(fallback) !== cwd && (await isDirectory(fallback))) {
setProjectDir(fallback);
}
} catch {
}
}
function discoverSystemPromptFile(): string | undefined {
const projectPath = findConfigFile("SYSTEM.md", { user: false });
if (projectPath) {
return projectPath;
}
const globalPath = findConfigFile("SYSTEM.md", { user: true });
if (globalPath) {
return globalPath;
}
return undefined;
}
function discoverAppendSystemPromptFile(): string | undefined {
const projectPath = findConfigFile("APPEND_SYSTEM.md", { user: false });
if (projectPath) {
return projectPath;
}
const globalPath = findConfigFile("APPEND_SYSTEM.md", { user: true });
if (globalPath) {
return globalPath;
}
return undefined;
}
async function buildSessionOptions(
parsed: Args,
scopedModels: ScopedModel[],
sessionManager: SessionManager | undefined,
modelRegistry: ModelRegistry,
activeSettings: Settings,
): Promise<{ options: CreateAgentSessionOptions }> {
const options: CreateAgentSessionOptions = {
cwd: parsed.cwd ?? getProjectDir(),
autoApprove: parsed.autoApprove ?? false,
};
const systemPromptSource = parsed.systemPrompt ?? discoverSystemPromptFile();
const resolvedSystemPrompt = await resolvePromptInput(systemPromptSource, "system prompt");
const appendPromptSource = parsed.appendSystemPrompt ?? discoverAppendSystemPromptFile();
const resolvedAppendPrompt = await resolvePromptInput(appendPromptSource, "append system prompt");
if (sessionManager) {
options.sessionManager = sessionManager;
}
if (parsed.providerSessionId) {
options.providerSessionId = parsed.providerSessionId;
}
const modelMatchPreferences = {
usageOrder: activeSettings.getStorage()?.getModelUsageOrder(),
};
if (parsed.model) {
const resolved = resolveCliModel({
cliProvider: parsed.provider,
cliModel: parsed.model,
modelRegistry,
preferences: modelMatchPreferences,
});
if (resolved.warning) {
process.stderr.write(`${chalk.yellow(`Warning: ${resolved.warning}`)}\n`);
}
if (resolved.error) {
if (!parsed.provider && !parsed.model.includes(":")) {
options.modelPattern = parsed.model;
} else {
process.stderr.write(`${chalk.red(resolved.error)}\n`);
process.exit(1);
}
} else if (resolved.model) {
options.model = resolved.model;
activeSettings.overrideModelRoles({
default: resolved.selector ?? `${resolved.model.provider}/${resolved.model.id}`,
});
if (!parsed.thinking && resolved.thinkingLevel) {
options.thinkingLevel = resolved.thinkingLevel;
}
}
} else if (scopedModels.length > 0 && !parsed.continue && !parsed.resume) {
const remembered = activeSettings.getModelRole("default");
if (remembered) {
const rememberedSpec = resolveModelRoleValue(
remembered,
scopedModels.map(scopedModel => scopedModel.model),
{
settings: activeSettings,
matchPreferences: modelMatchPreferences,
modelRegistry,
},
);
const rememberedResolvedModel = rememberedSpec.model;
const rememberedModel = rememberedResolvedModel
? scopedModels.find(
scopedModel =>
scopedModel.model.provider === rememberedResolvedModel.provider &&
scopedModel.model.id === rememberedResolvedModel.id,
)
: scopedModels.find(scopedModel => scopedModel.model.id.toLowerCase() === remembered.toLowerCase());
if (rememberedModel) {
options.model = rememberedModel.model;
if (!parsed.thinking && rememberedSpec.explicitThinkingLevel && rememberedSpec.thinkingLevel) {
options.thinkingLevel = rememberedSpec.thinkingLevel;
}
}
}
if (!options.model) options.model = scopedModels[0].model;
}
if (parsed.thinking) {
options.thinkingLevel = parsed.thinking;
} else if (
scopedModels.length > 0 &&
scopedModels[0].explicitThinkingLevel === true &&
!parsed.continue &&
!parsed.resume
) {
options.thinkingLevel = scopedModels[0].thinkingLevel;
}
if (scopedModels.length > 0) {
const defaultThinkingLevelSetting = activeSettings.get("defaultThinkingLevel");
const defaultThinkingLevel =
defaultThinkingLevelSetting === AUTO_THINKING ? undefined : defaultThinkingLevelSetting;
options.scopedModels = scopedModels.map(scopedModel => ({
model: scopedModel.model,
thinkingLevel: scopedModel.explicitThinkingLevel
? (scopedModel.thinkingLevel ?? defaultThinkingLevel)
: defaultThinkingLevel,
}));
}
if (resolvedSystemPrompt && resolvedAppendPrompt) {
options.systemPrompt = defaultPrompt => [resolvedSystemPrompt, resolvedAppendPrompt, ...defaultPrompt.slice(1)];
} else if (resolvedSystemPrompt) {
options.systemPrompt = defaultPrompt => [resolvedSystemPrompt, ...defaultPrompt.slice(1)];
} else if (resolvedAppendPrompt) {
options.systemPrompt = defaultPrompt => [...defaultPrompt, resolvedAppendPrompt];
}
if (parsed.noTools) {
options.toolNames = parsed.tools && parsed.tools.length > 0 ? parsed.tools : [];
} else if (parsed.tools) {
options.toolNames = parsed.tools;
}
if (parsed.noLsp) {
options.enableLsp = false;
}
if (parsed.noSkills) {
options.skills = [];
} else if (parsed.skills && parsed.skills.length > 0) {
activeSettings.override("skills.includeSkills", parsed.skills as string[]);
}
if (parsed.noRules) {
options.rules = [];
}
const cliExtensionPaths = parsed.noExtensions ? [] : [...(parsed.extensions ?? []), ...(parsed.hooks ?? [])];
if (cliExtensionPaths.length > 0) {
options.additionalExtensionPaths = cliExtensionPaths;
}
if (parsed.noExtensions) {
options.disableExtensionDiscovery = true;
options.additionalExtensionPaths = [];
}
return { options };
}
interface RunRootCommandDependencies {
createAgentSession?: typeof createAgentSession;
discoverAuthStorage?: typeof discoverAuthStorage;
runAcpMode?: typeof runAcpMode;
settings?: Settings;
forceSetupWizard?: boolean;
}
export async function runRootCommand(
parsed: Args,
rawArgs: string[],
deps: RunRootCommandDependencies = {},
): Promise<void> {
logger.startTiming();
await logger.time("initTheme:initial", initTheme);
const parsedArgs = parsed;
await logger.time("maybeAutoChdir", maybeAutoChdir, parsedArgs);
const notifs: (InteractiveModeNotify | null)[] = [];
const authStorage = await logger.time("discoverModels", deps.discoverAuthStorage ?? discoverAuthStorage);
const modelRegistry = new ModelRegistry(authStorage);
if (parsedArgs.version) {
process.stdout.write(`${VERSION}\n`);
process.exit(0);
}
if (parsedArgs.listModels !== undefined) {
const settingsInstance = await logger.time("settings:init:list-models", Settings.init, {
cwd: getProjectDir(),
});
await modelRegistry.refresh("online");
const cliExtensionPaths = parsedArgs.noExtensions
? []
: [...(parsedArgs.extensions ?? []), ...(parsedArgs.hooks ?? [])];
const settingsExtensions = settingsInstance.get("extensions") ?? [];
const disabledExtensionIds = settingsInstance.get("disabledExtensions") ?? [];
const searchPattern = typeof parsedArgs.listModels === "string" ? parsedArgs.listModels : undefined;
await runListModelsCommand({
modelRegistry,
cwd: getProjectDir(),
additionalExtensionPaths: cliExtensionPaths,
settingsExtensions,
disabledExtensionIds,
disableExtensionDiscovery: Boolean(parsedArgs.noExtensions),
searchPattern,
});
process.exit(0);
}
if (parsedArgs.export) {
let result: string;
try {
const outputPath = parsedArgs.messages.length > 0 ? parsedArgs.messages[0] : undefined;
result = await exportFromFile(parsedArgs.export, outputPath);
} catch (error: unknown) {
const message = error instanceof Error ? error.message : "Failed to export session";
process.stderr.write(`${chalk.red(`Error: ${message}`)}\n`);
process.exit(1);
}
process.stdout.write(`Exported to: ${result}\n`);
process.exit(0);
}
if ((parsedArgs.mode === "rpc" || parsedArgs.mode === "rpc-ui") && parsedArgs.fileArgs.length > 0) {
process.stderr.write(`${chalk.red("Error: @file arguments are not supported in RPC mode")}\n`);
process.exit(1);
}
const home = os.homedir();
const pluginPreloadPromise =
parsedArgs.pluginDirs && parsedArgs.pluginDirs.length > 0
? logger.time("injectPluginDirRoots", injectPluginDirRoots, home, parsedArgs.pluginDirs, getProjectDir())
: logger.time("preloadPluginRoots", preloadPluginRoots, home, getProjectDir());
pluginPreloadPromise.catch(() => {});
if (!parsedArgs.noExtensions) {
const cliExtensions = [...(parsedArgs.extensions ?? []), ...(parsedArgs.hooks ?? [])];
if (cliExtensions.length > 0) {
injectOmpExtensionCliRoots(cliExtensions, home, getProjectDir());
}
}
const cwd = getProjectDir();
const settingsInstance = deps.settings ?? (await logger.time("settings:init", Settings.init, { cwd }));
if (parsedArgs.approvalMode) {
settingsInstance.override("tools.approvalMode", parsedArgs.approvalMode);
}
if (parsedArgs.mode === "rpc" || parsedArgs.mode === "rpc-ui" || parsedArgs.mode === "acp") {
applyRpcDefaultSettingOverrides(settingsInstance);
}
if (parsedArgs.noPty || parsedArgs.mode === "rpc-ui") {
Bun.env.PI_NO_PTY = "1";
}
if (parsedArgs.noTitle || parsedArgs.mode === "rpc" || parsedArgs.mode === "rpc-ui" || parsedArgs.mode === "acp") {
Bun.env.PI_NO_TITLE = "1";
}
const pipedInput = await logger.time("readPipedInput", readPipedInput);
const autoPrint = pipedInput !== undefined && !parsedArgs.print && parsedArgs.mode === undefined;
const isInteractive = !parsedArgs.print && !autoPrint && parsedArgs.mode === undefined;
const mode = parsedArgs.mode || "text";
logger.time("initializeWithSettings", initializeWithSettings, settingsInstance);
const smolModel = parsedArgs.smol ?? $env.PI_SMOL_MODEL;
const slowModel = parsedArgs.slow ?? $env.PI_SLOW_MODEL;
const planModel = parsedArgs.plan ?? $env.PI_PLAN_MODEL;
if (smolModel || slowModel || planModel) {
settingsInstance.overrideModelRoles({
smol: smolModel,
slow: slowModel,
plan: planModel,
});
}
await logger.time(
"initTheme:final",
initTheme,
isInteractive,
settingsInstance.get("symbolPreset"),
settingsInstance.get("colorBlindMode"),
settingsInstance.get("theme.dark"),
settingsInstance.get("theme.light"),
);
let scopedModels: ScopedModel[] = [];
const modelPatterns = parsedArgs.models ?? settingsInstance.get("enabledModels");
const modelMatchPreferences = {
usageOrder: settingsInstance.getStorage()?.getModelUsageOrder(),
};
if (modelPatterns && modelPatterns.length > 0) {
scopedModels = await logger.time(
"resolveModelScope",
resolveModelScope,
modelPatterns,
modelRegistry,
modelMatchPreferences,
);
}
let sessionManager = await logger.time(
"createSessionManager",
createSessionManager,
parsedArgs,
cwd,
settingsInstance,
);
if (parsedArgs.resume === true && !parsedArgs.fork) {
const sessions = await logger.time("SessionManager.list", SessionManager.list, cwd, parsedArgs.sessionDir);
if (sessions.length === 0) {
process.stdout.write(`${chalk.dim("No sessions found")}\n`);
return;
}
const selectedPath = await logger.time("selectSession", selectSession, sessions);
if (!selectedPath) {
process.stdout.write(`${chalk.dim("No session selected")}\n`);
return;
}
sessionManager = await SessionManager.open(selectedPath);
}
await pluginPreloadPromise;
const autoUpdate = settingsInstance.get("marketplace.autoUpdate");
if (autoUpdate !== "off") {
void (async () => {
try {
const mgr = new MarketplaceManager({
marketplacesRegistryPath: getMarketplacesRegistryPath(),
installedRegistryPath: getInstalledPluginsRegistryPath(),
projectInstalledRegistryPath: (await resolveActiveProjectRegistryPath(getProjectDir())) ?? undefined,
marketplacesCacheDir: getMarketplacesCacheDir(),
pluginsCacheDir: getPluginsCacheDir(),
clearPluginRootsCache: clearPluginRootsAndCaches,
});
await mgr.refreshStaleMarketplaces();
const updates = await mgr.checkForUpdates();
if (updates.length === 0) return;
if (autoUpdate === "auto") {
await mgr.upgradeAllPlugins();
logger.debug(`Auto-upgraded ${updates.length} marketplace plugin(s)`);
} else {
logger.debug(`${updates.length} marketplace plugin update(s) available — /marketplace upgrade`);
}
} catch {
}
})();
}
const { options: sessionOptions } = await logger.time(
"buildSessionOptions",
buildSessionOptions,
parsedArgs,
scopedModels,
sessionManager,
modelRegistry,
settingsInstance,
);
sessionOptions.authStorage = authStorage;
sessionOptions.modelRegistry = modelRegistry;
sessionOptions.hasUI = isInteractive || mode === "rpc-ui";
sessionOptions.settings = settingsInstance;
if (parsedArgs.apiKey) {
if (!sessionOptions.model && !sessionOptions.modelPattern) {
process.stderr.write(
`${chalk.red("--api-key requires a model to be specified via --model, --provider/--model, or --models")}\n`,
);
process.exit(1);
}
if (sessionOptions.model) {
authStorage.setRuntimeApiKey(sessionOptions.model.provider, parsedArgs.apiKey);
}
}
const createAgentSessionImpl = deps.createAgentSession ?? createAgentSession;
const createSession = async (options: CreateAgentSessionOptions): Promise<CreateAgentSessionResult> => {
const result = await logger.time("createAgentSession", createAgentSessionImpl, options);
modelRegistry.refreshInBackground();
return result;
};
if (mode === "acp") {
const createAcpSession = createAcpSessionFactory({
baseOptions: sessionOptions,
settings: settingsInstance,
sessionDir: parsedArgs.sessionDir,
authStorage,
modelRegistry,
parsedArgs,
rawArgs,
createSession,
});
await (deps.runAcpMode ?? runAcpMode)(createAcpSession);
} else {
const eventBus = new EventBus();
const extensionsResult = await loadSessionExtensions(sessionOptions, cwd, settingsInstance, eventBus);
const extensionFlagSink: ExtensionFlagSink = {
getFlags: () => ExtensionRunner.aggregateFlags(extensionsResult.extensions),
setFlagValue: (name, value) => {
extensionsResult.runtime.flagValues.set(name, value);
},
};
const initialArgs = applyExtensionFlags(extensionFlagSink, rawArgs) ?? parsedArgs;
const processedFiles =
initialArgs.fileArgs.length > 0
? await logger.time("processFileArguments", () =>
processFileArguments(initialArgs.fileArgs, {
autoResizeImages: settingsInstance.get("images.autoResize"),
}),
)
: undefined;
const { initialMessage, initialImages } = buildInitialMessage({
parsed: initialArgs,
fileText: processedFiles?.text,
fileImages: processedFiles?.images,
stdinContent: pipedInput,
});
const { session, setToolUIContext, modelFallbackMessage, lspServers, mcpManager } = await createSession({
...sessionOptions,
eventBus,
preloadedExtensions: extensionsResult,
});
if (parsedArgs.apiKey && !sessionOptions.model && session.model) {
authStorage.setRuntimeApiKey(session.model.provider, parsedArgs.apiKey);
}
if (modelFallbackMessage) {
notifs.push({ kind: "warn", message: modelFallbackMessage });
}
const modelRegistryError = modelRegistry.getError();
if (modelRegistryError) {
notifs.push({ kind: "error", message: modelRegistryError.message });
}
if (!isInteractive && !session.model) {
if (modelFallbackMessage) {
process.stderr.write(`${chalk.red(modelFallbackMessage)}\n`);
} else {
process.stderr.write(`${chalk.red("No models available.")}\n`);
}
process.stderr.write(`${chalk.yellow("\nSet an API key environment variable:")}\n`);
process.stderr.write(" ANTHROPIC_API_KEY, OPENAI_API_KEY, GEMINI_API_KEY, etc.\n");
process.stderr.write(`${chalk.yellow(`\nOr create ${ModelsConfigFile.path()}`)}\n`);
process.exit(1);
}
if (mode === "rpc" || mode === "rpc-ui") {
await runRpcMode(session, mode === "rpc-ui" ? setToolUIContext : undefined);
} else if (isInteractive) {
const versionCheckPromise = checkForNewVersion(VERSION).catch(() => undefined);
const changelogMarkdown = await logger.time("main:getChangelogForDisplay", getChangelogForDisplay, parsedArgs);
const scopedModelsForDisplay = sessionOptions.scopedModels ?? scopedModels;
if (scopedModelsForDisplay.length > 0) {
const modelList = scopedModelsForDisplay
.map(scopedModel => {
const thinkingStr = !scopedModel.thinkingLevel ? `:${scopedModel.thinkingLevel}` : "";
return `${scopedModel.model.id}${thinkingStr}`;
})
.join(", ");
process.stdout.write(`${chalk.dim(`Model scope: ${modelList} ${chalk.gray("(Ctrl+P to cycle)")}`)}\n`);
}
if ($env.PI_TIMING) {
logger.printTimings();
if ($env.PI_TIMING === "x") {
process.exit(0);
}
}
logger.endTiming();
await runInteractiveMode(
session,
VERSION,
changelogMarkdown,
notifs,
versionCheckPromise,
initialArgs.messages,
setToolUIContext,
lspServers,
mcpManager,
Boolean(parsedArgs.continue || parsedArgs.resume || parsedArgs.fork),
deps.forceSetupWizard === true,
eventBus,
initialMessage,
initialImages,
);
} else {
await runPrintMode(session, {
mode,
messages: initialArgs.messages,
initialMessage,
initialImages,
});
if ($env.PI_TIMING) {
logger.printTimings();
}
await session.dispose();
stopThemeWatcher();
await postmortem.quit(0);
}
}
}
export async function main(args: string[]): Promise<void> {
const { runCli } = await import("./cli");
await runCli(args.length === 0 ? ["launch"] : args);
}