refactor(cron): move store into ~/.clawdis/cron
This commit is contained in:
@@ -11,13 +11,7 @@ import {
|
||||
} from "./run-log.js";
|
||||
|
||||
describe("cron run log", () => {
|
||||
it("resolves a flat store path to cron.runs.jsonl", () => {
|
||||
const storePath = path.join(os.tmpdir(), "cron.json");
|
||||
const p = resolveCronRunLogPath({ storePath, jobId: "job-1" });
|
||||
expect(p.endsWith(path.join(os.tmpdir(), "cron.runs.jsonl"))).toBe(true);
|
||||
});
|
||||
|
||||
it("resolves jobs.json to per-job runs/<jobId>.jsonl", () => {
|
||||
it("resolves store path to per-job runs/<jobId>.jsonl", () => {
|
||||
const storePath = path.join(os.tmpdir(), "cron", "jobs.json");
|
||||
const p = resolveCronRunLogPath({ storePath, jobId: "job-1" });
|
||||
expect(
|
||||
@@ -27,7 +21,7 @@ describe("cron run log", () => {
|
||||
|
||||
it("appends JSONL and prunes by line count", async () => {
|
||||
const dir = await fs.mkdtemp(path.join(os.tmpdir(), "clawdis-cron-log-"));
|
||||
const logPath = path.join(dir, "cron.runs.jsonl");
|
||||
const logPath = path.join(dir, "runs", "job-1.jsonl");
|
||||
|
||||
for (let i = 0; i < 10; i++) {
|
||||
await appendCronRunLog(
|
||||
@@ -59,15 +53,16 @@ describe("cron run log", () => {
|
||||
const dir = await fs.mkdtemp(
|
||||
path.join(os.tmpdir(), "clawdis-cron-log-read-"),
|
||||
);
|
||||
const logPath = path.join(dir, "cron.runs.jsonl");
|
||||
const logPathA = path.join(dir, "runs", "a.jsonl");
|
||||
const logPathB = path.join(dir, "runs", "b.jsonl");
|
||||
|
||||
await appendCronRunLog(logPath, {
|
||||
await appendCronRunLog(logPathA, {
|
||||
ts: 1,
|
||||
jobId: "a",
|
||||
action: "finished",
|
||||
status: "ok",
|
||||
});
|
||||
await appendCronRunLog(logPath, {
|
||||
await appendCronRunLog(logPathB, {
|
||||
ts: 2,
|
||||
jobId: "b",
|
||||
action: "finished",
|
||||
@@ -75,31 +70,37 @@ describe("cron run log", () => {
|
||||
error: "nope",
|
||||
summary: "oops",
|
||||
});
|
||||
await appendCronRunLog(logPath, {
|
||||
await appendCronRunLog(logPathA, {
|
||||
ts: 3,
|
||||
jobId: "a",
|
||||
action: "finished",
|
||||
status: "skipped",
|
||||
});
|
||||
|
||||
const all = await readCronRunLogEntries(logPath, { limit: 10 });
|
||||
expect(all.map((e) => e.jobId)).toEqual(["a", "b", "a"]);
|
||||
const allA = await readCronRunLogEntries(logPathA, { limit: 10 });
|
||||
expect(allA.map((e) => e.jobId)).toEqual(["a", "a"]);
|
||||
|
||||
const onlyA = await readCronRunLogEntries(logPath, {
|
||||
const onlyA = await readCronRunLogEntries(logPathA, {
|
||||
limit: 10,
|
||||
jobId: "a",
|
||||
});
|
||||
expect(onlyA.map((e) => e.ts)).toEqual([1, 3]);
|
||||
|
||||
const lastOne = await readCronRunLogEntries(logPath, { limit: 1 });
|
||||
const lastOne = await readCronRunLogEntries(logPathA, { limit: 1 });
|
||||
expect(lastOne.map((e) => e.ts)).toEqual([3]);
|
||||
|
||||
const onlyB = await readCronRunLogEntries(logPath, {
|
||||
const onlyB = await readCronRunLogEntries(logPathB, {
|
||||
limit: 10,
|
||||
jobId: "b",
|
||||
});
|
||||
expect(onlyB[0]?.summary).toBe("oops");
|
||||
|
||||
const wrongFilter = await readCronRunLogEntries(logPathA, {
|
||||
limit: 10,
|
||||
jobId: "b",
|
||||
});
|
||||
expect(wrongFilter).toEqual([]);
|
||||
|
||||
await fs.rm(dir, { recursive: true, force: true });
|
||||
});
|
||||
});
|
||||
|
||||
@@ -19,14 +19,7 @@ export function resolveCronRunLogPath(params: {
|
||||
}) {
|
||||
const storePath = path.resolve(params.storePath);
|
||||
const dir = path.dirname(storePath);
|
||||
const base = path.basename(storePath);
|
||||
if (base === "jobs.json") {
|
||||
return path.join(dir, "runs", `${params.jobId}.jsonl`);
|
||||
}
|
||||
|
||||
const ext = path.extname(base);
|
||||
const baseNoExt = ext ? base.slice(0, -ext.length) : base;
|
||||
return path.join(dir, `${baseNoExt}.runs.jsonl`);
|
||||
return path.join(dir, "runs", `${params.jobId}.jsonl`);
|
||||
}
|
||||
|
||||
const writesByPath = new Map<string, Promise<void>>();
|
||||
|
||||
@@ -16,7 +16,7 @@ const noopLogger = {
|
||||
async function makeStorePath() {
|
||||
const dir = await fs.mkdtemp(path.join(os.tmpdir(), "clawdis-cron-"));
|
||||
return {
|
||||
storePath: path.join(dir, "cron.json"),
|
||||
storePath: path.join(dir, "cron", "jobs.json"),
|
||||
cleanup: async () => {
|
||||
await fs.rm(dir, { recursive: true, force: true });
|
||||
},
|
||||
@@ -201,6 +201,7 @@ describe("CronService", () => {
|
||||
const requestReplyHeartbeatNow = vi.fn();
|
||||
|
||||
const atMs = Date.parse("2025-12-13T00:00:01.000Z");
|
||||
await fs.mkdir(path.dirname(store.storePath), { recursive: true });
|
||||
await fs.writeFile(
|
||||
store.storePath,
|
||||
JSON.stringify({
|
||||
|
||||
@@ -6,12 +6,9 @@ import JSON5 from "json5";
|
||||
import { CONFIG_DIR } from "../utils.js";
|
||||
import type { CronStoreFile } from "./types.js";
|
||||
|
||||
export const LEGACY_CRON_STORE_PATH = path.join(
|
||||
CONFIG_DIR,
|
||||
"cron",
|
||||
"jobs.json",
|
||||
);
|
||||
export const DEFAULT_CRON_STORE_PATH = path.join(CONFIG_DIR, "cron.json");
|
||||
export const DEFAULT_CRON_DIR = path.join(CONFIG_DIR, "cron");
|
||||
export const DEFAULT_CRON_STORE_PATH = path.join(DEFAULT_CRON_DIR, "jobs.json");
|
||||
export const LEGACY_FLAT_CRON_STORE_PATH = path.join(CONFIG_DIR, "cron.json");
|
||||
|
||||
export function resolveCronStorePath(storePath?: string) {
|
||||
if (storePath?.trim()) {
|
||||
@@ -20,11 +17,52 @@ export function resolveCronStorePath(storePath?: string) {
|
||||
return path.resolve(raw.replace("~", os.homedir()));
|
||||
return path.resolve(raw);
|
||||
}
|
||||
if (fs.existsSync(LEGACY_CRON_STORE_PATH)) return LEGACY_CRON_STORE_PATH;
|
||||
return DEFAULT_CRON_STORE_PATH;
|
||||
}
|
||||
|
||||
async function maybeMigrateLegacyFlatStore(storePath: string) {
|
||||
const resolved = path.resolve(storePath);
|
||||
const resolvedDefault = path.resolve(DEFAULT_CRON_STORE_PATH);
|
||||
if (resolved !== resolvedDefault) return;
|
||||
if (fs.existsSync(resolved)) return;
|
||||
if (!fs.existsSync(LEGACY_FLAT_CRON_STORE_PATH)) return;
|
||||
|
||||
try {
|
||||
const raw = await fs.promises.readFile(
|
||||
LEGACY_FLAT_CRON_STORE_PATH,
|
||||
"utf-8",
|
||||
);
|
||||
const parsed = JSON5.parse(raw) as Partial<CronStoreFile> | null;
|
||||
const jobs = Array.isArray(parsed?.jobs) ? (parsed?.jobs as never[]) : [];
|
||||
const store: CronStoreFile = {
|
||||
version: 1,
|
||||
jobs: jobs.filter(Boolean) as never as CronStoreFile["jobs"],
|
||||
};
|
||||
await saveCronStore(storePath, store);
|
||||
|
||||
await fs.promises.mkdir(DEFAULT_CRON_DIR, { recursive: true });
|
||||
const destBase = path.join(DEFAULT_CRON_DIR, "cron.json.migrated");
|
||||
const dest = fs.existsSync(destBase)
|
||||
? path.join(
|
||||
DEFAULT_CRON_DIR,
|
||||
`cron.json.migrated.${process.pid}.${Math.random().toString(16).slice(2)}`,
|
||||
)
|
||||
: destBase;
|
||||
try {
|
||||
await fs.promises.rename(LEGACY_FLAT_CRON_STORE_PATH, dest);
|
||||
} catch {
|
||||
await fs.promises.copyFile(LEGACY_FLAT_CRON_STORE_PATH, dest);
|
||||
await fs.promises.unlink(LEGACY_FLAT_CRON_STORE_PATH).catch(() => {
|
||||
/* ignore */
|
||||
});
|
||||
}
|
||||
} catch {
|
||||
// Best-effort; keep legacy store if anything fails.
|
||||
}
|
||||
}
|
||||
|
||||
export async function loadCronStore(storePath: string): Promise<CronStoreFile> {
|
||||
await maybeMigrateLegacyFlatStore(storePath);
|
||||
try {
|
||||
const raw = await fs.promises.readFile(storePath, "utf-8");
|
||||
const parsed = JSON5.parse(raw) as Partial<CronStoreFile> | null;
|
||||
|
||||
Reference in New Issue
Block a user