import fs from "node:fs/promises" ;
import os from "node:os" ;
import path from "node:path" ;
import { afterAll, beforeAll, describe, expect, it } from "vitest" ;
import { withEnv } from "../test-utils/env.js" ;
import {
buildGroupDisplayName,
deriveSessionKey,
loadSessionStore,
resolveSessionFilePath,
resolveSessionFilePathOptions,
resolveSessionKey,
resolveSessionTranscriptPath,
resolveSessionTranscriptsDir,
updateLastRoute,
updateSessionStore,
updateSessionStoreEntry,
} from "./sessions.js" ;
describe("sessions" , () => {
let fixtureRoot = "" ;
let fixtureCount = 0 ;
const createCaseDir = async (prefix: string) => {
const dir = path.join(fixtureRoot, `${prefix}-${fixtureCount++}`);
await fs.mkdir(dir, { recursive: true });
return dir;
};
beforeAll(async () => {
fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-sessions-suite-" ));
});
afterAll(async () => {
await fs.rm(fixtureRoot, { recursive: true , force: true });
});
const withStateDir = <T>(stateDir: string, fn: () => T): T =>
withEnv({ OPENCLAW_STATE_DIR: stateDir }, fn);
async function createSessionStoreFixture(params: {
prefix: string;
entries: Record<string, Record<string, unknown>>;
}): Promise<{ storePath: string }> {
const dir = await createCaseDir(params.prefix);
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(storePath, JSON.stringify(params.entries), "utf-8" );
return { storePath };
}
function expectedBot1FallbackSessionPath() {
return path.join(
path.resolve("/different/state" ),
"agents" ,
"bot1" ,
"sessions" ,
"sess-1.jsonl" ,
);
}
function buildMainSessionEntry(overrides: Record<string, unknown> = {}) {
return {
sessionId: "sess-1" ,
updatedAt: 123 ,
...overrides,
};
}
async function createAgentSessionsLayout(label: string): Promise<{
stateDir: string;
mainStorePath: string;
bot2SessionPath: string;
outsidePath: string;
}> {
const stateDir = await createCaseDir(label);
const mainSessionsDir = path.join(stateDir, "agents" , "main" , "sessions" );
const bot1SessionsDir = path.join(stateDir, "agents" , "bot1" , "sessions" );
const bot2SessionsDir = path.join(stateDir, "agents" , "bot2" , "sessions" );
await fs.mkdir(mainSessionsDir, { recursive: true });
await fs.mkdir(bot1SessionsDir, { recursive: true });
await fs.mkdir(bot2SessionsDir, { recursive: true });
const mainStorePath = path.join(mainSessionsDir, "sessions.json" );
await fs.writeFile(mainStorePath, "{}" , "utf-8" );
const bot2SessionPath = path.join(bot2SessionsDir, "sess-1.jsonl" );
await fs.writeFile(bot2SessionPath, "{}" , "utf-8" );
const outsidePath = path.join(stateDir, "outside" , "not-a-session.jsonl" );
await fs.mkdir(path.dirname(outsidePath), { recursive: true });
await fs.writeFile(outsidePath, "{}" , "utf-8" );
return { stateDir, mainStorePath, bot2SessionPath, outsidePath };
}
async function normalizePathForComparison(filePath: string): Promise<string> {
const canonicalFile = await fs.realpath(filePath).catch (() => null );
if (canonicalFile) {
return canonicalFile;
}
const parentDir = path.dirname(filePath);
const canonicalParent = await fs.realpath(parentDir).catch (() => parentDir);
return path.join(canonicalParent, path.basename(filePath));
}
const deriveSessionKeyCases = [
{
name: "returns normalized per-sender key" ,
scope: "per-sender" as const ,
ctx: { From: "chat:+1555" },
expected: "+1555" ,
},
{
name: "falls back to unknown when sender missing" ,
scope: "per-sender" as const ,
ctx: {},
expected: "unknown" ,
},
{
name: "global scope returns global" ,
scope: "global" as const ,
ctx: { From: "+1" },
expected: "global" ,
},
{
name: "keeps group chats distinct" ,
scope: "per-sender" as const ,
ctx: { From: "room-123" , ChatType: "group" , Provider: "demo-chat" },
expected: "demo-chat:group:room-123" ,
},
{
name: "prefixes group keys with provider when available" ,
scope: "per-sender" as const ,
ctx: { From: "room-456" , ChatType: "group" , Provider: "demo-chat" },
expected: "demo-chat:group:room-456" ,
},
] as const ;
for (const testCase of deriveSessionKeyCases) {
it(testCase.name, () => {
expect(deriveSessionKey(testCase.scope, testCase.ctx)).toBe(testCase.expected);
});
}
it("builds discord display name with guild+channel slugs" , () => {
expect(
buildGroupDisplayName({
provider: "discord" ,
groupChannel: "#general" ,
space: "friends-of-openclaw" ,
id: "123" ,
key: "discord:group:123" ,
}),
).toBe("discord:friends-of-openclaw#general" );
});
const resolveSessionKeyCases = [
{
name: "keeps explicit provider when provided in group key" ,
scope: "per-sender" as const ,
ctx: { From: "discord:group:12345" , ChatType: "group" },
mainKey: "main" ,
expected: "agent:main:discord:group:12345" ,
},
{
name: "collapses direct chats to main by default" ,
scope: "per-sender" as const ,
ctx: { From: "+1555" },
mainKey: undefined,
expected: "agent:main:main" ,
},
{
name: "collapses direct chats to main even when sender missing" ,
scope: "per-sender" as const ,
ctx: {},
mainKey: undefined,
expected: "agent:main:main" ,
},
{
name: "maps direct chats to main key when provided" ,
scope: "per-sender" as const ,
ctx: { From: "chat:+1555" },
mainKey: "main" ,
expected: "agent:main:main" ,
},
{
name: "uses custom main key when provided" ,
scope: "per-sender" as const ,
ctx: { From: "+1555" },
mainKey: "primary" ,
expected: "agent:main:primary" ,
},
{
name: "keeps global scope untouched" ,
scope: "global" as const ,
ctx: { From: "+1555" },
mainKey: undefined,
expected: "global" ,
},
{
name: "leaves groups untouched even with main key" ,
scope: "per-sender" as const ,
ctx: { From: "room-123" , ChatType: "group" , Provider: "demo-chat" },
mainKey: "main" ,
expected: "agent:main:demo-chat:group:room-123" ,
},
] as const ;
for (const testCase of resolveSessionKeyCases) {
it(testCase.name, () => {
expect(resolveSessionKey(testCase.scope, testCase.ctx, testCase.mainKey)).toBe(
testCase.expected,
);
});
}
it("updateLastRoute persists channel and target" , async () => {
const mainSessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateLastRoute" ,
entries: {
[mainSessionKey]: buildMainSessionEntry({
systemSent: true ,
thinkingLevel: "low" ,
responseUsage: "on" ,
queueDebounceMs: 1234 ,
reasoningLevel: "on" ,
elevatedLevel: "on" ,
authProfileOverride: "auth-1" ,
compactionCount: 2 ,
}),
},
});
await updateLastRoute({
storePath,
sessionKey: mainSessionKey,
deliveryContext: {
channel: "telegram" ,
to: " 12345 " ,
},
});
const store = loadSessionStore(storePath);
expect(store[mainSessionKey]?.sessionId).toBe("sess-1" );
// updateLastRoute must preserve existing updatedAt (activity timestamp)
expect(store[mainSessionKey]?.updatedAt).toBe(123 );
expect(store[mainSessionKey]?.lastChannel).toBe("telegram" );
expect(store[mainSessionKey]?.lastTo).toBe("12345" );
expect(store[mainSessionKey]?.deliveryContext).toEqual({
channel: "telegram" ,
to: "12345" ,
});
expect(store[mainSessionKey]?.responseUsage).toBe("on" );
expect(store[mainSessionKey]?.queueDebounceMs).toBe(1234 );
expect(store[mainSessionKey]?.reasoningLevel).toBe("on" );
expect(store[mainSessionKey]?.elevatedLevel).toBe("on" );
expect(store[mainSessionKey]?.authProfileOverride).toBe("auth-1" );
expect(store[mainSessionKey]?.compactionCount).toBe(2 );
});
it("updateLastRoute prefers explicit deliveryContext" , async () => {
const mainSessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateLastRoute" ,
entries: {},
});
await updateLastRoute({
storePath,
sessionKey: mainSessionKey,
channel: "demo-chat" ,
to: "111" ,
accountId: "legacy" ,
deliveryContext: {
channel: "telegram" ,
to: "222" ,
accountId: "primary" ,
},
});
const store = loadSessionStore(storePath);
expect(store[mainSessionKey]?.lastChannel).toBe("telegram" );
expect(store[mainSessionKey]?.lastTo).toBe("222" );
expect(store[mainSessionKey]?.lastAccountId).toBe("primary" );
expect(store[mainSessionKey]?.deliveryContext).toEqual({
channel: "telegram" ,
to: "222" ,
accountId: "primary" ,
});
});
it("updateLastRoute clears threadId when explicit route omits threadId" , async () => {
const mainSessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateLastRoute" ,
entries: {
[mainSessionKey]: buildMainSessionEntry({
deliveryContext: {
channel: "telegram" ,
to: "222" ,
threadId: "42" ,
},
lastChannel: "telegram" ,
lastTo: "222" ,
lastThreadId: "42" ,
}),
},
});
await updateLastRoute({
storePath,
sessionKey: mainSessionKey,
deliveryContext: {
channel: "telegram" ,
to: "222" ,
},
});
const store = loadSessionStore(storePath);
expect(store[mainSessionKey]?.deliveryContext).toEqual({
channel: "telegram" ,
to: "222" ,
});
expect(store[mainSessionKey]?.lastThreadId).toBeUndefined();
});
it("updateLastRoute records origin + group metadata when ctx is provided" , async () => {
const sessionKey = "agent:main:demo-chat:group:room-123" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateLastRoute" ,
entries: {},
});
await updateLastRoute({
storePath,
sessionKey,
deliveryContext: {
channel: "demo-chat" ,
to: "room-123" ,
},
ctx: {
Provider: "demo-chat" ,
ChatType: "group" ,
GroupSubject: "Family" ,
From: "room-123" ,
},
});
const store = loadSessionStore(storePath);
expect(store[sessionKey]?.subject).toBe("Family" );
expect(store[sessionKey]?.channel).toBe("demo-chat" );
expect(store[sessionKey]?.groupId).toBe("room-123" );
expect(store[sessionKey]?.origin?.label).toBe("Family" );
expect(store[sessionKey]?.origin?.provider).toBe("demo-chat" );
expect(store[sessionKey]?.origin?.chatType).toBe("group" );
});
it("updateLastRoute does not bump updatedAt on existing sessions (#49515)" , async () => {
const mainSessionKey = "agent:main:main" ;
const frozenUpdatedAt = 1000 ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateLastRoute-preserve-activity" ,
entries: {
[mainSessionKey]: buildMainSessionEntry({
updatedAt: frozenUpdatedAt,
}),
},
});
await updateLastRoute({
storePath,
sessionKey: mainSessionKey,
deliveryContext: {
channel: "telegram" ,
to: "99999" ,
},
});
const store = loadSessionStore(storePath);
// Route updates must not refresh activity timestamps; idle/daily reset
// evaluation relies on updatedAt from actual session turns.
expect(store[mainSessionKey]?.updatedAt).toBe(frozenUpdatedAt);
// Routing fields should still be updated
expect(store[mainSessionKey]?.lastChannel).toBe("telegram" );
expect(store[mainSessionKey]?.lastTo).toBe("99999" );
});
it("updateSessionStoreEntry preserves existing fields when patching" , async () => {
const sessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateSessionStoreEntry" ,
entries: {
[sessionKey]: {
sessionId: "sess-1" ,
updatedAt: 100 ,
reasoningLevel: "on" ,
},
},
});
await updateSessionStoreEntry({
storePath,
sessionKey,
update: async () => ({ updatedAt: 200 }),
});
const store = loadSessionStore(storePath);
expect(store[sessionKey]?.updatedAt).toBeGreaterThanOrEqual(200 );
expect(store[sessionKey]?.reasoningLevel).toBe("on" );
});
it("updateSessionStoreEntry returns null when session key does not exist" , async () => {
const { storePath } = await createSessionStoreFixture({
prefix: "updateSessionStoreEntry-missing" ,
entries: {},
});
const update = async () => ({ thinkingLevel: "high" as const });
const result = await updateSessionStoreEntry({
storePath,
sessionKey: "agent:main:missing" ,
update,
});
expect(result).toBeNull();
});
it("updateSessionStoreEntry keeps existing entry when patch callback returns null" , async () => {
const sessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateSessionStoreEntry-noop" ,
entries: {
[sessionKey]: {
sessionId: "sess-1" ,
updatedAt: 123 ,
thinkingLevel: "low" ,
},
},
});
const result = await updateSessionStoreEntry({
storePath,
sessionKey,
update: async () => null ,
});
expect(result).toEqual(expect.objectContaining({ sessionId: "sess-1" , thinkingLevel: "low" }));
const store = loadSessionStore(storePath);
expect(store[sessionKey]?.thinkingLevel).toBe("low" );
});
it("updateSessionStore preserves concurrent additions" , async () => {
const dir = await createCaseDir("updateSessionStore" );
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(storePath, "{}" , "utf-8" );
await Promise.all([
updateSessionStore(storePath, (store) => {
store["agent:main:one" ] = { sessionId: "sess-1" , updatedAt: Date.now() };
}),
updateSessionStore(storePath, (store) => {
store["agent:main:two" ] = { sessionId: "sess-2" , updatedAt: Date.now() };
}),
]);
const store = loadSessionStore(storePath);
expect(store["agent:main:one" ]?.sessionId).toBe("sess-1" );
expect(store["agent:main:two" ]?.sessionId).toBe("sess-2" );
});
it("recovers from array-backed session stores" , async () => {
const dir = await createCaseDir("updateSessionStore" );
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(storePath, "[]" , "utf-8" );
await updateSessionStore(storePath, (store) => {
store["agent:main:main" ] = { sessionId: "sess-1" , updatedAt: Date.now() };
});
const store = loadSessionStore(storePath);
expect(store["agent:main:main" ]?.sessionId).toBe("sess-1" );
const raw = await fs.readFile(storePath, "utf-8" );
expect(raw.trim().startsWith("{" )).toBe(true );
});
it("normalizes last route fields on write" , async () => {
const dir = await createCaseDir("updateSessionStore" );
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(storePath, "{}" , "utf-8" );
await updateSessionStore(storePath, (store) => {
store["agent:main:main" ] = {
sessionId: "sess-normalized" ,
updatedAt: Date.now(),
lastChannel: " Demo Chat " ,
lastTo: " +1555 " ,
lastAccountId: " acct-1 " ,
};
});
const store = loadSessionStore(storePath);
expect(store["agent:main:main" ]?.lastChannel).toBe("demo chat" );
expect(store["agent:main:main" ]?.lastTo).toBe("+1555" );
expect(store["agent:main:main" ]?.lastAccountId).toBe("acct-1" );
expect(store["agent:main:main" ]?.deliveryContext).toEqual({
channel: "demo chat" ,
to: "+1555" ,
accountId: "acct-1" ,
});
});
it("updateSessionStore keeps deletions when concurrent writes happen" , async () => {
const dir = await createCaseDir("updateSessionStore" );
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(
storePath,
JSON.stringify(
{
"agent:main:old" : { sessionId: "sess-old" , updatedAt: Date.now() },
"agent:main:keep" : { sessionId: "sess-keep" , updatedAt: Date.now() },
},
null ,
2 ,
),
"utf-8" ,
);
await Promise.all([
updateSessionStore(storePath, (store) => {
delete store["agent:main:old" ];
}),
updateSessionStore(storePath, (store) => {
store["agent:main:new" ] = { sessionId: "sess-new" , updatedAt: Date.now() };
}),
]);
const store = loadSessionStore(storePath);
expect(store["agent:main:old" ]).toBeUndefined();
expect(store["agent:main:keep" ]?.sessionId).toBe("sess-keep" );
expect(store["agent:main:new" ]?.sessionId).toBe("sess-new" );
});
it("loadSessionStore auto-migrates legacy provider keys to channel keys" , async () => {
const mainSessionKey = "agent:main:main" ;
const dir = await createCaseDir("loadSessionStore" );
const storePath = path.join(dir, "sessions.json" );
await fs.writeFile(
storePath,
JSON.stringify(
{
[mainSessionKey]: {
sessionId: "sess-legacy" ,
updatedAt: 123 ,
provider: "slack" ,
lastProvider: "telegram" ,
lastTo: "user:U123" ,
},
},
null ,
2 ,
),
"utf-8" ,
);
const store = loadSessionStore(storePath) as unknown as Record<string, Record<string, unknown>>;
const entry = store[mainSessionKey] ?? {};
expect(entry.channel).toBe("slack" );
expect(entry.provider).toBeUndefined();
expect(entry.lastChannel).toBe("telegram" );
expect(entry.lastProvider).toBeUndefined();
});
it("derives session transcripts dir from OPENCLAW_STATE_DIR" , () => {
const dir = resolveSessionTranscriptsDir(
{ OPENCLAW_STATE_DIR: "/custom/state" } as NodeJS.ProcessEnv,
() => "/home/ignored" ,
);
expect(dir).toBe(path.join(path.resolve("/custom/state" ), "agents" , "main" , "sessions" ));
});
it("includes topic ids in session transcript filenames" , () => {
withStateDir("/custom/state" , () => {
const sessionFile = resolveSessionTranscriptPath("sess-1" , "main" , 123 );
expect(sessionFile).toBe(
path.join(
path.resolve("/custom/state" ),
"agents" ,
"main" ,
"sessions" ,
"sess-1-topic-123.jsonl" ,
),
);
});
});
it("uses agent id when resolving session file fallback paths" , () => {
withStateDir("/custom/state" , () => {
const sessionFile = resolveSessionFilePath("sess-2" , undefined, {
agentId: "codex" ,
});
expect(sessionFile).toBe(
path.join(path.resolve("/custom/state" ), "agents" , "codex" , "sessions" , "sess-2.jsonl" ),
);
});
});
it("resolves cross-agent absolute sessionFile paths" , async () => {
const { stateDir, bot2SessionPath } = await createAgentSessionsLayout("cross-agent" );
const sessionFile = withStateDir(stateDir, () =>
// Agent bot1 resolves a sessionFile that belongs to agent bot2
resolveSessionFilePath("sess-1" , { sessionFile: bot2SessionPath }, { agentId: "bot1" }),
);
expect(await normalizePathForComparison(sessionFile)).toBe(
await normalizePathForComparison(bot2SessionPath),
);
});
it("resolves cross-agent paths when OPENCLAW_STATE_DIR differs from stored paths" , () => {
withStateDir(path.resolve("/different/state" ), () => {
const originalBase = path.resolve("/original/state" );
const bot2Session = path.join(originalBase, "agents" , "bot2" , "sessions" , "sess-1.jsonl" );
// sessionFile was created under a different state dir than current env
const sessionFile = resolveSessionFilePath(
"sess-1" ,
{ sessionFile: bot2Session },
{ agentId: "bot1" },
);
expect(sessionFile).toBe(bot2Session);
});
});
it("falls back when structural cross-root path traverses after sessions" , () => {
withStateDir(path.resolve("/different/state" ), () => {
const originalBase = path.resolve("/original/state" );
const unsafe = path.join(originalBase, "agents" , "bot2" , "sessions" , ".." , ".." , "etc" );
const sessionFile = resolveSessionFilePath(
"sess-1" ,
{ sessionFile: path.join(unsafe, "passwd" ) },
{ agentId: "bot1" },
);
expect(sessionFile).toBe(expectedBot1FallbackSessionPath());
});
});
it("falls back when structural cross-root path nests under sessions" , () => {
withStateDir(path.resolve("/different/state" ), () => {
const originalBase = path.resolve("/original/state" );
const nested = path.join(
originalBase,
"agents" ,
"bot2" ,
"sessions" ,
"nested" ,
"sess-1.jsonl" ,
);
const sessionFile = resolveSessionFilePath(
"sess-1" ,
{ sessionFile: nested },
{ agentId: "bot1" },
);
expect(sessionFile).toBe(expectedBot1FallbackSessionPath());
});
});
it("resolveSessionFilePathOptions keeps explicit agentId alongside absolute store path" , () => {
const storePath = "/tmp/openclaw/agents/main/sessions/sessions.json" ;
const resolved = resolveSessionFilePathOptions({
agentId: "bot2" ,
storePath,
});
expect(resolved?.agentId).toBe("bot2" );
expect(resolved?.sessionsDir).toBe(path.dirname(path.resolve(storePath)));
});
it("resolves sibling agent absolute sessionFile using alternate agentId from options" , async () => {
const { stateDir, mainStorePath, bot2SessionPath } =
await createAgentSessionsLayout("sibling-agent" );
const sessionFile = withStateDir(stateDir, () => {
const opts = resolveSessionFilePathOptions({
agentId: "bot2" ,
storePath: mainStorePath,
});
return resolveSessionFilePath("sess-1" , { sessionFile: bot2SessionPath }, opts);
});
expect(await normalizePathForComparison(sessionFile)).toBe(
await normalizePathForComparison(bot2SessionPath),
);
});
it("falls back to derived transcript path when sessionFile is outside agent sessions directories" , async () => {
const { stateDir, outsidePath } = await createAgentSessionsLayout("outside-fallback" );
const sessionFile = withStateDir(stateDir, () =>
resolveSessionFilePath("sess-1" , { sessionFile: outsidePath }, { agentId: "bot1" }),
);
const expectedPath = path.join(stateDir, "agents" , "bot1" , "sessions" , "sess-1.jsonl" );
expect(await normalizePathForComparison(sessionFile)).toBe(
await normalizePathForComparison(expectedPath),
);
});
it("updateSessionStoreEntry merges concurrent patches" , async () => {
const mainSessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateSessionStoreEntry" ,
entries: {
[mainSessionKey]: {
sessionId: "sess-1" ,
updatedAt: 123 ,
thinkingLevel: "low" ,
},
},
});
const createDeferred = <T>() => {
let resolve!: (value: T | PromiseLike<T>) => void ;
let reject!: (reason?: unknown) => void ;
const promise = new Promise<T>((res, rej) => {
resolve = res;
reject = rej;
});
return { promise, resolve, reject };
};
const firstStarted = createDeferred<void >();
const releaseFirst = createDeferred<void >();
const p1 = updateSessionStoreEntry({
storePath,
sessionKey: mainSessionKey,
update: async () => {
firstStarted.resolve();
await releaseFirst.promise;
return { modelOverride: "anthropic/claude-opus-4-6" };
},
});
const p2 = updateSessionStoreEntry({
storePath,
sessionKey: mainSessionKey,
update: async () => {
await firstStarted.promise;
return { thinkingLevel: "high" };
},
});
await firstStarted.promise;
releaseFirst.resolve();
await Promise.all([p1, p2]);
const store = loadSessionStore(storePath);
expect(store[mainSessionKey]?.modelOverride).toBe("anthropic/claude-opus-4-6" );
expect(store[mainSessionKey]?.thinkingLevel).toBe("high" );
await expect(fs.stat(`${storePath}.lock`)).rejects.toThrow();
});
it("updateSessionStoreEntry re-reads disk inside lock instead of using stale cache" , async () => {
const mainSessionKey = "agent:main:main" ;
const { storePath } = await createSessionStoreFixture({
prefix: "updateSessionStoreEntry-cache-bypass" ,
entries: {
[mainSessionKey]: {
sessionId: "sess-1" ,
updatedAt: 123 ,
thinkingLevel: "low" ,
},
},
});
// Prime the in-process cache with the original entry.
expect(loadSessionStore(storePath)[mainSessionKey]?.thinkingLevel).toBe("low" );
const originalStat = await fs.stat(storePath);
// Simulate an external writer that updates the store but preserves mtime.
const externalStore = JSON.parse(await fs.readFile(storePath, "utf-8" )) as Record<
string,
Record<string, unknown>
>;
externalStore[mainSessionKey] = {
...externalStore[mainSessionKey],
providerOverride: "anthropic" ,
updatedAt: 124 ,
};
await fs.writeFile(storePath, JSON.stringify(externalStore), "utf-8" );
await fs.utimes(storePath, originalStat.atime, originalStat.mtime);
await updateSessionStoreEntry({
storePath,
sessionKey: mainSessionKey,
update: async () => ({ thinkingLevel: "high" }),
});
const store = loadSessionStore(storePath);
expect(store[mainSessionKey]?.providerOverride).toBe("anthropic" );
expect(store[mainSessionKey]?.thinkingLevel).toBe("high" );
});
});
Messung V0.5 in Prozent C=99 H=98 G=98
¤ Dauer der Verarbeitung: 0.20 Sekunden
(vorverarbeitet am 2026-06-07)
¤
*© Formatika GbR, Deutschland