openclaw/src/auto-reply/status.test.ts

180 lines
5.1 KiB
TypeScript
Raw Normal View History

import fs from "node:fs";
import os from "node:os";
import path from "node:path";
2025-12-07 16:53:19 +00:00
import { afterEach, describe, expect, it, vi } from "vitest";
import { buildStatusMessage } from "./status.js";
afterEach(() => {
vi.restoreAllMocks();
});
describe("buildStatusMessage", () => {
it("summarizes agent readiness and context usage", () => {
const text = buildStatusMessage({
2025-12-26 00:43:44 +01:00
agent: {
model: "anthropic/pi:opus",
contextTokens: 32_000,
},
2025-12-07 16:53:19 +00:00
sessionEntry: {
sessionId: "abc",
updatedAt: 0,
totalTokens: 16_000,
contextTokens: 32_000,
thinkingLevel: "low",
verboseLevel: "on",
compactionCount: 2,
2025-12-07 16:53:19 +00:00
},
sessionKey: "agent:main:main",
2025-12-07 16:53:19 +00:00
sessionScope: "per-sender",
resolvedThink: "medium",
resolvedVerbose: "off",
2026-01-07 07:21:56 +01:00
queue: { mode: "collect", depth: 0 },
2025-12-07 16:53:19 +00:00
now: 10 * 60_000, // 10 minutes later
});
2026-01-07 07:21:56 +01:00
expect(text).toContain("🦞 ClawdBot");
expect(text).toContain("🧠 Model:");
2026-01-05 07:07:17 +01:00
expect(text).toContain("Runtime: direct");
2025-12-07 16:53:19 +00:00
expect(text).toContain("Context: 16k/32k (50%)");
2026-01-07 07:21:56 +01:00
expect(text).toContain("🧹 Compactions: 2");
expect(text).toContain("Session: agent:main:main");
2026-01-07 07:21:56 +01:00
expect(text).toContain("updated 10m ago");
expect(text).toContain("Think: medium");
expect(text).toContain("Verbose: off");
expect(text).toContain("Elevated: on");
expect(text).toContain("Queue: collect");
2025-12-07 16:53:19 +00:00
});
2025-12-17 11:29:04 +01:00
it("handles missing agent config gracefully", () => {
2025-12-07 16:53:19 +00:00
const text = buildStatusMessage({
2025-12-17 11:29:04 +01:00
agent: {},
2025-12-07 16:53:19 +00:00
sessionScope: "per-sender",
webLinked: false,
});
2026-01-07 07:21:56 +01:00
expect(text).toContain("🧠 Model:");
2025-12-07 16:53:19 +00:00
expect(text).toContain("Context:");
2026-01-07 07:21:56 +01:00
expect(text).toContain("Queue:");
2025-12-07 16:53:19 +00:00
});
2025-12-22 20:36:29 +01:00
it("includes group activation for group sessions", () => {
const text = buildStatusMessage({
agent: {},
sessionEntry: {
sessionId: "g1",
updatedAt: 0,
groupActivation: "always",
2026-01-02 10:14:58 +01:00
chatType: "group",
2025-12-22 20:36:29 +01:00
},
sessionKey: "agent:main:whatsapp:group:123@g.us",
2025-12-22 20:36:29 +01:00
sessionScope: "per-sender",
2026-01-07 07:21:56 +01:00
queue: { mode: "collect", depth: 0 },
2025-12-22 20:36:29 +01:00
});
2026-01-07 07:21:56 +01:00
expect(text).toContain("Activation: always");
});
it("shows queue details when overridden", () => {
const text = buildStatusMessage({
agent: {},
sessionEntry: { sessionId: "q1", updatedAt: 0 },
sessionKey: "agent:main:main",
sessionScope: "per-sender",
queue: {
mode: "collect",
depth: 3,
debounceMs: 2000,
cap: 5,
dropPolicy: "old",
showDetails: true,
},
});
expect(text).toContain(
"Queue: collect (depth 3 · debounce 2s · cap 5 · drop old)",
);
2025-12-22 20:36:29 +01:00
});
2026-01-07 11:42:41 +01:00
it("inserts usage summary beneath context line", () => {
const text = buildStatusMessage({
agent: { model: "anthropic/claude-opus-4-5", contextTokens: 32_000 },
sessionEntry: { sessionId: "u1", updatedAt: 0, totalTokens: 1000 },
sessionKey: "agent:main:main",
sessionScope: "per-sender",
queue: { mode: "collect", depth: 0 },
usageLine: "📊 Usage: Claude 80% left (5h)",
});
const lines = text.split("\n");
const contextIndex = lines.findIndex((line) => line.startsWith("📚 "));
expect(contextIndex).toBeGreaterThan(-1);
expect(lines[contextIndex + 1]).toBe("📊 Usage: Claude 80% left (5h)");
});
2025-12-17 11:29:04 +01:00
it("prefers cached prompt tokens from the session log", async () => {
2026-01-04 14:32:47 +00:00
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "clawdbot-status-"));
2025-12-17 11:29:04 +01:00
const previousHome = process.env.HOME;
process.env.HOME = dir;
try {
vi.resetModules();
const { buildStatusMessage: buildStatusMessageDynamic } = await import(
"./status.js"
);
2025-12-17 11:29:04 +01:00
const sessionId = "sess-1";
const logPath = path.join(
dir,
2026-01-04 14:32:47 +00:00
".clawdbot",
"agents",
"main",
2025-12-17 11:29:04 +01:00
"sessions",
`${sessionId}.jsonl`,
);
fs.mkdirSync(path.dirname(logPath), { recursive: true });
2025-12-17 11:29:04 +01:00
fs.writeFileSync(
logPath,
[
JSON.stringify({
type: "message",
message: {
role: "assistant",
model: "claude-opus-4-5",
usage: {
input: 1,
output: 2,
cacheRead: 1000,
cacheWrite: 0,
totalTokens: 1003,
},
},
}),
].join("\n"),
"utf-8",
);
2025-12-17 11:29:04 +01:00
const text = buildStatusMessageDynamic({
agent: {
2025-12-26 00:43:44 +01:00
model: "anthropic/claude-opus-4-5",
2025-12-17 11:29:04 +01:00
contextTokens: 32_000,
},
sessionEntry: {
sessionId,
updatedAt: 0,
totalTokens: 3, // would be wrong if cached prompt tokens exist
contextTokens: 32_000,
},
sessionKey: "agent:main:main",
2025-12-17 11:29:04 +01:00
sessionScope: "per-sender",
2026-01-07 07:21:56 +01:00
queue: { mode: "collect", depth: 0 },
includeTranscriptUsage: true,
2025-12-17 11:29:04 +01:00
});
2025-12-17 11:29:04 +01:00
expect(text).toContain("Context: 1.0k/32k");
} finally {
process.env.HOME = previousHome;
fs.rmSync(dir, { recursive: true, force: true });
}
});
2025-12-07 16:53:19 +00:00
});