255 lines
6.9 KiB
TypeScript
255 lines
6.9 KiB
TypeScript
import path from "node:path";
|
|
|
|
import { describe, expect, it } from "vitest";
|
|
|
|
import {
|
|
applySoulEvilOverride,
|
|
decideSoulEvil,
|
|
DEFAULT_SOUL_EVIL_FILENAME,
|
|
resolveSoulEvilConfigFromHook,
|
|
} from "./soul-evil.js";
|
|
import { DEFAULT_SOUL_FILENAME, type WorkspaceBootstrapFile } from "../agents/workspace.js";
|
|
import { makeTempWorkspace, writeWorkspaceFile } from "../test-helpers/workspace.js";
|
|
|
|
const makeFiles = (overrides?: Partial<WorkspaceBootstrapFile>) => [
|
|
{
|
|
name: DEFAULT_SOUL_FILENAME,
|
|
path: "/tmp/SOUL.md",
|
|
content: "friendly",
|
|
missing: false,
|
|
...overrides,
|
|
},
|
|
];
|
|
|
|
describe("decideSoulEvil", () => {
|
|
it("returns false when no config", () => {
|
|
const result = decideSoulEvil({});
|
|
expect(result.useEvil).toBe(false);
|
|
});
|
|
|
|
it("activates on random chance", () => {
|
|
const result = decideSoulEvil({
|
|
config: { chance: 0.5 },
|
|
random: () => 0.2,
|
|
});
|
|
expect(result.useEvil).toBe(true);
|
|
expect(result.reason).toBe("chance");
|
|
});
|
|
|
|
it("activates during purge window", () => {
|
|
const result = decideSoulEvil({
|
|
config: {
|
|
purge: { at: "00:00", duration: "10m" },
|
|
},
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-01T00:05:00Z"),
|
|
});
|
|
expect(result.useEvil).toBe(true);
|
|
expect(result.reason).toBe("purge");
|
|
});
|
|
|
|
it("prefers purge window over random chance", () => {
|
|
const result = decideSoulEvil({
|
|
config: {
|
|
chance: 0,
|
|
purge: { at: "00:00", duration: "10m" },
|
|
},
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-01T00:05:00Z"),
|
|
random: () => 0,
|
|
});
|
|
expect(result.useEvil).toBe(true);
|
|
expect(result.reason).toBe("purge");
|
|
});
|
|
|
|
it("skips purge window when outside duration", () => {
|
|
const result = decideSoulEvil({
|
|
config: {
|
|
purge: { at: "00:00", duration: "10m" },
|
|
},
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-01T00:30:00Z"),
|
|
});
|
|
expect(result.useEvil).toBe(false);
|
|
});
|
|
|
|
it("honors sub-minute purge durations", () => {
|
|
const config = {
|
|
purge: { at: "00:00", duration: "30s" },
|
|
};
|
|
const active = decideSoulEvil({
|
|
config,
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-01T00:00:20Z"),
|
|
});
|
|
const inactive = decideSoulEvil({
|
|
config,
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-01T00:00:40Z"),
|
|
});
|
|
expect(active.useEvil).toBe(true);
|
|
expect(active.reason).toBe("purge");
|
|
expect(inactive.useEvil).toBe(false);
|
|
});
|
|
|
|
it("handles purge windows that wrap past midnight", () => {
|
|
const result = decideSoulEvil({
|
|
config: {
|
|
purge: { at: "23:55", duration: "10m" },
|
|
},
|
|
userTimezone: "UTC",
|
|
now: new Date("2026-01-02T00:02:00Z"),
|
|
});
|
|
expect(result.useEvil).toBe(true);
|
|
expect(result.reason).toBe("purge");
|
|
});
|
|
|
|
it("clamps chance above 1", () => {
|
|
const result = decideSoulEvil({
|
|
config: { chance: 2 },
|
|
random: () => 0.5,
|
|
});
|
|
expect(result.useEvil).toBe(true);
|
|
expect(result.reason).toBe("chance");
|
|
});
|
|
});
|
|
|
|
describe("applySoulEvilOverride", () => {
|
|
it("replaces SOUL content when evil is active and file exists", async () => {
|
|
const tempDir = await makeTempWorkspace("clawdbot-soul-");
|
|
await writeWorkspaceFile({
|
|
dir: tempDir,
|
|
name: DEFAULT_SOUL_EVIL_FILENAME,
|
|
content: "chaotic",
|
|
});
|
|
|
|
const files = makeFiles({
|
|
path: path.join(tempDir, DEFAULT_SOUL_FILENAME),
|
|
});
|
|
|
|
const updated = await applySoulEvilOverride({
|
|
files,
|
|
workspaceDir: tempDir,
|
|
config: { chance: 1 },
|
|
userTimezone: "UTC",
|
|
random: () => 0,
|
|
});
|
|
|
|
const soul = updated.find((file) => file.name === DEFAULT_SOUL_FILENAME);
|
|
expect(soul?.content).toBe("chaotic");
|
|
});
|
|
|
|
it("leaves SOUL content when evil file is missing", async () => {
|
|
const tempDir = await makeTempWorkspace("clawdbot-soul-");
|
|
const files = makeFiles({
|
|
path: path.join(tempDir, DEFAULT_SOUL_FILENAME),
|
|
});
|
|
|
|
const updated = await applySoulEvilOverride({
|
|
files,
|
|
workspaceDir: tempDir,
|
|
config: { chance: 1 },
|
|
userTimezone: "UTC",
|
|
random: () => 0,
|
|
});
|
|
|
|
const soul = updated.find((file) => file.name === DEFAULT_SOUL_FILENAME);
|
|
expect(soul?.content).toBe("friendly");
|
|
});
|
|
|
|
it("uses custom evil filename when configured", async () => {
|
|
const tempDir = await makeTempWorkspace("clawdbot-soul-");
|
|
await writeWorkspaceFile({
|
|
dir: tempDir,
|
|
name: "SOUL_EVIL_CUSTOM.md",
|
|
content: "chaotic",
|
|
});
|
|
|
|
const files = makeFiles({
|
|
path: path.join(tempDir, DEFAULT_SOUL_FILENAME),
|
|
});
|
|
|
|
const updated = await applySoulEvilOverride({
|
|
files,
|
|
workspaceDir: tempDir,
|
|
config: { chance: 1, file: "SOUL_EVIL_CUSTOM.md" },
|
|
userTimezone: "UTC",
|
|
random: () => 0,
|
|
});
|
|
|
|
const soul = updated.find((file) => file.name === DEFAULT_SOUL_FILENAME);
|
|
expect(soul?.content).toBe("chaotic");
|
|
});
|
|
|
|
it("warns and skips when evil file is empty", async () => {
|
|
const tempDir = await makeTempWorkspace("clawdbot-soul-");
|
|
await writeWorkspaceFile({
|
|
dir: tempDir,
|
|
name: DEFAULT_SOUL_EVIL_FILENAME,
|
|
content: " ",
|
|
});
|
|
|
|
const warnings: string[] = [];
|
|
const files = makeFiles({
|
|
path: path.join(tempDir, DEFAULT_SOUL_FILENAME),
|
|
});
|
|
|
|
const updated = await applySoulEvilOverride({
|
|
files,
|
|
workspaceDir: tempDir,
|
|
config: { chance: 1 },
|
|
userTimezone: "UTC",
|
|
random: () => 0,
|
|
log: { warn: (message) => warnings.push(message) },
|
|
});
|
|
|
|
const soul = updated.find((file) => file.name === DEFAULT_SOUL_FILENAME);
|
|
expect(soul?.content).toBe("friendly");
|
|
expect(warnings.some((message) => message.includes("file empty"))).toBe(true);
|
|
});
|
|
|
|
it("leaves files untouched when SOUL.md is not in bootstrap files", async () => {
|
|
const tempDir = await makeTempWorkspace("clawdbot-soul-");
|
|
await writeWorkspaceFile({
|
|
dir: tempDir,
|
|
name: DEFAULT_SOUL_EVIL_FILENAME,
|
|
content: "chaotic",
|
|
});
|
|
|
|
const files: WorkspaceBootstrapFile[] = [
|
|
{
|
|
name: "AGENTS.md",
|
|
path: path.join(tempDir, "AGENTS.md"),
|
|
content: "agents",
|
|
missing: false,
|
|
},
|
|
];
|
|
|
|
const updated = await applySoulEvilOverride({
|
|
files,
|
|
workspaceDir: tempDir,
|
|
config: { chance: 1 },
|
|
userTimezone: "UTC",
|
|
random: () => 0,
|
|
});
|
|
|
|
expect(updated).toEqual(files);
|
|
});
|
|
});
|
|
|
|
describe("resolveSoulEvilConfigFromHook", () => {
|
|
it("returns null and warns when config is invalid", () => {
|
|
const warnings: string[] = [];
|
|
const result = resolveSoulEvilConfigFromHook(
|
|
{ file: 42, chance: "nope", purge: "later" },
|
|
{ warn: (message) => warnings.push(message) },
|
|
);
|
|
expect(result).toBeNull();
|
|
expect(warnings).toEqual([
|
|
"soul-evil config: file must be a string",
|
|
"soul-evil config: chance must be a number",
|
|
"soul-evil config: purge must be an object",
|
|
]);
|
|
});
|
|
});
|