Spracherkennung für: .ts vermutete Sprache: Unknown {[0] [0] [0]} [Methode: Schwerpunktbildung, einfache Gewichte, sechs Dimensionen]
import fs from "node:fs/promises";
import { describe, expect, it, vi } from "vitest";
import {
createAbortAwareIsolatedRunner,
createDeferred,
createDueIsolatedJob,
createIsolatedRegressionJob,
createRunningCronServiceState,
noopLogger,
setupCronRegressionFixtures,
writeCronJobs,
} from "../../../test/helpers/cron/service-regression-fixtures.js";
import {
clearCommandLane,
setCommandLaneConcurrency,
waitForActiveTasks,
} from "../../process/command-queue.js";
import { CommandLane } from "../../process/lanes.js";
import { enqueueRun, run } from "./ops.js";
import type { CronEvent } from "./state.js";
import { createCronServiceState } from "./state.js";
import { onTimer } from "./timer.js";
const FAST_TIMEOUT_SECONDS = 1;
const opsRegressionFixtures = setupCronRegressionFixtures({
prefix: "cron-service-ops-regressions-",
});
describe("cron service ops regressions", () => {
it("skips forced manual runs while a timer-triggered run is in progress", async () => {
const store = opsRegressionFixtures.makeStorePath();
const dueAt = Date.now() - 1;
const job = createIsolatedRegressionJob({
id: "timer-overlap",
name: "timer-overlap",
scheduledAt: dueAt,
schedule: { kind: "at", at: new Date(dueAt).toISOString() },
payload: { kind: "agentTurn", message: "long task" },
state: { nextRunAtMs: dueAt },
});
await writeCronJobs(store.storePath, [job]);
let resolveRun:
| ((value: { status: "ok" | "error" | "skipped"; summary?: string; error?: string }) => void)
| undefined;
const started = createDeferred<void>();
const finished = createDeferred<void>();
const runIsolatedAgentJob = vi.fn(
async () =>
await new Promise<{ status: "ok" | "error" | "skipped"; summary?: string; error?: string }>(
(resolve) => {
resolveRun = resolve;
},
),
);
const state = createCronServiceState({
cronEnabled: true,
storePath: store.storePath,
log: noopLogger,
enqueueSystemEvent: vi.fn(),
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob,
onEvent: (evt: CronEvent) => {
if (evt.jobId !== job.id) {
return;
}
if (evt.action === "started") {
started.resolve();
} else if (evt.action === "finished" && evt.status === "ok") {
finished.resolve();
}
},
});
const timerPromise = onTimer(state);
await started.promise;
expect(runIsolatedAgentJob).toHaveBeenCalledTimes(1);
const manualResult = await run(state, job.id, "force");
expect(manualResult).toEqual({ ok: true, ran: false, reason: "already-running" });
expect(runIsolatedAgentJob).toHaveBeenCalledTimes(1);
resolveRun?.({ status: "ok", summary: "done" });
await finished.promise;
await timerPromise;
});
it("does not double-run a job when cron.run overlaps a due timer tick", async () => {
const store = opsRegressionFixtures.makeStorePath();
const now = Date.parse("2026-02-06T10:05:00.000Z");
const job = createIsolatedRegressionJob({
id: "manual-overlap-no-double-run",
name: "manual overlap no double run",
scheduledAt: now,
schedule: { kind: "at", at: new Date(now).toISOString() },
payload: { kind: "agentTurn", message: "overlap" },
state: { nextRunAtMs: now },
});
await writeCronJobs(store.storePath, [job]);
const runStarted = createDeferred<void>();
const runFinished = createDeferred<void>();
const runResolvers: Array<
(value: { status: "ok" | "error" | "skipped"; summary?: string }) => void
> = [];
const runIsolatedAgentJob = vi.fn(async () => {
if (runIsolatedAgentJob.mock.calls.length === 1) {
runStarted.resolve();
}
return await new Promise<{ status: "ok" | "error" | "skipped"; summary?: string }>(
(resolve) => {
runResolvers.push(resolve);
},
);
});
const state = createCronServiceState({
cronEnabled: true,
storePath: store.storePath,
log: noopLogger,
nowMs: () => now,
enqueueSystemEvent: vi.fn(),
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob,
onEvent: (evt: CronEvent) => {
if (evt.jobId === job.id && evt.action === "finished") {
runFinished.resolve();
}
},
});
const manualRun = run(state, job.id, "force");
await runStarted.promise;
expect(runIsolatedAgentJob).toHaveBeenCalledTimes(1);
await onTimer(state);
expect(runIsolatedAgentJob).toHaveBeenCalledTimes(1);
runResolvers[0]?.({ status: "ok", summary: "done" });
await manualRun;
await runFinished.promise;
});
it("manual cron.run preserves unrelated due jobs but advances already-executed stale slots", async () => {
const store = opsRegressionFixtures.makeStorePath();
const nowMs = Date.now();
const dueNextRunAtMs = nowMs - 1_000;
const staleExecutedNextRunAtMs = nowMs - 2_000;
await writeCronJobs(store.storePath, [
createIsolatedRegressionJob({
id: "manual-target",
name: "manual target",
scheduledAt: nowMs,
schedule: { kind: "at", at: new Date(nowMs + 3_600_000).toISOString() },
payload: { kind: "agentTurn", message: "manual target" },
state: { nextRunAtMs: nowMs + 3_600_000 },
}),
createIsolatedRegressionJob({
id: "unrelated-due",
name: "unrelated due",
scheduledAt: nowMs,
schedule: { kind: "cron", expr: "*/5 * * * *", tz: "UTC" },
payload: { kind: "agentTurn", message: "unrelated due" },
state: { nextRunAtMs: dueNextRunAtMs },
}),
createIsolatedRegressionJob({
id: "unrelated-stale-executed",
name: "unrelated stale executed",
scheduledAt: nowMs,
schedule: { kind: "cron", expr: "*/5 * * * *", tz: "UTC" },
payload: { kind: "agentTurn", message: "unrelated stale executed" },
state: {
nextRunAtMs: staleExecutedNextRunAtMs,
lastRunAtMs: staleExecutedNextRunAtMs + 1,
},
}),
]);
const state = createCronServiceState({
cronEnabled: false,
storePath: store.storePath,
log: noopLogger,
enqueueSystemEvent: vi.fn(),
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob: vi.fn().mockResolvedValue({ status: "ok", summary: "ok" }),
});
const runResult = await run(state, "manual-target", "force");
expect(runResult).toEqual({ ok: true, ran: true });
const jobs = state.store?.jobs ?? [];
const unrelated = jobs.find((entry) => entry.id === "unrelated-due");
const staleExecuted = jobs.find((entry) => entry.id === "unrelated-stale-executed");
expect(unrelated?.state.nextRunAtMs).toBe(dueNextRunAtMs);
expect((staleExecuted?.state.nextRunAtMs ?? 0) > nowMs).toBe(true);
});
it("applies timeoutSeconds to manual cron.run isolated executions", async () => {
vi.useFakeTimers();
try {
const store = opsRegressionFixtures.makeStorePath();
const scheduledAt = Date.parse("2026-02-15T13:00:00.000Z");
const job = createIsolatedRegressionJob({
id: "manual-timeout",
name: "manual timeout",
scheduledAt,
schedule: { kind: "every", everyMs: 60_000, anchorMs: scheduledAt },
payload: { kind: "agentTurn", message: "work", timeoutSeconds: FAST_TIMEOUT_SECONDS },
state: { nextRunAtMs: scheduledAt },
});
await writeCronJobs(store.storePath, [job]);
const abortAwareRunner = createAbortAwareIsolatedRunner();
const state = createCronServiceState({
cronEnabled: false,
storePath: store.storePath,
log: noopLogger,
enqueueSystemEvent: vi.fn(),
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob: abortAwareRunner.runIsolatedAgentJob,
});
const resultPromise = run(state, job.id, "force");
await abortAwareRunner.waitForStart();
await vi.advanceTimersByTimeAsync(Math.ceil(FAST_TIMEOUT_SECONDS * 1_000) + 10);
const result = await resultPromise;
expect(result).toEqual({ ok: true, ran: true });
expect(abortAwareRunner.getObservedAbortSignal()?.aborted).toBe(true);
const updated = state.store?.jobs.find((entry) => entry.id === job.id);
expect(updated?.state.lastStatus).toBe("error");
expect(updated?.state.lastError).toContain("timed out");
expect(updated?.state.runningAtMs).toBeUndefined();
} finally {
vi.useRealTimers();
}
});
it("#17554: run() clears stale runningAtMs and executes the job", async () => {
const store = opsRegressionFixtures.makeStorePath();
const now = Date.parse("2026-02-06T10:05:00.000Z");
const staleRunningAtMs = now - 2 * 60 * 60 * 1000 - 1;
await writeCronJobs(store.storePath, [
{
id: "stale-running",
name: "stale-running",
enabled: true,
createdAtMs: now - 3_600_000,
updatedAtMs: now - 3_600_000,
schedule: { kind: "at", at: new Date(now - 60_000).toISOString() },
sessionTarget: "main",
wakeMode: "now",
payload: { kind: "systemEvent", text: "stale-running" },
state: {
runningAtMs: staleRunningAtMs,
lastRunAtMs: now - 3_600_000,
lastStatus: "ok",
nextRunAtMs: now - 60_000,
},
},
]);
const enqueueSystemEvent = vi.fn();
const state = createCronServiceState({
cronEnabled: true,
storePath: store.storePath,
log: noopLogger,
nowMs: () => now,
enqueueSystemEvent,
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob: vi.fn().mockResolvedValue({ status: "ok", summary: "ok" }),
});
const result = await run(state, "stale-running", "force");
expect(result).toEqual({ ok: true, ran: true });
expect(enqueueSystemEvent).toHaveBeenCalledWith(
"stale-running",
expect.objectContaining({ agentId: undefined }),
);
});
it("queues manual cron.run requests behind the cron execution lane", async () => {
vi.useRealTimers();
clearCommandLane(CommandLane.Cron);
setCommandLaneConcurrency(CommandLane.Cron, 1);
const store = opsRegressionFixtures.makeStorePath();
const dueAt = Date.parse("2026-02-06T10:05:02.000Z");
const first = createDueIsolatedJob({ id: "queued-first", nowMs: dueAt, nextRunAtMs: dueAt });
const second = createDueIsolatedJob({
id: "queued-second",
nowMs: dueAt,
nextRunAtMs: dueAt,
});
await fs.writeFile(
store.storePath,
JSON.stringify({ version: 1, jobs: [first, second] }),
"utf-8",
);
let now = dueAt;
let activeRuns = 0;
let peakActiveRuns = 0;
const firstStarted = createDeferred<void>();
const firstRun = createDeferred<{ status: "ok"; summary: string }>();
const secondRun = createDeferred<{ status: "ok"; summary: string }>();
const secondStarted = createDeferred<void>();
const bothFinished = createDeferred<void>();
const runIsolatedAgentJob = vi.fn(async (params: { job: { id: string } }) => {
activeRuns += 1;
peakActiveRuns = Math.max(peakActiveRuns, activeRuns);
if (params.job.id === first.id) {
firstStarted.resolve();
}
if (params.job.id === second.id) {
secondStarted.resolve();
}
try {
const result =
params.job.id === first.id ? await firstRun.promise : await secondRun.promise;
now += 10;
return result;
} finally {
activeRuns -= 1;
}
});
const state = createCronServiceState({
cronEnabled: true,
storePath: store.storePath,
cronConfig: { maxConcurrentRuns: 1 },
log: noopLogger,
nowMs: () => now,
enqueueSystemEvent: vi.fn(),
requestHeartbeatNow: vi.fn(),
runIsolatedAgentJob,
onEvent: (evt) => {
if (evt.action === "finished" && evt.jobId === second.id && evt.status === "ok") {
bothFinished.resolve();
}
},
});
const firstAck = await enqueueRun(state, first.id, "force");
const secondAck = await enqueueRun(state, second.id, "force");
expect(firstAck).toEqual({ ok: true, enqueued: true, runId: expect.any(String) });
expect(secondAck).toEqual({ ok: true, enqueued: true, runId: expect.any(String) });
await firstStarted.promise;
expect(runIsolatedAgentJob.mock.calls[0]?.[0]).toMatchObject({ job: { id: first.id } });
expect(peakActiveRuns).toBe(1);
firstRun.resolve({ status: "ok", summary: "first queued run" });
await secondStarted.promise;
expect(runIsolatedAgentJob).toHaveBeenCalledTimes(2);
expect(runIsolatedAgentJob.mock.calls[1]?.[0]).toMatchObject({ job: { id: second.id } });
expect(peakActiveRuns).toBe(1);
secondRun.resolve({ status: "ok", summary: "second queued run" });
await bothFinished.promise;
await waitForActiveTasks(5_000);
const jobs = state.store?.jobs ?? [];
expect(jobs.find((job) => job.id === first.id)?.state.lastStatus).toBe("ok");
expect(jobs.find((job) => job.id === second.id)?.state.lastStatus).toBe("ok");
clearCommandLane(CommandLane.Cron);
});
it("logs unexpected queued manual run background failures once", async () => {
vi.useRealTimers();
clearCommandLane(CommandLane.Cron);
setCommandLaneConcurrency(CommandLane.Cron, 1);
const dueAt = Date.parse("2026-02-06T10:05:03.000Z");
const job = createDueIsolatedJob({ id: "queued-failure", nowMs: dueAt, nextRunAtMs: dueAt });
const errorLogged = createDeferred<void>();
const log = {
...noopLogger,
error: vi.fn<(payload: unknown, message?: string) => void>(() => {
errorLogged.resolve();
}),
};
const badStore = `${opsRegressionFixtures.makeStorePath().storePath}.dir`;
await fs.mkdir(badStore, { recursive: true });
const state = createRunningCronServiceState({
storePath: badStore,
log,
nowMs: () => dueAt,
jobs: [job],
});
const result = await enqueueRun(state, job.id, "force");
expect(result).toEqual({ ok: true, enqueued: true, runId: expect.any(String) });
await errorLogged.promise;
expect(log.error).toHaveBeenCalledTimes(1);
expect(log.error.mock.calls[0]?.[1]).toBe(
"cron: queued manual run background execution failed",
);
clearCommandLane(CommandLane.Cron);
});
});
¤ Dauer der Verarbeitung: 0.2 Sekunden
(vorverarbeitet am 2026-04-27)
¤
*© Formatika GbR, Deutschland
|
|