mirror of
https://github.com/moltbot/moltbot.git
synced 2026-03-07 22:44:16 +00:00
371 lines
12 KiB
TypeScript
371 lines
12 KiB
TypeScript
import { abortEmbeddedPiRun } from "../../agents/pi-embedded.js";
|
|
import { isRestartEnabled } from "../../config/commands.js";
|
|
import type { SessionEntry } from "../../config/sessions.js";
|
|
import { updateSessionStore } from "../../config/sessions.js";
|
|
import { logVerbose } from "../../globals.js";
|
|
import { createInternalHookEvent, triggerInternalHook } from "../../hooks/internal-hooks.js";
|
|
import { scheduleGatewaySigusr1Restart, triggerOpenClawRestart } from "../../infra/restart.js";
|
|
import { loadCostUsageSummary, loadSessionCostSummary } from "../../infra/session-cost-usage.js";
|
|
import { formatTokenCount, formatUsd } from "../../utils/usage-format.js";
|
|
import { parseActivationCommand } from "../group-activation.js";
|
|
import { parseSendPolicyCommand } from "../send-policy.js";
|
|
import { normalizeUsageDisplay, resolveResponseUsageMode } from "../thinking.js";
|
|
import {
|
|
formatAbortReplyText,
|
|
isAbortTrigger,
|
|
resolveSessionEntryForKey,
|
|
setAbortMemory,
|
|
stopSubagentsForRequester,
|
|
} from "./abort.js";
|
|
import type { CommandHandler } from "./commands-types.js";
|
|
import { clearSessionQueues } from "./queue.js";
|
|
|
|
function resolveAbortTarget(params: {
|
|
ctx: { CommandTargetSessionKey?: string | null };
|
|
sessionKey?: string;
|
|
sessionEntry?: SessionEntry;
|
|
sessionStore?: Record<string, SessionEntry>;
|
|
}) {
|
|
const targetSessionKey = params.ctx.CommandTargetSessionKey?.trim() || params.sessionKey;
|
|
const { entry, key } = resolveSessionEntryForKey(params.sessionStore, targetSessionKey);
|
|
if (entry && key) {
|
|
return { entry, key, sessionId: entry.sessionId };
|
|
}
|
|
if (params.sessionEntry && params.sessionKey) {
|
|
return {
|
|
entry: params.sessionEntry,
|
|
key: params.sessionKey,
|
|
sessionId: params.sessionEntry.sessionId,
|
|
};
|
|
}
|
|
return { entry: undefined, key: targetSessionKey, sessionId: undefined };
|
|
}
|
|
|
|
async function applyAbortTarget(params: {
|
|
abortTarget: ReturnType<typeof resolveAbortTarget>;
|
|
sessionStore?: Record<string, SessionEntry>;
|
|
storePath?: string;
|
|
abortKey?: string;
|
|
}) {
|
|
const { abortTarget } = params;
|
|
if (abortTarget.sessionId) {
|
|
abortEmbeddedPiRun(abortTarget.sessionId);
|
|
}
|
|
if (abortTarget.entry && params.sessionStore && abortTarget.key) {
|
|
abortTarget.entry.abortedLastRun = true;
|
|
abortTarget.entry.updatedAt = Date.now();
|
|
params.sessionStore[abortTarget.key] = abortTarget.entry;
|
|
if (params.storePath) {
|
|
await updateSessionStore(params.storePath, (store) => {
|
|
store[abortTarget.key] = abortTarget.entry;
|
|
});
|
|
}
|
|
} else if (params.abortKey) {
|
|
setAbortMemory(params.abortKey, true);
|
|
}
|
|
}
|
|
|
|
async function persistSessionEntry(params: Parameters<CommandHandler>[0]): Promise<boolean> {
|
|
if (!params.sessionEntry || !params.sessionStore || !params.sessionKey) {
|
|
return false;
|
|
}
|
|
params.sessionEntry.updatedAt = Date.now();
|
|
params.sessionStore[params.sessionKey] = params.sessionEntry;
|
|
if (params.storePath) {
|
|
await updateSessionStore(params.storePath, (store) => {
|
|
store[params.sessionKey] = params.sessionEntry as SessionEntry;
|
|
});
|
|
}
|
|
return true;
|
|
}
|
|
|
|
export const handleActivationCommand: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
const activationCommand = parseActivationCommand(params.command.commandBodyNormalized);
|
|
if (!activationCommand.hasCommand) {
|
|
return null;
|
|
}
|
|
if (!params.isGroup) {
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: "⚙️ Group activation only applies to group chats." },
|
|
};
|
|
}
|
|
if (!params.command.isAuthorizedSender) {
|
|
logVerbose(
|
|
`Ignoring /activation from unauthorized sender in group: ${params.command.senderId || "<unknown>"}`,
|
|
);
|
|
return { shouldContinue: false };
|
|
}
|
|
if (!activationCommand.mode) {
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: "⚙️ Usage: /activation mention|always" },
|
|
};
|
|
}
|
|
if (params.sessionEntry && params.sessionStore && params.sessionKey) {
|
|
params.sessionEntry.groupActivation = activationCommand.mode;
|
|
params.sessionEntry.groupActivationNeedsSystemIntro = true;
|
|
await persistSessionEntry(params);
|
|
}
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: `⚙️ Group activation set to ${activationCommand.mode}.`,
|
|
},
|
|
};
|
|
};
|
|
|
|
export const handleSendPolicyCommand: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
const sendPolicyCommand = parseSendPolicyCommand(params.command.commandBodyNormalized);
|
|
if (!sendPolicyCommand.hasCommand) {
|
|
return null;
|
|
}
|
|
if (!params.command.isAuthorizedSender) {
|
|
logVerbose(
|
|
`Ignoring /send from unauthorized sender: ${params.command.senderId || "<unknown>"}`,
|
|
);
|
|
return { shouldContinue: false };
|
|
}
|
|
if (!sendPolicyCommand.mode) {
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: "⚙️ Usage: /send on|off|inherit" },
|
|
};
|
|
}
|
|
if (params.sessionEntry && params.sessionStore && params.sessionKey) {
|
|
if (sendPolicyCommand.mode === "inherit") {
|
|
delete params.sessionEntry.sendPolicy;
|
|
} else {
|
|
params.sessionEntry.sendPolicy = sendPolicyCommand.mode;
|
|
}
|
|
await persistSessionEntry(params);
|
|
}
|
|
const label =
|
|
sendPolicyCommand.mode === "inherit"
|
|
? "inherit"
|
|
: sendPolicyCommand.mode === "allow"
|
|
? "on"
|
|
: "off";
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: `⚙️ Send policy set to ${label}.` },
|
|
};
|
|
};
|
|
|
|
export const handleUsageCommand: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
const normalized = params.command.commandBodyNormalized;
|
|
if (normalized !== "/usage" && !normalized.startsWith("/usage ")) {
|
|
return null;
|
|
}
|
|
if (!params.command.isAuthorizedSender) {
|
|
logVerbose(
|
|
`Ignoring /usage from unauthorized sender: ${params.command.senderId || "<unknown>"}`,
|
|
);
|
|
return { shouldContinue: false };
|
|
}
|
|
|
|
const rawArgs = normalized === "/usage" ? "" : normalized.slice("/usage".length).trim();
|
|
const requested = rawArgs ? normalizeUsageDisplay(rawArgs) : undefined;
|
|
if (rawArgs.toLowerCase().startsWith("cost")) {
|
|
const sessionSummary = await loadSessionCostSummary({
|
|
sessionId: params.sessionEntry?.sessionId,
|
|
sessionEntry: params.sessionEntry,
|
|
sessionFile: params.sessionEntry?.sessionFile,
|
|
config: params.cfg,
|
|
agentId: params.agentId,
|
|
});
|
|
const summary = await loadCostUsageSummary({ days: 30, config: params.cfg });
|
|
|
|
const sessionCost = formatUsd(sessionSummary?.totalCost);
|
|
const sessionTokens = sessionSummary?.totalTokens
|
|
? formatTokenCount(sessionSummary.totalTokens)
|
|
: undefined;
|
|
const sessionMissing = sessionSummary?.missingCostEntries ?? 0;
|
|
const sessionSuffix = sessionMissing > 0 ? " (partial)" : "";
|
|
const sessionLine =
|
|
sessionCost || sessionTokens
|
|
? `Session ${sessionCost ?? "n/a"}${sessionSuffix}${sessionTokens ? ` · ${sessionTokens} tokens` : ""}`
|
|
: "Session n/a";
|
|
|
|
const todayKey = new Date().toLocaleDateString("en-CA");
|
|
const todayEntry = summary.daily.find((entry) => entry.date === todayKey);
|
|
const todayCost = formatUsd(todayEntry?.totalCost);
|
|
const todayMissing = todayEntry?.missingCostEntries ?? 0;
|
|
const todaySuffix = todayMissing > 0 ? " (partial)" : "";
|
|
const todayLine = `Today ${todayCost ?? "n/a"}${todaySuffix}`;
|
|
|
|
const last30Cost = formatUsd(summary.totals.totalCost);
|
|
const last30Missing = summary.totals.missingCostEntries;
|
|
const last30Suffix = last30Missing > 0 ? " (partial)" : "";
|
|
const last30Line = `Last 30d ${last30Cost ?? "n/a"}${last30Suffix}`;
|
|
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: `💸 Usage cost\n${sessionLine}\n${todayLine}\n${last30Line}` },
|
|
};
|
|
}
|
|
|
|
if (rawArgs && !requested) {
|
|
return {
|
|
shouldContinue: false,
|
|
reply: { text: "⚙️ Usage: /usage off|tokens|full|cost" },
|
|
};
|
|
}
|
|
|
|
const currentRaw =
|
|
params.sessionEntry?.responseUsage ??
|
|
(params.sessionKey ? params.sessionStore?.[params.sessionKey]?.responseUsage : undefined);
|
|
const current = resolveResponseUsageMode(currentRaw);
|
|
const next = requested ?? (current === "off" ? "tokens" : current === "tokens" ? "full" : "off");
|
|
|
|
if (params.sessionEntry && params.sessionStore && params.sessionKey) {
|
|
if (next === "off") {
|
|
delete params.sessionEntry.responseUsage;
|
|
} else {
|
|
params.sessionEntry.responseUsage = next;
|
|
}
|
|
await persistSessionEntry(params);
|
|
}
|
|
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: `⚙️ Usage footer: ${next}.`,
|
|
},
|
|
};
|
|
};
|
|
|
|
export const handleRestartCommand: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
if (params.command.commandBodyNormalized !== "/restart") {
|
|
return null;
|
|
}
|
|
if (!params.command.isAuthorizedSender) {
|
|
logVerbose(
|
|
`Ignoring /restart from unauthorized sender: ${params.command.senderId || "<unknown>"}`,
|
|
);
|
|
return { shouldContinue: false };
|
|
}
|
|
if (!isRestartEnabled(params.cfg)) {
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: "⚠️ /restart is disabled (commands.restart=false).",
|
|
},
|
|
};
|
|
}
|
|
const hasSigusr1Listener = process.listenerCount("SIGUSR1") > 0;
|
|
if (hasSigusr1Listener) {
|
|
scheduleGatewaySigusr1Restart({ reason: "/restart" });
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: "⚙️ Restarting OpenClaw in-process (SIGUSR1); back in a few seconds.",
|
|
},
|
|
};
|
|
}
|
|
const restartMethod = triggerOpenClawRestart();
|
|
if (!restartMethod.ok) {
|
|
const detail = restartMethod.detail ? ` Details: ${restartMethod.detail}` : "";
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: `⚠️ Restart failed (${restartMethod.method}).${detail}`,
|
|
},
|
|
};
|
|
}
|
|
return {
|
|
shouldContinue: false,
|
|
reply: {
|
|
text: `⚙️ Restarting OpenClaw via ${restartMethod.method}; give me a few seconds to come back online.`,
|
|
},
|
|
};
|
|
};
|
|
|
|
export const handleStopCommand: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
if (params.command.commandBodyNormalized !== "/stop") {
|
|
return null;
|
|
}
|
|
if (!params.command.isAuthorizedSender) {
|
|
logVerbose(
|
|
`Ignoring /stop from unauthorized sender: ${params.command.senderId || "<unknown>"}`,
|
|
);
|
|
return { shouldContinue: false };
|
|
}
|
|
const abortTarget = resolveAbortTarget({
|
|
ctx: params.ctx,
|
|
sessionKey: params.sessionKey,
|
|
sessionEntry: params.sessionEntry,
|
|
sessionStore: params.sessionStore,
|
|
});
|
|
const cleared = clearSessionQueues([abortTarget.key, abortTarget.sessionId]);
|
|
if (cleared.followupCleared > 0 || cleared.laneCleared > 0) {
|
|
logVerbose(
|
|
`stop: cleared followups=${cleared.followupCleared} lane=${cleared.laneCleared} keys=${cleared.keys.join(",")}`,
|
|
);
|
|
}
|
|
await applyAbortTarget({
|
|
abortTarget,
|
|
sessionStore: params.sessionStore,
|
|
storePath: params.storePath,
|
|
abortKey: params.command.abortKey,
|
|
});
|
|
|
|
// Trigger internal hook for stop command
|
|
const hookEvent = createInternalHookEvent(
|
|
"command",
|
|
"stop",
|
|
abortTarget.key ?? params.sessionKey ?? "",
|
|
{
|
|
sessionEntry: abortTarget.entry ?? params.sessionEntry,
|
|
sessionId: abortTarget.sessionId,
|
|
commandSource: params.command.surface,
|
|
senderId: params.command.senderId,
|
|
},
|
|
);
|
|
await triggerInternalHook(hookEvent);
|
|
|
|
const { stopped } = stopSubagentsForRequester({
|
|
cfg: params.cfg,
|
|
requesterSessionKey: abortTarget.key ?? params.sessionKey,
|
|
});
|
|
|
|
return { shouldContinue: false, reply: { text: formatAbortReplyText(stopped) } };
|
|
};
|
|
|
|
export const handleAbortTrigger: CommandHandler = async (params, allowTextCommands) => {
|
|
if (!allowTextCommands) {
|
|
return null;
|
|
}
|
|
if (!isAbortTrigger(params.command.rawBodyNormalized)) {
|
|
return null;
|
|
}
|
|
const abortTarget = resolveAbortTarget({
|
|
ctx: params.ctx,
|
|
sessionKey: params.sessionKey,
|
|
sessionEntry: params.sessionEntry,
|
|
sessionStore: params.sessionStore,
|
|
});
|
|
await applyAbortTarget({
|
|
abortTarget,
|
|
sessionStore: params.sessionStore,
|
|
storePath: params.storePath,
|
|
abortKey: params.command.abortKey,
|
|
});
|
|
return { shouldContinue: false, reply: { text: "⚙️ Agent was aborted." } };
|
|
};
|