Files
openclaw/extensions/msteams/src/feedback-reflection-store.ts
Peter Steinberger b85b1c68d1 Refactor file access to use fs-safe primitives (#78255)
* refactor: use fs-safe primitives across file access

* fix: preserve invalid managed npm manifests

* fix: keep fs seams for startup metadata
2026-05-06 05:03:11 +01:00

114 lines
4.0 KiB
TypeScript

import fs from "node:fs/promises";
import { writeJsonFileAtomically } from "openclaw/plugin-sdk/json-store";
/** Default cooldown between reflections per session (5 minutes). */
export const DEFAULT_COOLDOWN_MS = 300_000;
/** Tracks last reflection time per session to enforce cooldown. */
const lastReflectionBySession = new Map<string, number>();
/** Maximum cooldown entries before pruning expired ones. */
const MAX_COOLDOWN_ENTRIES = 500;
function legacySanitizeSessionKey(sessionKey: string): string {
return sessionKey.replace(/[^a-zA-Z0-9_-]/g, "_");
}
function encodeSessionKey(sessionKey: string): string {
return Buffer.from(sessionKey, "utf8").toString("base64url");
}
function resolveLearningsFilePath(storePath: string, sessionKey: string): string {
return `${storePath}/${encodeSessionKey(sessionKey)}.learnings.json`;
}
function resolveLegacyLearningsFilePath(storePath: string, sessionKey: string): string {
return `${storePath}/${legacySanitizeSessionKey(sessionKey)}.learnings.json`;
}
async function readLearningsFile(
filePath: string,
): Promise<{ exists: boolean; learnings: string[] }> {
try {
const content = await fs.readFile(filePath, "utf-8");
const parsed = JSON.parse(content);
return { exists: true, learnings: Array.isArray(parsed) ? parsed : [] };
} catch {
return { exists: false, learnings: [] };
}
}
/** Prune expired cooldown entries to prevent unbounded memory growth. */
function pruneExpiredCooldowns(cooldownMs: number): void {
if (lastReflectionBySession.size <= MAX_COOLDOWN_ENTRIES) {
return;
}
const now = Date.now();
for (const [key, time] of lastReflectionBySession) {
if (now - time >= cooldownMs) {
lastReflectionBySession.delete(key);
}
}
}
/** Check if a reflection is allowed (cooldown not active). */
export function isReflectionAllowed(sessionKey: string, cooldownMs?: number): boolean {
const cooldown = cooldownMs ?? DEFAULT_COOLDOWN_MS;
const lastTime = lastReflectionBySession.get(sessionKey);
if (lastTime == null) {
return true;
}
return Date.now() - lastTime >= cooldown;
}
/** Record that a reflection was run for a session. */
export function recordReflectionTime(sessionKey: string, cooldownMs?: number): void {
lastReflectionBySession.set(sessionKey, Date.now());
pruneExpiredCooldowns(cooldownMs ?? DEFAULT_COOLDOWN_MS);
}
/** Clear reflection cooldown tracking (for tests). */
export function clearReflectionCooldowns(): void {
lastReflectionBySession.clear();
}
/** Store a learning derived from feedback reflection in a session companion file. */
export async function storeSessionLearning(params: {
storePath: string;
sessionKey: string;
learning: string;
}): Promise<void> {
const learningsFile = resolveLearningsFilePath(params.storePath, params.sessionKey);
const legacyLearningsFile = resolveLegacyLearningsFilePath(params.storePath, params.sessionKey);
const { exists, learnings: existingLearnings } = await readLearningsFile(learningsFile);
const { learnings: legacyLearnings } =
exists || legacyLearningsFile === learningsFile
? { learnings: [] as string[] }
: await readLearningsFile(legacyLearningsFile);
let learnings = exists ? existingLearnings : legacyLearnings;
learnings.push(params.learning);
if (learnings.length > 10) {
learnings = learnings.slice(-10);
}
await writeJsonFileAtomically(learningsFile, learnings);
if (!exists && legacyLearningsFile !== learningsFile) {
await fs.rm(legacyLearningsFile, { force: true }).catch(() => undefined);
}
}
/** Load session learnings for injection into extraSystemPrompt. */
export async function loadSessionLearnings(
storePath: string,
sessionKey: string,
): Promise<string[]> {
const learningsFile = resolveLearningsFilePath(storePath, sessionKey);
const { exists, learnings } = await readLearningsFile(learningsFile);
if (exists) {
return learnings;
}
return (await readLearningsFile(resolveLegacyLearningsFilePath(storePath, sessionKey))).learnings;
}