mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-17 12:11:20 +00:00
Merged via squash.
Prepared head SHA: b3ef62b973
Co-authored-by: jalehman <550978+jalehman@users.noreply.github.com>
Co-authored-by: jalehman <550978+jalehman@users.noreply.github.com>
Reviewed-by: @jalehman
111 lines
3.8 KiB
TypeScript
111 lines
3.8 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
|
import type { OpenClawConfig } from "../../config/config.js";
|
|
import type { SessionEntry } from "../../config/sessions.js";
|
|
import type { HookRunner } from "../../plugins/hooks.js";
|
|
|
|
const hookRunnerMocks = vi.hoisted(() => ({
|
|
hasHooks: vi.fn<HookRunner["hasHooks"]>(),
|
|
runSessionEnd: vi.fn<HookRunner["runSessionEnd"]>(),
|
|
runSessionStart: vi.fn<HookRunner["runSessionStart"]>(),
|
|
}));
|
|
|
|
let incrementCompactionCount: typeof import("./session-updates.js").incrementCompactionCount;
|
|
const tempDirs: string[] = [];
|
|
|
|
async function createFixture() {
|
|
const root = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-session-updates-"));
|
|
tempDirs.push(root);
|
|
const storePath = path.join(root, "sessions.json");
|
|
const sessionKey = "agent:main:telegram:direct:compaction";
|
|
const transcriptPath = path.join(root, "s1.jsonl");
|
|
await fs.writeFile(transcriptPath, '{"type":"message"}\n', "utf-8");
|
|
const entry = {
|
|
sessionId: "s1",
|
|
sessionFile: transcriptPath,
|
|
updatedAt: Date.now(),
|
|
compactionCount: 0,
|
|
} as SessionEntry;
|
|
const sessionStore: Record<string, SessionEntry> = {
|
|
[sessionKey]: entry,
|
|
};
|
|
await fs.writeFile(storePath, JSON.stringify(sessionStore, null, 2), "utf-8");
|
|
return { storePath, sessionKey, sessionStore, entry, transcriptPath };
|
|
}
|
|
|
|
describe("session-updates lifecycle hooks", () => {
|
|
beforeEach(async () => {
|
|
vi.resetModules();
|
|
vi.doMock("../../plugins/hook-runner-global.js", () => ({
|
|
getGlobalHookRunner: () =>
|
|
({
|
|
hasHooks: hookRunnerMocks.hasHooks,
|
|
runSessionEnd: hookRunnerMocks.runSessionEnd,
|
|
runSessionStart: hookRunnerMocks.runSessionStart,
|
|
}) as unknown as HookRunner,
|
|
}));
|
|
hookRunnerMocks.hasHooks.mockReset();
|
|
hookRunnerMocks.runSessionEnd.mockReset();
|
|
hookRunnerMocks.runSessionStart.mockReset();
|
|
hookRunnerMocks.hasHooks.mockImplementation(
|
|
(hookName) => hookName === "session_end" || hookName === "session_start",
|
|
);
|
|
hookRunnerMocks.runSessionEnd.mockResolvedValue(undefined);
|
|
hookRunnerMocks.runSessionStart.mockResolvedValue(undefined);
|
|
({ incrementCompactionCount } = await import("./session-updates.js"));
|
|
});
|
|
|
|
afterEach(async () => {
|
|
vi.restoreAllMocks();
|
|
await Promise.all(
|
|
tempDirs.splice(0).map((dir) => fs.rm(dir, { recursive: true, force: true })),
|
|
);
|
|
});
|
|
|
|
it("emits compaction lifecycle hooks when newSessionId replaces the session", async () => {
|
|
const { storePath, sessionKey, sessionStore, entry, transcriptPath } = await createFixture();
|
|
const cfg = { session: { store: storePath } } as OpenClawConfig;
|
|
|
|
await incrementCompactionCount({
|
|
cfg,
|
|
sessionEntry: entry,
|
|
sessionStore,
|
|
sessionKey,
|
|
storePath,
|
|
newSessionId: "s2",
|
|
});
|
|
|
|
expect(hookRunnerMocks.runSessionEnd).toHaveBeenCalledTimes(1);
|
|
expect(hookRunnerMocks.runSessionStart).toHaveBeenCalledTimes(1);
|
|
|
|
const [endEvent, endContext] = hookRunnerMocks.runSessionEnd.mock.calls[0] ?? [];
|
|
const [startEvent, startContext] = hookRunnerMocks.runSessionStart.mock.calls[0] ?? [];
|
|
|
|
expect(endEvent).toMatchObject({
|
|
sessionId: "s1",
|
|
sessionKey,
|
|
reason: "compaction",
|
|
transcriptArchived: false,
|
|
});
|
|
expect(endEvent?.sessionFile).toBe(await fs.realpath(transcriptPath));
|
|
expect(endContext).toMatchObject({
|
|
sessionId: "s1",
|
|
sessionKey,
|
|
agentId: "main",
|
|
});
|
|
expect(endEvent?.nextSessionId).toBe(startEvent?.sessionId);
|
|
expect(startEvent).toMatchObject({
|
|
sessionId: "s2",
|
|
sessionKey,
|
|
resumedFrom: "s1",
|
|
});
|
|
expect(startContext).toMatchObject({
|
|
sessionId: "s2",
|
|
sessionKey,
|
|
agentId: "main",
|
|
});
|
|
});
|
|
});
|