mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-29 23:43:39 +00:00
128 lines
4.3 KiB
TypeScript
128 lines
4.3 KiB
TypeScript
/**
|
|
* Shared session persistence and prompt-body helpers for agent attempt
|
|
* execution paths.
|
|
*/
|
|
import { patchSessionEntry } from "../../config/sessions/session-accessor.js";
|
|
import { mergeSessionEntry, type SessionEntry } from "../../config/sessions/types.js";
|
|
import {
|
|
formatAgentInternalEventsForPlainPrompt,
|
|
formatAgentInternalEventsForPrompt,
|
|
} from "../internal-events.js";
|
|
import {
|
|
hasInternalRuntimeContext,
|
|
stripInternalRuntimeContext,
|
|
} from "../internal-runtime-context.js";
|
|
import type { AgentCommandOpts } from "./types.js";
|
|
|
|
/** Parameters for merging and persisting a session entry update. */
|
|
type PersistSessionEntryParams = {
|
|
sessionStore: Record<string, SessionEntry>;
|
|
sessionKey: string;
|
|
storePath: string;
|
|
entry: SessionEntry;
|
|
clearedFields?: string[];
|
|
preserveTranscriptMarkerUpdatedAt?: boolean;
|
|
shouldPersist?: (entry: SessionEntry | undefined) => boolean;
|
|
};
|
|
|
|
/** Persists one session entry while keeping the caller's in-memory store aligned. */
|
|
|
|
function normalizeTranscriptMarkerUpdatedAt(value: number | undefined): number | undefined {
|
|
return typeof value === "number" && Number.isFinite(value) && value >= 0 ? value : undefined;
|
|
}
|
|
|
|
export async function persistSessionEntry(
|
|
params: PersistSessionEntryParams,
|
|
): Promise<SessionEntry | undefined> {
|
|
let rejectedMissingEntry = false;
|
|
const persisted = await patchSessionEntry(
|
|
{ sessionKey: params.sessionKey, storePath: params.storePath },
|
|
(_entry, context) => {
|
|
if (params.shouldPersist && !params.shouldPersist(context.existingEntry)) {
|
|
rejectedMissingEntry = !context.existingEntry;
|
|
return null;
|
|
}
|
|
const merged = mergeSessionEntry(context.existingEntry, params.entry);
|
|
if (params.preserveTranscriptMarkerUpdatedAt) {
|
|
const currentUpdatedAt = normalizeTranscriptMarkerUpdatedAt(
|
|
context.existingEntry?.updatedAt,
|
|
);
|
|
const markerUpdatedAt = normalizeTranscriptMarkerUpdatedAt(params.entry.updatedAt);
|
|
if (markerUpdatedAt !== undefined) {
|
|
merged.updatedAt = Math.max(currentUpdatedAt ?? 0, markerUpdatedAt);
|
|
}
|
|
}
|
|
for (const field of params.clearedFields ?? []) {
|
|
// Cleared fields only apply when the replacement entry did not set the
|
|
// field again; this preserves explicit false/null updates.
|
|
if (!Object.hasOwn(params.entry, field)) {
|
|
Reflect.deleteProperty(merged, field);
|
|
}
|
|
}
|
|
return merged;
|
|
},
|
|
{
|
|
fallbackEntry: params.sessionStore[params.sessionKey] ?? params.entry,
|
|
replaceEntry: true,
|
|
},
|
|
);
|
|
if (rejectedMissingEntry) {
|
|
delete params.sessionStore[params.sessionKey];
|
|
return undefined;
|
|
}
|
|
if (persisted) {
|
|
params.sessionStore[params.sessionKey] = persisted;
|
|
} else {
|
|
delete params.sessionStore[params.sessionKey];
|
|
}
|
|
return persisted ?? undefined;
|
|
}
|
|
|
|
/** Prepends hidden internal event context unless the body already carries it. */
|
|
export function prependInternalEventContext(
|
|
body: string,
|
|
events: AgentCommandOpts["internalEvents"],
|
|
): string {
|
|
if (hasInternalRuntimeContext(body)) {
|
|
return body;
|
|
}
|
|
const renderedEvents = formatAgentInternalEventsForPrompt(events);
|
|
if (!renderedEvents) {
|
|
return body;
|
|
}
|
|
return [renderedEvents, body].filter(Boolean).join("\n\n");
|
|
}
|
|
|
|
// ACP/plain transcript bodies cannot carry internal runtime context markup, so
|
|
// render events as visible plain text before stripping hidden sections.
|
|
function resolvePlainInternalEventBody(
|
|
body: string,
|
|
events: AgentCommandOpts["internalEvents"],
|
|
): string {
|
|
const renderedEvents = formatAgentInternalEventsForPlainPrompt(events);
|
|
if (!renderedEvents) {
|
|
return body;
|
|
}
|
|
const visibleBody = stripInternalRuntimeContext(body).trim();
|
|
return [renderedEvents, visibleBody].filter(Boolean).join("\n\n") || body;
|
|
}
|
|
|
|
/** Resolves the prompt body submitted to ACP runtimes. */
|
|
export function resolveAcpPromptBody(
|
|
body: string,
|
|
events: AgentCommandOpts["internalEvents"],
|
|
): string {
|
|
return events?.length ? resolvePlainInternalEventBody(body, events) : body;
|
|
}
|
|
|
|
/** Resolves the body stored in transcripts after internal event rendering. */
|
|
export function resolveInternalEventTranscriptBody(
|
|
body: string,
|
|
events: AgentCommandOpts["internalEvents"],
|
|
): string {
|
|
if (!hasInternalRuntimeContext(body)) {
|
|
return body;
|
|
}
|
|
return resolvePlainInternalEventBody(body, events);
|
|
}
|