mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-28 17:43:05 +00:00
227 lines
7.5 KiB
TypeScript
227 lines
7.5 KiB
TypeScript
import type {
|
|
MemoryFlushPlan,
|
|
MemoryPromptSectionBuilder,
|
|
OpenClawConfig,
|
|
} from "openclaw/plugin-sdk/memory-core";
|
|
import {
|
|
DEFAULT_PI_COMPACTION_RESERVE_TOKENS_FLOOR,
|
|
parseNonNegativeByteSize,
|
|
resolveCronStyleNow,
|
|
SILENT_REPLY_TOKEN,
|
|
} from "openclaw/plugin-sdk/memory-core";
|
|
import { definePluginEntry } from "openclaw/plugin-sdk/plugin-entry";
|
|
import { registerMemoryCli } from "./src/cli.js";
|
|
import { createMemoryGetTool, createMemorySearchTool } from "./src/tools.js";
|
|
|
|
export const buildPromptSection: MemoryPromptSectionBuilder = ({
|
|
availableTools,
|
|
citationsMode,
|
|
}) => {
|
|
const hasMemorySearch = availableTools.has("memory_search");
|
|
const hasMemoryGet = availableTools.has("memory_get");
|
|
|
|
if (!hasMemorySearch && !hasMemoryGet) {
|
|
return [];
|
|
}
|
|
|
|
let toolGuidance: string;
|
|
if (hasMemorySearch && hasMemoryGet) {
|
|
toolGuidance =
|
|
"Before answering anything about prior work, decisions, dates, people, preferences, or todos: run memory_search on MEMORY.md + memory/*.md; then use memory_get to pull only the needed lines. If low confidence after search, say you checked.";
|
|
} else if (hasMemorySearch) {
|
|
toolGuidance =
|
|
"Before answering anything about prior work, decisions, dates, people, preferences, or todos: run memory_search on MEMORY.md + memory/*.md and answer from the matching results. If low confidence after search, say you checked.";
|
|
} else {
|
|
toolGuidance =
|
|
"Before answering anything about prior work, decisions, dates, people, preferences, or todos that already point to a specific memory file or note: run memory_get to pull only the needed lines. If low confidence after reading them, say you checked.";
|
|
}
|
|
|
|
const lines = ["## Memory Recall", toolGuidance];
|
|
if (citationsMode === "off") {
|
|
lines.push(
|
|
"Citations are disabled: do not mention file paths or line numbers in replies unless the user explicitly asks.",
|
|
);
|
|
} else {
|
|
lines.push(
|
|
"Citations: include Source: <path#line> when it helps the user verify memory snippets.",
|
|
);
|
|
}
|
|
lines.push("");
|
|
return lines;
|
|
};
|
|
|
|
export const DEFAULT_MEMORY_FLUSH_SOFT_TOKENS = 4000;
|
|
export const DEFAULT_MEMORY_FLUSH_FORCE_TRANSCRIPT_BYTES = 2 * 1024 * 1024;
|
|
|
|
const MEMORY_FLUSH_TARGET_HINT =
|
|
"Store durable memories only in memory/YYYY-MM-DD.md (create memory/ if needed).";
|
|
const MEMORY_FLUSH_APPEND_ONLY_HINT =
|
|
"If memory/YYYY-MM-DD.md already exists, APPEND new content only and do not overwrite existing entries.";
|
|
const MEMORY_FLUSH_READ_ONLY_HINT =
|
|
"Treat workspace bootstrap/reference files such as MEMORY.md, SOUL.md, TOOLS.md, and AGENTS.md as read-only during this flush; never overwrite, replace, or edit them.";
|
|
const MEMORY_FLUSH_REQUIRED_HINTS = [
|
|
MEMORY_FLUSH_TARGET_HINT,
|
|
MEMORY_FLUSH_APPEND_ONLY_HINT,
|
|
MEMORY_FLUSH_READ_ONLY_HINT,
|
|
];
|
|
|
|
export const DEFAULT_MEMORY_FLUSH_PROMPT = [
|
|
"Pre-compaction memory flush.",
|
|
MEMORY_FLUSH_TARGET_HINT,
|
|
MEMORY_FLUSH_READ_ONLY_HINT,
|
|
MEMORY_FLUSH_APPEND_ONLY_HINT,
|
|
"Do NOT create timestamped variant files (e.g., YYYY-MM-DD-HHMM.md); always use the canonical YYYY-MM-DD.md filename.",
|
|
`If nothing to store, reply with ${SILENT_REPLY_TOKEN}.`,
|
|
].join(" ");
|
|
|
|
export const DEFAULT_MEMORY_FLUSH_SYSTEM_PROMPT = [
|
|
"Pre-compaction memory flush turn.",
|
|
"The session is near auto-compaction; capture durable memories to disk.",
|
|
MEMORY_FLUSH_TARGET_HINT,
|
|
MEMORY_FLUSH_READ_ONLY_HINT,
|
|
MEMORY_FLUSH_APPEND_ONLY_HINT,
|
|
`You may reply, but usually ${SILENT_REPLY_TOKEN} is correct.`,
|
|
].join(" ");
|
|
|
|
function formatDateStampInTimezone(nowMs: number, timezone: string): string {
|
|
const parts = new Intl.DateTimeFormat("en-US", {
|
|
timeZone: timezone,
|
|
year: "numeric",
|
|
month: "2-digit",
|
|
day: "2-digit",
|
|
}).formatToParts(new Date(nowMs));
|
|
const year = parts.find((part) => part.type === "year")?.value;
|
|
const month = parts.find((part) => part.type === "month")?.value;
|
|
const day = parts.find((part) => part.type === "day")?.value;
|
|
if (year && month && day) {
|
|
return `${year}-${month}-${day}`;
|
|
}
|
|
return new Date(nowMs).toISOString().slice(0, 10);
|
|
}
|
|
|
|
function normalizeNonNegativeInt(value: unknown): number | null {
|
|
if (typeof value !== "number" || !Number.isFinite(value)) {
|
|
return null;
|
|
}
|
|
const int = Math.floor(value);
|
|
return int >= 0 ? int : null;
|
|
}
|
|
|
|
function ensureNoReplyHint(text: string): string {
|
|
if (text.includes(SILENT_REPLY_TOKEN)) {
|
|
return text;
|
|
}
|
|
return `${text}\n\nIf no user-visible reply is needed, start with ${SILENT_REPLY_TOKEN}.`;
|
|
}
|
|
|
|
function ensureMemoryFlushSafetyHints(text: string): string {
|
|
let next = text.trim();
|
|
for (const hint of MEMORY_FLUSH_REQUIRED_HINTS) {
|
|
if (!next.includes(hint)) {
|
|
next = next ? `${next}\n\n${hint}` : hint;
|
|
}
|
|
}
|
|
return next;
|
|
}
|
|
|
|
function appendCurrentTimeLine(text: string, timeLine: string): string {
|
|
const trimmed = text.trimEnd();
|
|
if (!trimmed) {
|
|
return timeLine;
|
|
}
|
|
if (trimmed.includes("Current time:")) {
|
|
return trimmed;
|
|
}
|
|
return `${trimmed}\n${timeLine}`;
|
|
}
|
|
|
|
export function buildMemoryFlushPlan(
|
|
params: {
|
|
cfg?: OpenClawConfig;
|
|
nowMs?: number;
|
|
} = {},
|
|
): MemoryFlushPlan | null {
|
|
const resolved = params;
|
|
const nowMs = Number.isFinite(resolved.nowMs) ? (resolved.nowMs as number) : Date.now();
|
|
const cfg = resolved.cfg;
|
|
const defaults = cfg?.agents?.defaults?.compaction?.memoryFlush;
|
|
if (defaults?.enabled === false) {
|
|
return null;
|
|
}
|
|
|
|
const softThresholdTokens =
|
|
normalizeNonNegativeInt(defaults?.softThresholdTokens) ?? DEFAULT_MEMORY_FLUSH_SOFT_TOKENS;
|
|
const forceFlushTranscriptBytes =
|
|
parseNonNegativeByteSize(defaults?.forceFlushTranscriptBytes) ??
|
|
DEFAULT_MEMORY_FLUSH_FORCE_TRANSCRIPT_BYTES;
|
|
const reserveTokensFloor =
|
|
normalizeNonNegativeInt(cfg?.agents?.defaults?.compaction?.reserveTokensFloor) ??
|
|
DEFAULT_PI_COMPACTION_RESERVE_TOKENS_FLOOR;
|
|
|
|
const { timeLine, userTimezone } = resolveCronStyleNow(cfg ?? {}, nowMs);
|
|
const dateStamp = formatDateStampInTimezone(nowMs, userTimezone);
|
|
const relativePath = `memory/${dateStamp}.md`;
|
|
|
|
const promptBase = ensureNoReplyHint(
|
|
ensureMemoryFlushSafetyHints(defaults?.prompt?.trim() || DEFAULT_MEMORY_FLUSH_PROMPT),
|
|
);
|
|
const systemPrompt = ensureNoReplyHint(
|
|
ensureMemoryFlushSafetyHints(
|
|
defaults?.systemPrompt?.trim() || DEFAULT_MEMORY_FLUSH_SYSTEM_PROMPT,
|
|
),
|
|
);
|
|
|
|
return {
|
|
softThresholdTokens,
|
|
forceFlushTranscriptBytes,
|
|
reserveTokensFloor,
|
|
prompt: appendCurrentTimeLine(promptBase.replaceAll("YYYY-MM-DD", dateStamp), timeLine),
|
|
systemPrompt: systemPrompt.replaceAll("YYYY-MM-DD", dateStamp),
|
|
relativePath,
|
|
};
|
|
}
|
|
|
|
export default definePluginEntry({
|
|
id: "memory-core",
|
|
name: "Memory (Core)",
|
|
description: "File-backed memory search tools and CLI",
|
|
kind: "memory",
|
|
register(api) {
|
|
api.registerMemoryPromptSection(buildPromptSection);
|
|
api.registerMemoryFlushPlan(buildMemoryFlushPlan);
|
|
|
|
api.registerTool(
|
|
(ctx) =>
|
|
createMemorySearchTool({
|
|
config: ctx.config,
|
|
agentSessionKey: ctx.sessionKey,
|
|
}),
|
|
{ names: ["memory_search"] },
|
|
);
|
|
|
|
api.registerTool(
|
|
(ctx) =>
|
|
createMemoryGetTool({
|
|
config: ctx.config,
|
|
agentSessionKey: ctx.sessionKey,
|
|
}),
|
|
{ names: ["memory_get"] },
|
|
);
|
|
|
|
api.registerCli(
|
|
({ program }) => {
|
|
registerMemoryCli(program);
|
|
},
|
|
{
|
|
descriptors: [
|
|
{
|
|
name: "memory",
|
|
description: "Search, inspect, and reindex memory files",
|
|
hasSubcommands: true,
|
|
},
|
|
],
|
|
},
|
|
);
|
|
},
|
|
});
|