@absurd-sqlite/sdk 0.2.0-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -0
- package/dist/absurd-types.d.ts +109 -0
- package/dist/absurd-types.d.ts.map +1 -0
- package/dist/absurd-types.js +2 -0
- package/dist/absurd-types.js.map +1 -0
- package/dist/cjs/absurd-types.js +2 -0
- package/dist/cjs/index.js +18 -0
- package/dist/cjs/package.json +1 -0
- package/dist/cjs/sqlite-types.js +2 -0
- package/dist/cjs/sqlite.js +117 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +15 -0
- package/dist/index.js.map +1 -0
- package/dist/sqlite-types.d.ts +23 -0
- package/dist/sqlite-types.d.ts.map +1 -0
- package/dist/sqlite-types.js +2 -0
- package/dist/sqlite-types.js.map +1 -0
- package/dist/sqlite.d.ts +11 -0
- package/dist/sqlite.d.ts.map +1 -0
- package/dist/sqlite.js +114 -0
- package/dist/sqlite.js.map +1 -0
- package/package.json +51 -0
- package/src/absurd-types.ts +149 -0
- package/src/index.ts +46 -0
- package/src/sqlite-types.ts +35 -0
- package/src/sqlite.ts +162 -0
- package/test/basic.test.ts +505 -0
- package/test/events.test.ts +207 -0
- package/test/hooks.test.ts +347 -0
- package/test/idempotent.test.ts +195 -0
- package/test/index.test.ts +92 -0
- package/test/retry.test.ts +389 -0
- package/test/setup.ts +563 -0
- package/test/sqlite.test.ts +85 -0
- package/test/step.test.ts +259 -0
- package/test/worker.test.ts +193 -0
- package/tsconfig.build.json +9 -0
- package/tsconfig.cjs.json +11 -0
- package/tsconfig.json +20 -0
- package/typedoc.json +11 -0
- package/vitest.config.ts +11 -0
|
@@ -0,0 +1,505 @@
|
|
|
1
|
+
import {
|
|
2
|
+
describe,
|
|
3
|
+
test,
|
|
4
|
+
assert,
|
|
5
|
+
expect,
|
|
6
|
+
beforeAll,
|
|
7
|
+
afterEach,
|
|
8
|
+
vi,
|
|
9
|
+
} from "vitest";
|
|
10
|
+
import { createTestAbsurd, randomName, type TestContext } from "./setup.js";
|
|
11
|
+
import type { Absurd } from "../src/index.js";
|
|
12
|
+
import { EventEmitter, once } from "events";
|
|
13
|
+
|
|
14
|
+
describe("Basic SDK Operations", () => {
|
|
15
|
+
let ctx: TestContext;
|
|
16
|
+
let absurd: Absurd;
|
|
17
|
+
|
|
18
|
+
beforeAll(async () => {
|
|
19
|
+
ctx = await createTestAbsurd(randomName("test_queue"));
|
|
20
|
+
absurd = ctx.absurd;
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
afterEach(async () => {
|
|
24
|
+
await ctx.cleanupTasks();
|
|
25
|
+
await ctx.setFakeNow(null);
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
describe("Queue management", () => {
|
|
29
|
+
test("create, list, and drop queue", async () => {
|
|
30
|
+
const queueName = randomName("test_queue");
|
|
31
|
+
await absurd.createQueue(queueName);
|
|
32
|
+
|
|
33
|
+
let queues = await absurd.listQueues();
|
|
34
|
+
expect(queues).toContain(queueName);
|
|
35
|
+
|
|
36
|
+
const storage = await ctx.getQueueStorageState(queueName);
|
|
37
|
+
expect(storage.exists).toBe(true);
|
|
38
|
+
expect(storage.tables).toEqual(
|
|
39
|
+
expect.arrayContaining([
|
|
40
|
+
"absurd_tasks",
|
|
41
|
+
"absurd_runs",
|
|
42
|
+
"absurd_events",
|
|
43
|
+
"absurd_waits",
|
|
44
|
+
"absurd_checkpoints",
|
|
45
|
+
"absurd_queues",
|
|
46
|
+
])
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
await absurd.dropQueue(queueName);
|
|
50
|
+
|
|
51
|
+
queues = await absurd.listQueues();
|
|
52
|
+
expect(queues).not.toContain(queueName);
|
|
53
|
+
|
|
54
|
+
const storageAfterDrop = await ctx.getQueueStorageState(queueName);
|
|
55
|
+
expect(storageAfterDrop.exists).toBe(false);
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
describe("Task spawning", () => {
|
|
60
|
+
test("spawn with maxAttempts override", async () => {
|
|
61
|
+
absurd.registerTask<{ shouldFail: boolean }>(
|
|
62
|
+
{ name: "test-max-attempts", defaultMaxAttempts: 5 },
|
|
63
|
+
async () => {
|
|
64
|
+
throw new Error("Always fails");
|
|
65
|
+
},
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
const { taskID } = await absurd.spawn("test-max-attempts", undefined, {
|
|
69
|
+
maxAttempts: 2,
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
await absurd.workBatch("test-worker-attempts", 60, 1);
|
|
73
|
+
await absurd.workBatch("test-worker-attempts", 60, 1);
|
|
74
|
+
|
|
75
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
76
|
+
state: "failed",
|
|
77
|
+
attempts: 2,
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
test("rejects spawning unregistered task without queue override", async () => {
|
|
82
|
+
await expect(
|
|
83
|
+
absurd.spawn("unregistered-task", { value: 1 }),
|
|
84
|
+
).rejects.toThrowError(
|
|
85
|
+
'Task "unregistered-task" is not registered. Provide options.queue when spawning unregistered tasks.',
|
|
86
|
+
);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
test("rejects spawning registered task on mismatched queue", async () => {
|
|
90
|
+
const taskName = "registered-queue-task";
|
|
91
|
+
const otherQueue = randomName("other_queue");
|
|
92
|
+
|
|
93
|
+
absurd.registerTask(
|
|
94
|
+
{ name: taskName, queue: ctx.queueName },
|
|
95
|
+
async () => ({ success: true }),
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
await expect(
|
|
99
|
+
absurd.spawn(taskName, undefined, { queue: otherQueue }),
|
|
100
|
+
).rejects.toThrowError(
|
|
101
|
+
`Task "${taskName}" is registered for queue "${ctx.queueName}" but spawn requested queue "${otherQueue}".`,
|
|
102
|
+
);
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
describe("Task claiming", () => {
|
|
107
|
+
test("claim tasks with various batch sizes", async () => {
|
|
108
|
+
await ctx.cleanupTasks();
|
|
109
|
+
|
|
110
|
+
absurd.registerTask<{ id: number }>(
|
|
111
|
+
{ name: "test-claim" },
|
|
112
|
+
async (params) => {
|
|
113
|
+
return params;
|
|
114
|
+
},
|
|
115
|
+
);
|
|
116
|
+
|
|
117
|
+
const spawned = await Promise.all(
|
|
118
|
+
[1, 2, 3].map((id) => absurd.spawn("test-claim", { id })),
|
|
119
|
+
);
|
|
120
|
+
|
|
121
|
+
// Test batch claim
|
|
122
|
+
const claimed = await absurd.claimTasks({
|
|
123
|
+
batchSize: 3,
|
|
124
|
+
claimTimeout: 60,
|
|
125
|
+
workerId: "test-worker",
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
expect(claimed.length).toBe(3);
|
|
129
|
+
expect(claimed.map((c) => c.task_id).sort()).toEqual(
|
|
130
|
+
spawned.map((s) => s.taskID).sort(),
|
|
131
|
+
);
|
|
132
|
+
|
|
133
|
+
// Should now be "running"
|
|
134
|
+
expect((await ctx.getTask(spawned[0].taskID))?.state).toBe("running");
|
|
135
|
+
|
|
136
|
+
// There should be none to claim
|
|
137
|
+
expect(
|
|
138
|
+
await absurd.claimTasks({
|
|
139
|
+
batchSize: 10,
|
|
140
|
+
claimTimeout: 60,
|
|
141
|
+
workerId: "test-worker-empty",
|
|
142
|
+
}),
|
|
143
|
+
).toEqual([]);
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
describe("State transitions", () => {
|
|
148
|
+
test("scheduleRun moves run between running and sleeping", async () => {
|
|
149
|
+
await ctx.cleanupTasks();
|
|
150
|
+
const baseTime = new Date("2024-04-01T10:00:00Z");
|
|
151
|
+
await ctx.setFakeNow(baseTime);
|
|
152
|
+
|
|
153
|
+
absurd.registerTask<{ step: string }>(
|
|
154
|
+
{ name: "schedule-task" },
|
|
155
|
+
async () => {
|
|
156
|
+
return { done: true };
|
|
157
|
+
},
|
|
158
|
+
);
|
|
159
|
+
|
|
160
|
+
const { runID } = await absurd.spawn("schedule-task", { step: "start" });
|
|
161
|
+
const [claim] = await absurd.claimTasks({
|
|
162
|
+
workerId: "worker-1",
|
|
163
|
+
claimTimeout: 120,
|
|
164
|
+
});
|
|
165
|
+
expect(claim.run_id).toBe(runID);
|
|
166
|
+
|
|
167
|
+
const wakeAt = new Date(baseTime.getTime() + 5 * 60 * 1000);
|
|
168
|
+
await ctx.scheduleRun(runID, wakeAt);
|
|
169
|
+
|
|
170
|
+
const scheduledRun = await ctx.getRun(runID);
|
|
171
|
+
expect(scheduledRun).toMatchObject({
|
|
172
|
+
state: "sleeping",
|
|
173
|
+
available_at: wakeAt,
|
|
174
|
+
wake_event: null,
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
const scheduledTask = await ctx.getTask(claim.task_id);
|
|
178
|
+
expect(scheduledTask?.state).toBe("sleeping");
|
|
179
|
+
|
|
180
|
+
await ctx.setFakeNow(wakeAt);
|
|
181
|
+
const [resumed] = await absurd.claimTasks({
|
|
182
|
+
workerId: "worker-1",
|
|
183
|
+
claimTimeout: 120,
|
|
184
|
+
});
|
|
185
|
+
expect(resumed.run_id).toBe(runID);
|
|
186
|
+
expect(resumed.attempt).toBe(1);
|
|
187
|
+
|
|
188
|
+
const resumedRun = await ctx.getRun(runID);
|
|
189
|
+
expect(resumedRun).toMatchObject({
|
|
190
|
+
state: "running",
|
|
191
|
+
started_at: wakeAt,
|
|
192
|
+
});
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
test("claim timeout releases run to a new worker", async () => {
|
|
196
|
+
await ctx.cleanupTasks();
|
|
197
|
+
const baseTime = new Date("2024-04-02T09:00:00Z");
|
|
198
|
+
await ctx.setFakeNow(baseTime);
|
|
199
|
+
|
|
200
|
+
absurd.registerTask<{ step: string }>(
|
|
201
|
+
{ name: "lease-task" },
|
|
202
|
+
async () => {
|
|
203
|
+
return { ok: true };
|
|
204
|
+
},
|
|
205
|
+
);
|
|
206
|
+
|
|
207
|
+
const { taskID } = await absurd.spawn("lease-task", { step: "attempt" });
|
|
208
|
+
const [claim] = await absurd.claimTasks({
|
|
209
|
+
workerId: "worker-a",
|
|
210
|
+
claimTimeout: 30,
|
|
211
|
+
});
|
|
212
|
+
expect(claim.task_id).toBe(taskID);
|
|
213
|
+
|
|
214
|
+
const running = await ctx.getRun(claim.run_id);
|
|
215
|
+
expect(running).toMatchObject({
|
|
216
|
+
state: "running",
|
|
217
|
+
claimed_by: "worker-a",
|
|
218
|
+
claim_expires_at: new Date(baseTime.getTime() + 30 * 1000),
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
await ctx.setFakeNow(new Date(baseTime.getTime() + 5 * 60 * 1000));
|
|
222
|
+
const [reclaim] = await absurd.claimTasks({
|
|
223
|
+
workerId: "worker-b",
|
|
224
|
+
claimTimeout: 45,
|
|
225
|
+
});
|
|
226
|
+
expect(reclaim.run_id).not.toBe(claim.run_id);
|
|
227
|
+
expect(reclaim.attempt).toBe(2);
|
|
228
|
+
|
|
229
|
+
const expiredRun = await ctx.getRun(claim.run_id);
|
|
230
|
+
expect(expiredRun?.state).toBe("failed");
|
|
231
|
+
expect(expiredRun?.failure_reason).toMatchObject({
|
|
232
|
+
name: "$ClaimTimeout",
|
|
233
|
+
workerId: "worker-a",
|
|
234
|
+
attempt: 1,
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
const newRun = await ctx.getRun(reclaim.run_id);
|
|
238
|
+
expect(newRun).toMatchObject({
|
|
239
|
+
state: "running",
|
|
240
|
+
claimed_by: "worker-b",
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
const taskRow = await ctx.getTask(taskID);
|
|
244
|
+
expect(taskRow).toMatchObject({
|
|
245
|
+
state: "running",
|
|
246
|
+
attempts: 2,
|
|
247
|
+
});
|
|
248
|
+
});
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
describe("Cleanup maintenance", () => {
|
|
252
|
+
test("cleanup tasks and events respect TTLs", async () => {
|
|
253
|
+
await ctx.cleanupTasks();
|
|
254
|
+
const base = new Date("2024-03-01T08:00:00Z");
|
|
255
|
+
await ctx.setFakeNow(base);
|
|
256
|
+
|
|
257
|
+
absurd.registerTask<{ step: string }>({ name: "cleanup" }, async () => {
|
|
258
|
+
return { status: "done" };
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
const { runID } = await absurd.spawn("cleanup", { step: "start" });
|
|
262
|
+
const [claim] = await absurd.claimTasks({
|
|
263
|
+
workerId: "worker-clean",
|
|
264
|
+
claimTimeout: 60,
|
|
265
|
+
});
|
|
266
|
+
expect(claim.run_id).toBe(runID);
|
|
267
|
+
|
|
268
|
+
const finishTime = new Date(base.getTime() + 10 * 60 * 1000);
|
|
269
|
+
await ctx.setFakeNow(finishTime);
|
|
270
|
+
await ctx.completeRun(runID, { status: "done" });
|
|
271
|
+
|
|
272
|
+
await absurd.emitEvent("cleanup-event", { kind: "notify" });
|
|
273
|
+
|
|
274
|
+
const runRow = await ctx.getRun(runID);
|
|
275
|
+
expect(runRow).toMatchObject({
|
|
276
|
+
claimed_by: "worker-clean",
|
|
277
|
+
claim_expires_at: new Date(base.getTime() + 60 * 1000),
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
const beforeTTL = new Date(finishTime.getTime() + 30 * 60 * 1000);
|
|
281
|
+
await ctx.setFakeNow(beforeTTL);
|
|
282
|
+
const beforeTasks = await ctx.cleanupTasksByTTL(3600, 10);
|
|
283
|
+
expect(beforeTasks).toBe(0);
|
|
284
|
+
const beforeEvents = await ctx.cleanupEventsByTTL(3600, 10);
|
|
285
|
+
expect(beforeEvents).toBe(0);
|
|
286
|
+
|
|
287
|
+
const later = new Date(finishTime.getTime() + 26 * 60 * 60 * 1000);
|
|
288
|
+
await ctx.setFakeNow(later);
|
|
289
|
+
const deletedTasks = await ctx.cleanupTasksByTTL(3600, 10);
|
|
290
|
+
expect(deletedTasks).toBe(1);
|
|
291
|
+
const deletedEvents = await ctx.cleanupEventsByTTL(3600, 10);
|
|
292
|
+
expect(deletedEvents).toBe(1);
|
|
293
|
+
|
|
294
|
+
const remainingTasks = await ctx.getRemainingTasksCount();
|
|
295
|
+
expect(remainingTasks).toBe(0);
|
|
296
|
+
const remainingEvents = await ctx.getRemainingEventsCount();
|
|
297
|
+
expect(remainingEvents).toBe(0);
|
|
298
|
+
});
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
describe("Task state transitions", () => {
|
|
302
|
+
test("task transitions through all states: pending -> running -> completed", async () => {
|
|
303
|
+
absurd.registerTask<{ value: number }>(
|
|
304
|
+
{ name: "test-task-complete" },
|
|
305
|
+
async (params, ctx) => {
|
|
306
|
+
const doubled = await ctx.step("double", async () => {
|
|
307
|
+
return params.value * 2;
|
|
308
|
+
});
|
|
309
|
+
return { doubled };
|
|
310
|
+
},
|
|
311
|
+
);
|
|
312
|
+
|
|
313
|
+
// Spawn: transitions to pending
|
|
314
|
+
const { taskID } = await absurd.spawn("test-task-complete", {
|
|
315
|
+
value: 21,
|
|
316
|
+
});
|
|
317
|
+
expect((await ctx.getTask(taskID))?.state).toBe("pending");
|
|
318
|
+
|
|
319
|
+
// Process with workBatch: transitions pending -> running -> completed
|
|
320
|
+
await absurd.workBatch("test-worker-complete", 60, 1);
|
|
321
|
+
|
|
322
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
323
|
+
state: "completed",
|
|
324
|
+
attempts: 1,
|
|
325
|
+
completed_payload: { doubled: 42 },
|
|
326
|
+
});
|
|
327
|
+
});
|
|
328
|
+
|
|
329
|
+
test("task transitions to sleeping state when suspended (waiting for event)", async () => {
|
|
330
|
+
const eventName = randomName("suspend_event");
|
|
331
|
+
absurd.registerTask(
|
|
332
|
+
{ name: "test-task-suspend" },
|
|
333
|
+
async (params, ctx) => {
|
|
334
|
+
return { received: await ctx.awaitEvent(eventName) };
|
|
335
|
+
},
|
|
336
|
+
);
|
|
337
|
+
|
|
338
|
+
const { taskID } = await absurd.spawn("test-task-suspend", undefined);
|
|
339
|
+
|
|
340
|
+
// Process task (suspends waiting for event)
|
|
341
|
+
await absurd.workBatch("test-worker-suspend", 60, 1);
|
|
342
|
+
expect((await ctx.getTask(taskID))?.state).toBe("sleeping");
|
|
343
|
+
|
|
344
|
+
// Emit event and resume
|
|
345
|
+
await absurd.emitEvent(eventName, { data: "wakeup" });
|
|
346
|
+
await absurd.workBatch("test-worker-suspend", 60, 1);
|
|
347
|
+
|
|
348
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
349
|
+
state: "completed",
|
|
350
|
+
completed_payload: { received: { data: "wakeup" } },
|
|
351
|
+
});
|
|
352
|
+
});
|
|
353
|
+
|
|
354
|
+
test("task transitions to failed state after all retries exhausted", async () => {
|
|
355
|
+
absurd.registerTask(
|
|
356
|
+
{ name: "test-task-fail", defaultMaxAttempts: 2 },
|
|
357
|
+
async () => {
|
|
358
|
+
throw new Error("Task intentionally failed");
|
|
359
|
+
},
|
|
360
|
+
);
|
|
361
|
+
|
|
362
|
+
const { taskID, runID: firstRunID } = await absurd.spawn(
|
|
363
|
+
"test-task-fail",
|
|
364
|
+
undefined,
|
|
365
|
+
);
|
|
366
|
+
|
|
367
|
+
// First attempt fails (task: pending, run: failed)
|
|
368
|
+
await absurd.workBatch("test-worker-fail", 60, 1);
|
|
369
|
+
expect((await ctx.getRun(firstRunID))?.state).toBe("failed");
|
|
370
|
+
expect((await ctx.getTask(taskID))?.state).toBe("pending");
|
|
371
|
+
// Second attempt fails (task: failed, run: failed)
|
|
372
|
+
await absurd.workBatch("test-worker-fail", 60, 1);
|
|
373
|
+
expect((await ctx.getTask(taskID))?.state).toBe("failed");
|
|
374
|
+
expect(await ctx.getRun(firstRunID)).toMatchObject({
|
|
375
|
+
state: "failed",
|
|
376
|
+
attempt: 1,
|
|
377
|
+
failure_reason: expect.objectContaining({
|
|
378
|
+
message: "Task intentionally failed",
|
|
379
|
+
}),
|
|
380
|
+
});
|
|
381
|
+
});
|
|
382
|
+
});
|
|
383
|
+
|
|
384
|
+
describe("Event system", () => {
|
|
385
|
+
test("task receives event emitted before task was spawned", async () => {
|
|
386
|
+
absurd.registerTask<{ eventName: string }, { received: any }>(
|
|
387
|
+
{ name: "test-cached-event" },
|
|
388
|
+
async (params, ctx) => {
|
|
389
|
+
const payload = await ctx.awaitEvent(params.eventName);
|
|
390
|
+
return { received: payload };
|
|
391
|
+
},
|
|
392
|
+
);
|
|
393
|
+
|
|
394
|
+
const eventName = randomName("test_event");
|
|
395
|
+
|
|
396
|
+
await absurd.emitEvent(eventName, { data: "cached-payload" });
|
|
397
|
+
|
|
398
|
+
const { taskID } = await absurd.spawn("test-cached-event", { eventName });
|
|
399
|
+
|
|
400
|
+
await absurd.workBatch("test-worker-cached", 60, 1);
|
|
401
|
+
|
|
402
|
+
const taskInfo = await ctx.getTask(taskID);
|
|
403
|
+
assert(taskInfo);
|
|
404
|
+
expect(taskInfo).toMatchObject({
|
|
405
|
+
state: "completed",
|
|
406
|
+
completed_payload: { received: { data: "cached-payload" } },
|
|
407
|
+
});
|
|
408
|
+
});
|
|
409
|
+
});
|
|
410
|
+
|
|
411
|
+
describe("Batch processing", () => {
|
|
412
|
+
test("workBatch processes multiple tasks", async () => {
|
|
413
|
+
absurd.registerTask<{ id: number }>(
|
|
414
|
+
{ name: "test-work-batch" },
|
|
415
|
+
async (params) => {
|
|
416
|
+
return { result: `task-${params.id}` };
|
|
417
|
+
},
|
|
418
|
+
);
|
|
419
|
+
|
|
420
|
+
const tasks = await Promise.all(
|
|
421
|
+
[1, 2, 3].map((id) => absurd.spawn("test-work-batch", { id })),
|
|
422
|
+
);
|
|
423
|
+
|
|
424
|
+
await absurd.workBatch("test-worker-batch", 60, 5);
|
|
425
|
+
|
|
426
|
+
for (let i = 0; i < tasks.length; i++) {
|
|
427
|
+
const task = tasks[i];
|
|
428
|
+
expect(await ctx.getTask(task.taskID)).toMatchObject({
|
|
429
|
+
state: "completed",
|
|
430
|
+
completed_payload: { result: `task-${i + 1}` },
|
|
431
|
+
});
|
|
432
|
+
}
|
|
433
|
+
});
|
|
434
|
+
|
|
435
|
+
test("workBatch handles mixed success and failure", async () => {
|
|
436
|
+
absurd.registerTask<{ fail: boolean }>(
|
|
437
|
+
{ name: "mixed", defaultMaxAttempts: 1 },
|
|
438
|
+
async (params) => {
|
|
439
|
+
if (params.fail) {
|
|
440
|
+
throw new Error("Task failed in batch");
|
|
441
|
+
}
|
|
442
|
+
return { result: "success" };
|
|
443
|
+
},
|
|
444
|
+
);
|
|
445
|
+
|
|
446
|
+
const bad = await absurd.spawn("mixed", {
|
|
447
|
+
fail: true,
|
|
448
|
+
});
|
|
449
|
+
const ok = await absurd.spawn("mixed", {
|
|
450
|
+
fail: false,
|
|
451
|
+
});
|
|
452
|
+
|
|
453
|
+
await absurd.workBatch("mixed", 60, 2);
|
|
454
|
+
|
|
455
|
+
expect((await ctx.getTask(bad.taskID))?.state).toBe("failed");
|
|
456
|
+
expect((await ctx.getTask(ok.taskID))?.state).toBe("completed");
|
|
457
|
+
});
|
|
458
|
+
});
|
|
459
|
+
|
|
460
|
+
describe("Heartbeat", () => {
|
|
461
|
+
test("heartbeat extends claim timeout", async () => {
|
|
462
|
+
const gate = new EventEmitter();
|
|
463
|
+
const baseTime = new Date("2025-01-01T00:00:00Z");
|
|
464
|
+
await ctx.setFakeNow(baseTime);
|
|
465
|
+
|
|
466
|
+
const claimTimeout = 60;
|
|
467
|
+
const extension = 120;
|
|
468
|
+
|
|
469
|
+
absurd.registerTask(
|
|
470
|
+
{ name: "heartbeat-extends" },
|
|
471
|
+
async (params: { extension: number }, taskCtx) => {
|
|
472
|
+
gate.emit("task-started");
|
|
473
|
+
await once(gate, "heartbeat");
|
|
474
|
+
await taskCtx.heartbeat(params.extension);
|
|
475
|
+
},
|
|
476
|
+
);
|
|
477
|
+
|
|
478
|
+
const { runID } = await absurd.spawn("heartbeat-extends", {
|
|
479
|
+
extension,
|
|
480
|
+
});
|
|
481
|
+
|
|
482
|
+
const getExpiresAt = async (runID: string) => {
|
|
483
|
+
const run = await ctx.getRun(runID);
|
|
484
|
+
return run?.claim_expires_at ? run.claim_expires_at.getTime() : 0;
|
|
485
|
+
};
|
|
486
|
+
|
|
487
|
+
absurd.workBatch("test-worker", claimTimeout);
|
|
488
|
+
|
|
489
|
+
await once(gate, "task-started");
|
|
490
|
+
await vi.waitFor(async () => {
|
|
491
|
+
expect(await getExpiresAt(runID)).toBe(
|
|
492
|
+
baseTime.getTime() + claimTimeout * 1000,
|
|
493
|
+
);
|
|
494
|
+
});
|
|
495
|
+
|
|
496
|
+
gate.emit("heartbeat");
|
|
497
|
+
|
|
498
|
+
await vi.waitFor(async () => {
|
|
499
|
+
expect(await getExpiresAt(runID)).toBe(
|
|
500
|
+
baseTime.getTime() + extension * 1000,
|
|
501
|
+
);
|
|
502
|
+
});
|
|
503
|
+
});
|
|
504
|
+
});
|
|
505
|
+
});
|
|
@@ -0,0 +1,207 @@
|
|
|
1
|
+
import { describe, test, expect, beforeAll, afterEach } from "vitest";
|
|
2
|
+
import { createTestAbsurd, randomName, type TestContext } from "./setup.js";
|
|
3
|
+
import type { Absurd } from "../src/index.js";
|
|
4
|
+
import { TimeoutError } from "absurd-sdk";
|
|
5
|
+
|
|
6
|
+
describe("Event system", () => {
|
|
7
|
+
let ctx: TestContext;
|
|
8
|
+
let absurd: Absurd;
|
|
9
|
+
|
|
10
|
+
beforeAll(async () => {
|
|
11
|
+
ctx = await createTestAbsurd(randomName("event_queue"));
|
|
12
|
+
absurd = ctx.absurd;
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
afterEach(async () => {
|
|
16
|
+
await ctx.cleanupTasks();
|
|
17
|
+
await ctx.setFakeNow(null);
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
test("await and emit event flow", async () => {
|
|
21
|
+
const eventName = randomName("test_event");
|
|
22
|
+
|
|
23
|
+
absurd.registerTask({ name: "waiter" }, async (params, ctx) => {
|
|
24
|
+
const payload = await ctx.awaitEvent(eventName, { timeout: 60 });
|
|
25
|
+
return { received: payload };
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
const { taskID, runID } = await absurd.spawn("waiter", { step: 1 });
|
|
29
|
+
|
|
30
|
+
// Start processing, task should suspend waiting for event
|
|
31
|
+
await absurd.workBatch("worker1", 60, 1);
|
|
32
|
+
|
|
33
|
+
const sleepingRun = await ctx.getRun(runID);
|
|
34
|
+
expect(sleepingRun).toMatchObject({
|
|
35
|
+
state: "sleeping",
|
|
36
|
+
wake_event: eventName,
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
// Emit event
|
|
40
|
+
const payload = { value: 42 };
|
|
41
|
+
await absurd.emitEvent(eventName, payload);
|
|
42
|
+
|
|
43
|
+
// Task should now be pending
|
|
44
|
+
const pendingRun = await ctx.getRun(runID);
|
|
45
|
+
expect(pendingRun?.state).toBe("pending");
|
|
46
|
+
|
|
47
|
+
// Resume and complete
|
|
48
|
+
await absurd.workBatch("worker1", 60, 1);
|
|
49
|
+
|
|
50
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
51
|
+
state: "completed",
|
|
52
|
+
completed_payload: { received: payload },
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
test("event emitted before await is cached and retrieved", async () => {
|
|
57
|
+
const eventName = randomName("cached_event");
|
|
58
|
+
const payload = { data: "pre-emitted" };
|
|
59
|
+
|
|
60
|
+
// Emit event before task even exists
|
|
61
|
+
await absurd.emitEvent(eventName, payload);
|
|
62
|
+
|
|
63
|
+
absurd.registerTask({ name: "late-waiter" }, async (params, ctx) => {
|
|
64
|
+
const received = await ctx.awaitEvent(eventName);
|
|
65
|
+
return { received };
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
const { taskID } = await absurd.spawn("late-waiter", undefined);
|
|
69
|
+
|
|
70
|
+
// Should complete immediately with cached event
|
|
71
|
+
await absurd.workBatch("worker1", 60, 1);
|
|
72
|
+
|
|
73
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
74
|
+
state: "completed",
|
|
75
|
+
completed_payload: { received: payload },
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
test("awaitEvent with timeout expires and wakes task", async () => {
|
|
80
|
+
const eventName = randomName("timeout_event");
|
|
81
|
+
const baseTime = new Date("2024-05-01T10:00:00Z");
|
|
82
|
+
const timeoutSeconds = 600;
|
|
83
|
+
|
|
84
|
+
await ctx.setFakeNow(baseTime);
|
|
85
|
+
|
|
86
|
+
absurd.registerTask({ name: "timeout-waiter" }, async (_params, ctx) => {
|
|
87
|
+
try {
|
|
88
|
+
const payload = await ctx.awaitEvent(eventName, {
|
|
89
|
+
timeout: timeoutSeconds,
|
|
90
|
+
});
|
|
91
|
+
return { timedOut: false, result: payload };
|
|
92
|
+
} catch (err) {
|
|
93
|
+
if (err instanceof TimeoutError) {
|
|
94
|
+
return { timedOut: true, result: null };
|
|
95
|
+
}
|
|
96
|
+
throw err;
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
const { taskID, runID } = await absurd.spawn("timeout-waiter", undefined);
|
|
101
|
+
await absurd.workBatch("worker1", 120, 1);
|
|
102
|
+
|
|
103
|
+
const waitCountBefore = await ctx.getWaitsCount();
|
|
104
|
+
expect(waitCountBefore).toBe(1);
|
|
105
|
+
|
|
106
|
+
const sleepingRun = await ctx.getRun(runID);
|
|
107
|
+
expect(sleepingRun).toMatchObject({
|
|
108
|
+
state: "sleeping",
|
|
109
|
+
wake_event: eventName,
|
|
110
|
+
});
|
|
111
|
+
const expectedWake = new Date(baseTime.getTime() + timeoutSeconds * 1000);
|
|
112
|
+
expect(sleepingRun?.available_at?.getTime()).toBe(expectedWake.getTime());
|
|
113
|
+
|
|
114
|
+
await ctx.setFakeNow(new Date(expectedWake.getTime() + 1000));
|
|
115
|
+
await absurd.workBatch("worker1", 120, 1);
|
|
116
|
+
|
|
117
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
118
|
+
state: "completed",
|
|
119
|
+
completed_payload: { timedOut: true, result: null },
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
const waitCountAfter = await ctx.getWaitsCount();
|
|
123
|
+
expect(waitCountAfter).toBe(0);
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
test("multiple tasks can await the same event", async () => {
|
|
127
|
+
const eventName = randomName("broadcast_event");
|
|
128
|
+
|
|
129
|
+
absurd.registerTask<{ taskNum: number }>(
|
|
130
|
+
{ name: "multi-waiter" },
|
|
131
|
+
async (params, ctx) => {
|
|
132
|
+
const payload = await ctx.awaitEvent(eventName);
|
|
133
|
+
return { taskNum: params.taskNum, received: payload };
|
|
134
|
+
},
|
|
135
|
+
);
|
|
136
|
+
|
|
137
|
+
const tasks = await Promise.all([
|
|
138
|
+
absurd.spawn("multi-waiter", { taskNum: 1 }),
|
|
139
|
+
absurd.spawn("multi-waiter", { taskNum: 2 }),
|
|
140
|
+
absurd.spawn("multi-waiter", { taskNum: 3 }),
|
|
141
|
+
]);
|
|
142
|
+
|
|
143
|
+
// All tasks suspend waiting for event
|
|
144
|
+
await absurd.workBatch("worker1", 60, 10);
|
|
145
|
+
|
|
146
|
+
for (const task of tasks) {
|
|
147
|
+
expect((await ctx.getTask(task.taskID))?.state).toBe("sleeping");
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Emit event once
|
|
151
|
+
const payload = { data: "broadcast" };
|
|
152
|
+
await absurd.emitEvent(eventName, payload);
|
|
153
|
+
|
|
154
|
+
// All tasks should resume and complete
|
|
155
|
+
await absurd.workBatch("worker1", 60, 10);
|
|
156
|
+
|
|
157
|
+
for (let i = 0; i < tasks.length; i++) {
|
|
158
|
+
const task = tasks[i];
|
|
159
|
+
expect(await ctx.getTask(task.taskID)).toMatchObject({
|
|
160
|
+
state: "completed",
|
|
161
|
+
completed_payload: { taskNum: i + 1, received: payload },
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
test("awaitEvent timeout does not recreate wait on resume", async () => {
|
|
167
|
+
const eventName = randomName("timeout_no_loop");
|
|
168
|
+
const baseTime = new Date("2024-05-02T11:00:00Z");
|
|
169
|
+
await ctx.setFakeNow(baseTime);
|
|
170
|
+
|
|
171
|
+
absurd.registerTask({ name: "timeout-no-loop" }, async (_params, ctx) => {
|
|
172
|
+
try {
|
|
173
|
+
await ctx.awaitEvent(eventName, { stepName: "wait", timeout: 10 });
|
|
174
|
+
return { stage: "unexpected" };
|
|
175
|
+
} catch (err) {
|
|
176
|
+
if (err instanceof TimeoutError) {
|
|
177
|
+
const payload = await ctx.awaitEvent(eventName, {
|
|
178
|
+
stepName: "wait",
|
|
179
|
+
timeout: 10,
|
|
180
|
+
});
|
|
181
|
+
return { stage: "resumed", payload };
|
|
182
|
+
}
|
|
183
|
+
throw err;
|
|
184
|
+
}
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
const { taskID, runID } = await absurd.spawn("timeout-no-loop", undefined);
|
|
188
|
+
await absurd.workBatch("worker-timeout", 60, 1);
|
|
189
|
+
|
|
190
|
+
const waitCount = await ctx.getWaitsCount();
|
|
191
|
+
expect(waitCount).toBe(1);
|
|
192
|
+
|
|
193
|
+
await ctx.setFakeNow(new Date(baseTime.getTime() + 15 * 1000));
|
|
194
|
+
await absurd.workBatch("worker-timeout", 60, 1);
|
|
195
|
+
|
|
196
|
+
const waitCountAfter = await ctx.getWaitsCount();
|
|
197
|
+
expect(waitCountAfter).toBe(0);
|
|
198
|
+
|
|
199
|
+
const run = await ctx.getRun(runID);
|
|
200
|
+
expect(run?.state).toBe("completed");
|
|
201
|
+
|
|
202
|
+
expect(await ctx.getTask(taskID)).toMatchObject({
|
|
203
|
+
state: "completed",
|
|
204
|
+
completed_payload: { stage: "resumed", payload: null },
|
|
205
|
+
});
|
|
206
|
+
});
|
|
207
|
+
});
|