mirror of https://github.com/openclaw/openclaw.git
206 lines
6.5 KiB
TypeScript
206 lines
6.5 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { afterEach, describe, expect, it, vi } from "vitest";
|
|
import { createCronStoreHarness } from "./service.test-harness.js";
|
|
import { loadCronStore, resolveCronStorePath, saveCronStore } from "./store.js";
|
|
import type { CronStoreFile } from "./types.js";
|
|
|
|
const { makeStorePath } = createCronStoreHarness({ prefix: "openclaw-cron-store-" });
|
|
|
|
function makeStore(jobId: string, enabled: boolean): CronStoreFile {
|
|
const now = Date.now();
|
|
return {
|
|
version: 1,
|
|
jobs: [
|
|
{
|
|
id: jobId,
|
|
name: `Job ${jobId}`,
|
|
enabled,
|
|
createdAtMs: now,
|
|
updatedAtMs: now,
|
|
schedule: { kind: "every", everyMs: 60_000 },
|
|
sessionTarget: "main",
|
|
wakeMode: "next-heartbeat",
|
|
payload: { kind: "systemEvent", text: `tick-${jobId}` },
|
|
state: {},
|
|
},
|
|
],
|
|
};
|
|
}
|
|
|
|
describe("resolveCronStorePath", () => {
|
|
afterEach(() => {
|
|
vi.unstubAllEnvs();
|
|
});
|
|
|
|
it("uses OPENCLAW_HOME for tilde expansion", () => {
|
|
vi.stubEnv("OPENCLAW_HOME", "/srv/openclaw-home");
|
|
vi.stubEnv("HOME", "/home/other");
|
|
|
|
const result = resolveCronStorePath("~/cron/jobs.json");
|
|
expect(result).toBe(path.resolve("/srv/openclaw-home", "cron", "jobs.json"));
|
|
});
|
|
});
|
|
|
|
describe("cron store", () => {
|
|
it("returns empty store when file does not exist", async () => {
|
|
const store = await makeStorePath();
|
|
const loaded = await loadCronStore(store.storePath);
|
|
expect(loaded).toEqual({ version: 1, jobs: [] });
|
|
});
|
|
|
|
it("throws when store contains invalid JSON", async () => {
|
|
const store = await makeStorePath();
|
|
await fs.mkdir(path.dirname(store.storePath), { recursive: true });
|
|
await fs.writeFile(store.storePath, "{ not json", "utf-8");
|
|
await expect(loadCronStore(store.storePath)).rejects.toThrow(/Failed to parse cron store/i);
|
|
});
|
|
|
|
it("accepts JSON5 syntax when loading an existing cron store", async () => {
|
|
const store = await makeStorePath();
|
|
await fs.mkdir(path.dirname(store.storePath), { recursive: true });
|
|
await fs.writeFile(
|
|
store.storePath,
|
|
`{
|
|
// hand-edited legacy store
|
|
version: 1,
|
|
jobs: [
|
|
{
|
|
id: 'job-1',
|
|
name: 'Job 1',
|
|
enabled: true,
|
|
createdAtMs: 1,
|
|
updatedAtMs: 1,
|
|
schedule: { kind: 'every', everyMs: 60000 },
|
|
sessionTarget: 'main',
|
|
wakeMode: 'next-heartbeat',
|
|
payload: { kind: 'systemEvent', text: 'tick-job-1' },
|
|
state: {},
|
|
},
|
|
],
|
|
}`,
|
|
"utf-8",
|
|
);
|
|
|
|
await expect(loadCronStore(store.storePath)).resolves.toMatchObject({
|
|
version: 1,
|
|
jobs: [{ id: "job-1", enabled: true }],
|
|
});
|
|
});
|
|
|
|
it("does not create a backup file when saving unchanged content", async () => {
|
|
const store = await makeStorePath();
|
|
const payload = makeStore("job-1", true);
|
|
|
|
await saveCronStore(store.storePath, payload);
|
|
await saveCronStore(store.storePath, payload);
|
|
|
|
await expect(fs.stat(`${store.storePath}.bak`)).rejects.toThrow();
|
|
});
|
|
|
|
it("backs up previous content before replacing the store", async () => {
|
|
const store = await makeStorePath();
|
|
const first = makeStore("job-1", true);
|
|
const second = makeStore("job-2", false);
|
|
|
|
await saveCronStore(store.storePath, first);
|
|
await saveCronStore(store.storePath, second);
|
|
|
|
const currentRaw = await fs.readFile(store.storePath, "utf-8");
|
|
const backupRaw = await fs.readFile(`${store.storePath}.bak`, "utf-8");
|
|
expect(JSON.parse(currentRaw)).toEqual(second);
|
|
expect(JSON.parse(backupRaw)).toEqual(first);
|
|
});
|
|
|
|
it.skipIf(process.platform === "win32")(
|
|
"writes store and backup files with secure permissions",
|
|
async () => {
|
|
const store = await makeStorePath();
|
|
const first = makeStore("job-1", true);
|
|
const second = makeStore("job-2", false);
|
|
|
|
await saveCronStore(store.storePath, first);
|
|
await saveCronStore(store.storePath, second);
|
|
|
|
const storeMode = (await fs.stat(store.storePath)).mode & 0o777;
|
|
const backupMode = (await fs.stat(`${store.storePath}.bak`)).mode & 0o777;
|
|
|
|
expect(storeMode).toBe(0o600);
|
|
expect(backupMode).toBe(0o600);
|
|
},
|
|
);
|
|
|
|
it.skipIf(process.platform === "win32")(
|
|
"hardens an existing cron store directory to owner-only permissions",
|
|
async () => {
|
|
const store = await makeStorePath();
|
|
const storeDir = path.dirname(store.storePath);
|
|
await fs.mkdir(storeDir, { recursive: true, mode: 0o755 });
|
|
await fs.chmod(storeDir, 0o755);
|
|
|
|
await saveCronStore(store.storePath, makeStore("job-1", true));
|
|
|
|
const storeDirMode = (await fs.stat(storeDir)).mode & 0o777;
|
|
expect(storeDirMode).toBe(0o700);
|
|
},
|
|
);
|
|
});
|
|
|
|
describe("saveCronStore", () => {
|
|
const dummyStore: CronStoreFile = { version: 1, jobs: [] };
|
|
|
|
it("persists and round-trips a store file", async () => {
|
|
const { storePath } = await makeStorePath();
|
|
await saveCronStore(storePath, dummyStore);
|
|
const loaded = await loadCronStore(storePath);
|
|
expect(loaded).toEqual(dummyStore);
|
|
});
|
|
|
|
it("retries rename on EBUSY then succeeds", async () => {
|
|
const { storePath } = await makeStorePath();
|
|
const realSetTimeout = globalThis.setTimeout;
|
|
const setTimeoutSpy = vi
|
|
.spyOn(globalThis, "setTimeout")
|
|
.mockImplementation(((handler: TimerHandler, _timeout?: number, ...args: unknown[]) =>
|
|
realSetTimeout(handler, 0, ...args)) as typeof setTimeout);
|
|
const origRename = fs.rename.bind(fs);
|
|
let ebusyCount = 0;
|
|
const spy = vi.spyOn(fs, "rename").mockImplementation(async (src, dest) => {
|
|
if (ebusyCount < 2) {
|
|
ebusyCount++;
|
|
const err = new Error("EBUSY") as NodeJS.ErrnoException;
|
|
err.code = "EBUSY";
|
|
throw err;
|
|
}
|
|
return origRename(src, dest);
|
|
});
|
|
|
|
try {
|
|
await saveCronStore(storePath, dummyStore);
|
|
|
|
expect(ebusyCount).toBe(2);
|
|
const loaded = await loadCronStore(storePath);
|
|
expect(loaded).toEqual(dummyStore);
|
|
} finally {
|
|
spy.mockRestore();
|
|
setTimeoutSpy.mockRestore();
|
|
}
|
|
});
|
|
|
|
it("falls back to copyFile on EPERM (Windows)", async () => {
|
|
const { storePath } = await makeStorePath();
|
|
|
|
const spy = vi.spyOn(fs, "rename").mockImplementation(async () => {
|
|
const err = new Error("EPERM") as NodeJS.ErrnoException;
|
|
err.code = "EPERM";
|
|
throw err;
|
|
});
|
|
|
|
await saveCronStore(storePath, dummyStore);
|
|
const loaded = await loadCronStore(storePath);
|
|
expect(loaded).toEqual(dummyStore);
|
|
|
|
spy.mockRestore();
|
|
});
|
|
});
|