import type { Command } from "commander"; import type { CronJob } from "../../cron/types.js"; import { danger } from "../../globals.js"; import { normalizeAgentId } from "../../routing/session-key.js"; import { defaultRuntime } from "../../runtime.js"; import type { GatewayRpcOpts } from "../gateway-rpc.js"; import { addGatewayClientOptions, callGatewayFromCli } from "../gateway-rpc.js"; import { parsePositiveIntOrUndefined } from "../program/helpers.js"; import { CRON_CHANNEL_OPTIONS, parseAtMs, parseDurationMs, printCronList, warnIfCronSchedulerDisabled, } from "./shared.js"; export function registerCronStatusCommand(cron: Command) { addGatewayClientOptions( cron .command("status") .description("Show cron scheduler status") .option("--json", "Output JSON", false) .action(async (opts) => { try { const res = await callGatewayFromCli("cron.status", opts, {}); defaultRuntime.log(JSON.stringify(res, null, 2)); } catch (err) { defaultRuntime.error(danger(String(err))); defaultRuntime.exit(1); } }), ); } export function registerCronListCommand(cron: Command) { addGatewayClientOptions( cron .command("list") .description("List cron jobs") .option("--all", "Include disabled jobs", false) .option("--json", "Output JSON", false) .action(async (opts) => { try { const res = await callGatewayFromCli("cron.list", opts, { includeDisabled: Boolean(opts.all), }); if (opts.json) { defaultRuntime.log(JSON.stringify(res, null, 2)); return; } const jobs = (res as { jobs?: CronJob[] } | null)?.jobs ?? []; printCronList(jobs, defaultRuntime); } catch (err) { defaultRuntime.error(danger(String(err))); defaultRuntime.exit(1); } }), ); } export function registerCronAddCommand(cron: Command) { addGatewayClientOptions( cron .command("add") .alias("create") .description("Add a cron job") .requiredOption("--name ", "Job name") .option("--description ", "Optional description") .option("--disabled", "Create job disabled", false) .option( "--delete-after-run", "Delete one-shot job after it succeeds", false, ) .option("--agent ", "Agent id for this job") .option("--session ", "Session target (main|isolated)", "main") .option( "--wake ", "Wake mode (now|next-heartbeat)", "next-heartbeat", ) .option("--at ", "Run once at time (ISO) or +duration (e.g. 20m)") .option("--every ", "Run every duration (e.g. 10m, 1h)") .option("--cron ", "Cron expression (5-field)") .option("--tz ", "Timezone for cron expressions (IANA)", "") .option("--system-event ", "System event payload (main session)") .option("--message ", "Agent message payload") .option( "--thinking ", "Thinking level for agent jobs (off|minimal|low|medium|high)", ) .option( "--model ", "Model override for agent jobs (provider/model or alias)", ) .option("--timeout-seconds ", "Timeout seconds for agent jobs") .option("--deliver", "Deliver agent output", false) .option( "--channel ", `Delivery channel (${CRON_CHANNEL_OPTIONS})`, "last", ) .option( "--to ", "Delivery destination (E.164, Telegram chatId, or Discord channel/user)", ) .option( "--best-effort-deliver", "Do not fail the job if delivery fails", false, ) .option( "--post-prefix ", "Prefix for summary system event", "Cron", ) .option("--json", "Output JSON", false) .action(async (opts: GatewayRpcOpts & Record) => { try { const schedule = (() => { const at = typeof opts.at === "string" ? opts.at : ""; const every = typeof opts.every === "string" ? opts.every : ""; const cronExpr = typeof opts.cron === "string" ? opts.cron : ""; const chosen = [ Boolean(at), Boolean(every), Boolean(cronExpr), ].filter(Boolean).length; if (chosen !== 1) { throw new Error( "Choose exactly one schedule: --at, --every, or --cron", ); } if (at) { const atMs = parseAtMs(at); if (!atMs) throw new Error( "Invalid --at; use ISO time or duration like 20m", ); return { kind: "at" as const, atMs }; } if (every) { const everyMs = parseDurationMs(every); if (!everyMs) throw new Error("Invalid --every; use e.g. 10m, 1h, 1d"); return { kind: "every" as const, everyMs }; } return { kind: "cron" as const, expr: cronExpr, tz: typeof opts.tz === "string" && opts.tz.trim() ? opts.tz.trim() : undefined, }; })(); const sessionTargetRaw = typeof opts.session === "string" ? opts.session : "main"; const sessionTarget = sessionTargetRaw.trim() || "main"; if (sessionTarget !== "main" && sessionTarget !== "isolated") { throw new Error("--session must be main or isolated"); } const wakeModeRaw = typeof opts.wake === "string" ? opts.wake : "next-heartbeat"; const wakeMode = wakeModeRaw.trim() || "next-heartbeat"; if (wakeMode !== "now" && wakeMode !== "next-heartbeat") { throw new Error("--wake must be now or next-heartbeat"); } const agentId = typeof opts.agent === "string" && opts.agent.trim() ? normalizeAgentId(opts.agent) : undefined; const payload = (() => { const systemEvent = typeof opts.systemEvent === "string" ? opts.systemEvent.trim() : ""; const message = typeof opts.message === "string" ? opts.message.trim() : ""; const chosen = [Boolean(systemEvent), Boolean(message)].filter( Boolean, ).length; if (chosen !== 1) { throw new Error( "Choose exactly one payload: --system-event or --message", ); } if (systemEvent) return { kind: "systemEvent" as const, text: systemEvent }; const timeoutSeconds = parsePositiveIntOrUndefined( opts.timeoutSeconds, ); return { kind: "agentTurn" as const, message, model: typeof opts.model === "string" && opts.model.trim() ? opts.model.trim() : undefined, thinking: typeof opts.thinking === "string" && opts.thinking.trim() ? opts.thinking.trim() : undefined, timeoutSeconds: timeoutSeconds && Number.isFinite(timeoutSeconds) ? timeoutSeconds : undefined, deliver: Boolean(opts.deliver), channel: typeof opts.channel === "string" ? opts.channel : "last", to: typeof opts.to === "string" && opts.to.trim() ? opts.to.trim() : undefined, bestEffortDeliver: Boolean(opts.bestEffortDeliver), }; })(); if (sessionTarget === "main" && payload.kind !== "systemEvent") { throw new Error("Main jobs require --system-event (systemEvent)."); } if (sessionTarget === "isolated" && payload.kind !== "agentTurn") { throw new Error("Isolated jobs require --message (agentTurn)."); } const isolation = sessionTarget === "isolated" ? { postToMainPrefix: typeof opts.postPrefix === "string" && opts.postPrefix.trim() ? opts.postPrefix.trim() : "Cron", } : undefined; const nameRaw = typeof opts.name === "string" ? opts.name : ""; const name = nameRaw.trim(); if (!name) throw new Error("--name is required"); const description = typeof opts.description === "string" && opts.description.trim() ? opts.description.trim() : undefined; const params = { name, description, enabled: !opts.disabled, deleteAfterRun: Boolean(opts.deleteAfterRun), agentId, schedule, sessionTarget, wakeMode, payload, isolation, }; const res = await callGatewayFromCli("cron.add", opts, params); defaultRuntime.log(JSON.stringify(res, null, 2)); await warnIfCronSchedulerDisabled(opts); } catch (err) { defaultRuntime.error(danger(String(err))); defaultRuntime.exit(1); } }), ); }