mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 05:47:39 +00:00
refactor(test): share auto-reply temp home harness
This commit is contained in:
@@ -1,7 +1,5 @@
|
|||||||
import fs from "node:fs/promises";
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
import os from "node:os";
|
import { createTempHomeHarness, makeReplyConfig } from "./reply.test-harness.js";
|
||||||
import { join } from "node:path";
|
|
||||||
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
|
||||||
|
|
||||||
const runEmbeddedPiAgentMock = vi.fn();
|
const runEmbeddedPiAgentMock = vi.fn();
|
||||||
|
|
||||||
@@ -40,95 +38,11 @@ vi.mock("../web/session.js", () => webMocks);
|
|||||||
|
|
||||||
import { getReplyFromConfig } from "./reply.js";
|
import { getReplyFromConfig } from "./reply.js";
|
||||||
|
|
||||||
type HomeEnvSnapshot = {
|
const { withTempHome } = createTempHomeHarness({
|
||||||
HOME: string | undefined;
|
prefix: "openclaw-typing-",
|
||||||
USERPROFILE: string | undefined;
|
beforeEachCase: () => runEmbeddedPiAgentMock.mockClear(),
|
||||||
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;
|
|
||||||
|
|
||||||
beforeAll(async () => {
|
|
||||||
fixtureRoot = await fs.mkdtemp(join(os.tmpdir(), "openclaw-typing-"));
|
|
||||||
});
|
});
|
||||||
|
|
||||||
afterAll(async () => {
|
|
||||||
if (!fixtureRoot) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await fs.rm(fixtureRoot, { recursive: true, force: true });
|
|
||||||
});
|
|
||||||
|
|
||||||
async function withTempHome<T>(fn: (home: string) => Promise<T>): Promise<T> {
|
|
||||||
const home = join(fixtureRoot, `case-${++caseId}`);
|
|
||||||
await fs.mkdir(join(home, ".openclaw", "agents", "main", "sessions"), { recursive: true });
|
|
||||||
const envSnapshot = snapshotHomeEnv();
|
|
||||||
process.env.HOME = home;
|
|
||||||
process.env.USERPROFILE = home;
|
|
||||||
process.env.OPENCLAW_STATE_DIR = join(home, ".openclaw");
|
|
||||||
process.env.OPENCLAW_AGENT_DIR = join(home, ".openclaw", "agent");
|
|
||||||
process.env.PI_CODING_AGENT_DIR = 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 {
|
|
||||||
runEmbeddedPiAgentMock.mockClear();
|
|
||||||
return await fn(home);
|
|
||||||
} finally {
|
|
||||||
restoreHomeEnv(envSnapshot);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function makeCfg(home: string) {
|
|
||||||
return {
|
|
||||||
agents: {
|
|
||||||
defaults: {
|
|
||||||
model: "anthropic/claude-opus-4-5",
|
|
||||||
workspace: join(home, "openclaw"),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
channels: {
|
|
||||||
whatsapp: {
|
|
||||||
allowFrom: ["*"],
|
|
||||||
},
|
|
||||||
},
|
|
||||||
session: { store: join(home, "sessions.json") },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
vi.restoreAllMocks();
|
vi.restoreAllMocks();
|
||||||
});
|
});
|
||||||
@@ -149,7 +63,7 @@ describe("getReplyFromConfig typing (heartbeat)", () => {
|
|||||||
await getReplyFromConfig(
|
await getReplyFromConfig(
|
||||||
{ Body: "hi", From: "+1000", To: "+2000", Provider: "whatsapp" },
|
{ Body: "hi", From: "+1000", To: "+2000", Provider: "whatsapp" },
|
||||||
{ onReplyStart, isHeartbeat: false },
|
{ onReplyStart, isHeartbeat: false },
|
||||||
makeCfg(home),
|
makeReplyConfig(home),
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(onReplyStart).toHaveBeenCalled();
|
expect(onReplyStart).toHaveBeenCalled();
|
||||||
@@ -167,7 +81,7 @@ describe("getReplyFromConfig typing (heartbeat)", () => {
|
|||||||
await getReplyFromConfig(
|
await getReplyFromConfig(
|
||||||
{ Body: "hi", From: "+1000", To: "+2000", Provider: "whatsapp" },
|
{ Body: "hi", From: "+1000", To: "+2000", Provider: "whatsapp" },
|
||||||
{ onReplyStart, isHeartbeat: true },
|
{ onReplyStart, isHeartbeat: true },
|
||||||
makeCfg(home),
|
makeReplyConfig(home),
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(onReplyStart).not.toHaveBeenCalled();
|
expect(onReplyStart).not.toHaveBeenCalled();
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
import fs from "node:fs/promises";
|
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
import os from "node:os";
|
import { createTempHomeHarness, makeReplyConfig } from "./reply.test-harness.js";
|
||||||
import path from "node:path";
|
|
||||||
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, it, vi } from "vitest";
|
|
||||||
|
|
||||||
const agentMocks = vi.hoisted(() => ({
|
const agentMocks = vi.hoisted(() => ({
|
||||||
runEmbeddedPiAgent: vi.fn(),
|
runEmbeddedPiAgent: vi.fn(),
|
||||||
@@ -32,75 +30,9 @@ vi.mock("../web/session.js", () => ({
|
|||||||
|
|
||||||
import { getReplyFromConfig } from "./reply.js";
|
import { getReplyFromConfig } from "./reply.js";
|
||||||
|
|
||||||
type HomeEnvSnapshot = {
|
const { withTempHome } = createTempHomeHarness({ prefix: "openclaw-rawbody-" });
|
||||||
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", () => {
|
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(() => {
|
beforeEach(() => {
|
||||||
vi.stubEnv("OPENCLAW_TEST_FAST", "1");
|
vi.stubEnv("OPENCLAW_TEST_FAST", "1");
|
||||||
agentMocks.runEmbeddedPiAgent.mockReset();
|
agentMocks.runEmbeddedPiAgent.mockReset();
|
||||||
@@ -138,20 +70,7 @@ describe("RawBody directive parsing", () => {
|
|||||||
CommandAuthorized: true,
|
CommandAuthorized: true,
|
||||||
};
|
};
|
||||||
|
|
||||||
const res = await getReplyFromConfig(
|
const res = await getReplyFromConfig(groupMessageCtx, {}, makeReplyConfig(home));
|
||||||
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;
|
const text = Array.isArray(res) ? res[0]?.text : res?.text;
|
||||||
expect(text).toBe("ok");
|
expect(text).toBe("ok");
|
||||||
|
|||||||
97
src/auto-reply/reply.test-harness.ts
Normal file
97
src/auto-reply/reply.test-harness.ts
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
import fs from "node:fs/promises";
|
||||||
|
import os from "node:os";
|
||||||
|
import path from "node:path";
|
||||||
|
import { afterAll, beforeAll } from "vitest";
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createTempHomeHarness(options: { prefix: string; beforeEachCase?: () => void }) {
|
||||||
|
let fixtureRoot = "";
|
||||||
|
let caseId = 0;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), options.prefix));
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(async () => {
|
||||||
|
if (!fixtureRoot) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
await fs.rm(fixtureRoot, { recursive: true, force: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
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 {
|
||||||
|
options.beforeEachCase?.();
|
||||||
|
return await fn(home);
|
||||||
|
} finally {
|
||||||
|
restoreHomeEnv(envSnapshot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { withTempHome };
|
||||||
|
}
|
||||||
|
|
||||||
|
export function makeReplyConfig(home: string) {
|
||||||
|
return {
|
||||||
|
agents: {
|
||||||
|
defaults: {
|
||||||
|
model: "anthropic/claude-opus-4-5",
|
||||||
|
workspace: path.join(home, "openclaw"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
channels: {
|
||||||
|
whatsapp: {
|
||||||
|
allowFrom: ["*"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
session: { store: path.join(home, "sessions.json") },
|
||||||
|
};
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user