mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-12 07:20:45 +00:00
Merged via /review-pr -> /prepare-pr -> /merge-pr.
Prepared head SHA: 7533b85156
Co-authored-by: gumadeiras <5599352+gumadeiras@users.noreply.github.com>
Co-authored-by: shakkernerd <165377636+shakkernerd@users.noreply.github.com>
Reviewed-by: @shakkernerd
157 lines
5.1 KiB
TypeScript
157 lines
5.1 KiB
TypeScript
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
import type { RuntimeEnv } from "../runtime.js";
|
|
|
|
const loadConfigMock = vi.hoisted(() =>
|
|
vi.fn(() => ({
|
|
agents: {
|
|
defaults: {
|
|
model: { primary: "pi:opus" },
|
|
models: { "pi:opus": {} },
|
|
contextTokens: 32000,
|
|
},
|
|
list: [
|
|
{ id: "main", default: false },
|
|
{ id: "voice", default: true },
|
|
],
|
|
},
|
|
session: {
|
|
store: "/tmp/sessions-{agentId}.json",
|
|
},
|
|
})),
|
|
);
|
|
|
|
const resolveStorePathMock = vi.hoisted(() =>
|
|
vi.fn((_store: string | undefined, opts?: { agentId?: string }) => {
|
|
return `/tmp/sessions-${opts?.agentId ?? "missing"}.json`;
|
|
}),
|
|
);
|
|
const loadSessionStoreMock = vi.hoisted(() => vi.fn(() => ({})));
|
|
|
|
vi.mock("../config/config.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../config/config.js")>();
|
|
return {
|
|
...actual,
|
|
loadConfig: loadConfigMock,
|
|
};
|
|
});
|
|
|
|
vi.mock("../config/sessions.js", async (importOriginal) => {
|
|
const actual = await importOriginal<typeof import("../config/sessions.js")>();
|
|
return {
|
|
...actual,
|
|
resolveStorePath: resolveStorePathMock,
|
|
loadSessionStore: loadSessionStoreMock,
|
|
};
|
|
});
|
|
|
|
import { sessionsCommand } from "./sessions.js";
|
|
|
|
function createRuntime(): { runtime: RuntimeEnv; logs: string[] } {
|
|
const logs: string[] = [];
|
|
return {
|
|
runtime: {
|
|
log: (msg: unknown) => logs.push(String(msg)),
|
|
error: vi.fn(),
|
|
exit: vi.fn(),
|
|
},
|
|
logs,
|
|
};
|
|
}
|
|
|
|
describe("sessionsCommand default store agent selection", () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
resolveStorePathMock.mockImplementation(
|
|
(_store: string | undefined, opts?: { agentId?: string }) => {
|
|
return `/tmp/sessions-${opts?.agentId ?? "missing"}.json`;
|
|
},
|
|
);
|
|
loadSessionStoreMock.mockImplementation(() => ({}));
|
|
});
|
|
|
|
it("includes agentId on sessions rows for --all-agents JSON output", async () => {
|
|
resolveStorePathMock.mockClear();
|
|
loadSessionStoreMock.mockReset();
|
|
loadSessionStoreMock
|
|
.mockReturnValueOnce({
|
|
main_row: { sessionId: "s1", updatedAt: Date.now() - 60_000, model: "pi:opus" },
|
|
})
|
|
.mockReturnValueOnce({
|
|
voice_row: { sessionId: "s2", updatedAt: Date.now() - 120_000, model: "pi:opus" },
|
|
});
|
|
const { runtime, logs } = createRuntime();
|
|
|
|
await sessionsCommand({ allAgents: true, json: true }, runtime);
|
|
|
|
const payload = JSON.parse(logs[0] ?? "{}") as {
|
|
allAgents?: boolean;
|
|
sessions?: Array<{ key: string; agentId?: string }>;
|
|
};
|
|
expect(payload.allAgents).toBe(true);
|
|
expect(payload.sessions?.map((session) => session.agentId)).toContain("main");
|
|
expect(payload.sessions?.map((session) => session.agentId)).toContain("voice");
|
|
});
|
|
|
|
it("avoids duplicate rows when --all-agents resolves to a shared store path", async () => {
|
|
resolveStorePathMock.mockReset();
|
|
resolveStorePathMock.mockReturnValue("/tmp/shared-sessions.json");
|
|
loadSessionStoreMock.mockReset();
|
|
loadSessionStoreMock.mockReturnValue({
|
|
"agent:main:room": { sessionId: "s1", updatedAt: Date.now() - 60_000, model: "pi:opus" },
|
|
"agent:voice:room": { sessionId: "s2", updatedAt: Date.now() - 30_000, model: "pi:opus" },
|
|
});
|
|
const { runtime, logs } = createRuntime();
|
|
|
|
await sessionsCommand({ allAgents: true, json: true }, runtime);
|
|
|
|
const payload = JSON.parse(logs[0] ?? "{}") as {
|
|
count?: number;
|
|
stores?: Array<{ agentId: string; path: string }>;
|
|
allAgents?: boolean;
|
|
sessions?: Array<{ key: string; agentId?: string }>;
|
|
};
|
|
expect(payload.count).toBe(2);
|
|
expect(payload.allAgents).toBe(true);
|
|
expect(payload.stores).toEqual([{ agentId: "main", path: "/tmp/shared-sessions.json" }]);
|
|
expect(payload.sessions?.map((session) => session.agentId).toSorted()).toEqual([
|
|
"main",
|
|
"voice",
|
|
]);
|
|
expect(loadSessionStoreMock).toHaveBeenCalledTimes(1);
|
|
});
|
|
|
|
it("uses configured default agent id when resolving implicit session store path", async () => {
|
|
resolveStorePathMock.mockClear();
|
|
const { runtime, logs } = createRuntime();
|
|
|
|
await sessionsCommand({}, runtime);
|
|
|
|
expect(resolveStorePathMock).toHaveBeenCalledWith("/tmp/sessions-{agentId}.json", {
|
|
agentId: "voice",
|
|
});
|
|
expect(logs[0]).toContain("Session store: /tmp/sessions-voice.json");
|
|
});
|
|
|
|
it("uses all configured agent stores with --all-agents", async () => {
|
|
resolveStorePathMock.mockClear();
|
|
loadSessionStoreMock.mockReset();
|
|
loadSessionStoreMock
|
|
.mockReturnValueOnce({
|
|
main_row: { sessionId: "s1", updatedAt: Date.now() - 60_000, model: "pi:opus" },
|
|
})
|
|
.mockReturnValueOnce({});
|
|
const { runtime, logs } = createRuntime();
|
|
|
|
await sessionsCommand({ allAgents: true }, runtime);
|
|
|
|
expect(resolveStorePathMock).toHaveBeenCalledWith("/tmp/sessions-{agentId}.json", {
|
|
agentId: "main",
|
|
});
|
|
expect(resolveStorePathMock).toHaveBeenCalledWith("/tmp/sessions-{agentId}.json", {
|
|
agentId: "voice",
|
|
});
|
|
expect(logs[0]).toContain("Session stores: 2 (main, voice)");
|
|
expect(logs[2]).toContain("Agent");
|
|
});
|
|
});
|