170 lines
5.3 KiB
TypeScript
170 lines
5.3 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
|
|
|
const agentMocks = vi.hoisted(() => ({
|
|
runEmbeddedPiAgent: vi.fn(),
|
|
loadModelCatalog: vi.fn(),
|
|
webAuthExists: vi.fn().mockResolvedValue(true),
|
|
getWebAuthAgeMs: vi.fn().mockReturnValue(120_000),
|
|
readWebSelfId: vi.fn().mockReturnValue({ e164: "+1999" }),
|
|
}));
|
|
|
|
vi.mock("../agents/pi-embedded.js", () => ({
|
|
abortEmbeddedPiRun: vi.fn().mockReturnValue(false),
|
|
runEmbeddedPiAgent: agentMocks.runEmbeddedPiAgent,
|
|
queueEmbeddedPiMessage: vi.fn().mockReturnValue(false),
|
|
resolveEmbeddedSessionLane: (key: string) => `session:${key.trim() || "main"}`,
|
|
isEmbeddedPiRunActive: vi.fn().mockReturnValue(false),
|
|
isEmbeddedPiRunStreaming: vi.fn().mockReturnValue(false),
|
|
}));
|
|
|
|
vi.mock("../agents/model-catalog.js", () => ({
|
|
loadModelCatalog: agentMocks.loadModelCatalog,
|
|
}));
|
|
|
|
vi.mock("../web/session.js", () => ({
|
|
webAuthExists: agentMocks.webAuthExists,
|
|
getWebAuthAgeMs: agentMocks.getWebAuthAgeMs,
|
|
readWebSelfId: agentMocks.readWebSelfId,
|
|
}));
|
|
|
|
import { getReplyFromConfig } from "./reply.js";
|
|
|
|
type HomeEnvSnapshot = {
|
|
HOME: string | undefined;
|
|
USERPROFILE: string | undefined;
|
|
HOMEDRIVE: string | undefined;
|
|
HOMEPATH: string | undefined;
|
|
OPENCLAW_STATE_DIR: string | undefined;
|
|
OPENCLAW_AGENT_DIR: string | undefined;
|
|
PI_CODING_AGENT_DIR: string | undefined;
|
|
};
|
|
|
|
function snapshotHomeEnv(): HomeEnvSnapshot {
|
|
return {
|
|
HOME: process.env.HOME,
|
|
USERPROFILE: process.env.USERPROFILE,
|
|
HOMEDRIVE: process.env.HOMEDRIVE,
|
|
HOMEPATH: process.env.HOMEPATH,
|
|
OPENCLAW_STATE_DIR: process.env.OPENCLAW_STATE_DIR,
|
|
OPENCLAW_AGENT_DIR: process.env.OPENCLAW_AGENT_DIR,
|
|
PI_CODING_AGENT_DIR: process.env.PI_CODING_AGENT_DIR,
|
|
};
|
|
}
|
|
|
|
function restoreHomeEnv(snapshot: HomeEnvSnapshot) {
|
|
for (const [key, value] of Object.entries(snapshot)) {
|
|
if (value === undefined) {
|
|
delete process.env[key];
|
|
} else {
|
|
process.env[key] = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
let fixtureRoot = "";
|
|
let caseId = 0;
|
|
|
|
async function withTempHome<T>(fn: (home: string) => Promise<T>): Promise<T> {
|
|
const home = path.join(fixtureRoot, `case-${++caseId}`);
|
|
await fs.mkdir(path.join(home, ".openclaw", "agents", "main", "sessions"), { recursive: true });
|
|
const envSnapshot = snapshotHomeEnv();
|
|
process.env.HOME = home;
|
|
process.env.USERPROFILE = home;
|
|
process.env.OPENCLAW_STATE_DIR = path.join(home, ".openclaw");
|
|
process.env.OPENCLAW_AGENT_DIR = path.join(home, ".openclaw", "agent");
|
|
process.env.PI_CODING_AGENT_DIR = path.join(home, ".openclaw", "agent");
|
|
|
|
if (process.platform === "win32") {
|
|
const match = home.match(/^([A-Za-z]:)(.*)$/);
|
|
if (match) {
|
|
process.env.HOMEDRIVE = match[1];
|
|
process.env.HOMEPATH = match[2] || "\\";
|
|
}
|
|
}
|
|
|
|
try {
|
|
return await fn(home);
|
|
} finally {
|
|
restoreHomeEnv(envSnapshot);
|
|
}
|
|
}
|
|
|
|
describe("RawBody directive parsing", () => {
|
|
beforeAll(async () => {
|
|
fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-rawbody-"));
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await fs.rm(fixtureRoot, { recursive: true, force: true });
|
|
});
|
|
|
|
beforeEach(() => {
|
|
vi.stubEnv("OPENCLAW_TEST_FAST", "1");
|
|
agentMocks.runEmbeddedPiAgent.mockReset();
|
|
agentMocks.loadModelCatalog.mockReset();
|
|
agentMocks.loadModelCatalog.mockResolvedValue([
|
|
{ id: "claude-opus-4-5", name: "Opus 4.5", provider: "anthropic" },
|
|
]);
|
|
});
|
|
|
|
afterEach(() => {
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it("handles directives and history in the prompt", async () => {
|
|
await withTempHome(async (home) => {
|
|
agentMocks.runEmbeddedPiAgent.mockResolvedValue({
|
|
payloads: [{ text: "ok" }],
|
|
meta: {
|
|
durationMs: 1,
|
|
agentMeta: { sessionId: "s", provider: "p", model: "m" },
|
|
},
|
|
});
|
|
|
|
const groupMessageCtx = {
|
|
Body: "/think:high status please",
|
|
BodyForAgent: "/think:high status please",
|
|
RawBody: "/think:high status please",
|
|
InboundHistory: [{ sender: "Peter", body: "hello", timestamp: 1700000000000 }],
|
|
From: "+1222",
|
|
To: "+1222",
|
|
ChatType: "group",
|
|
GroupSubject: "Ops",
|
|
SenderName: "Jake McInteer",
|
|
SenderE164: "+6421807830",
|
|
CommandAuthorized: true,
|
|
};
|
|
|
|
const res = await getReplyFromConfig(
|
|
groupMessageCtx,
|
|
{},
|
|
{
|
|
agents: {
|
|
defaults: {
|
|
model: "anthropic/claude-opus-4-5",
|
|
workspace: path.join(home, "openclaw"),
|
|
},
|
|
},
|
|
channels: { whatsapp: { allowFrom: ["*"] } },
|
|
session: { store: path.join(home, "sessions.json") },
|
|
},
|
|
);
|
|
|
|
const text = Array.isArray(res) ? res[0]?.text : res?.text;
|
|
expect(text).toBe("ok");
|
|
expect(agentMocks.runEmbeddedPiAgent).toHaveBeenCalledOnce();
|
|
const prompt =
|
|
(agentMocks.runEmbeddedPiAgent.mock.calls[0]?.[0] as { prompt?: string } | undefined)
|
|
?.prompt ?? "";
|
|
expect(prompt).toContain("Chat history since last reply (untrusted, for context):");
|
|
expect(prompt).toContain('"sender": "Peter"');
|
|
expect(prompt).toContain('"body": "hello"');
|
|
expect(prompt).toContain("status please");
|
|
expect(prompt).not.toContain("/think:high");
|
|
});
|
|
});
|
|
});
|