feat(jobs): echo worker + CLI bootstrap

Job queue starts only in the CLI gate (not inside createApp), so tests
manage their own queue lifecycle. waitForJob() takes a (name, id) pair
to match pg-boss v10's getJobById signature.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
root
2026-06-01 03:28:06 +10:00
parent 17a13dddb8
commit 53ffd705c4
5 changed files with 56 additions and 9 deletions

View File

@@ -6,18 +6,14 @@ export async function stopBoss() {
try { await pool.query('DROP SCHEMA IF EXISTS pgboss CASCADE'); } catch { /* ignore */ }
}
export async function waitForJob(id, { timeoutMs = 5_000 } = {}) {
export async function waitForJob(name, id, { timeoutMs = 5_000 } = {}) {
const boss = queue.instance();
if (!boss) throw new Error('queue not started');
const start = Date.now();
while (Date.now() - start < timeoutMs) {
const j = await boss.getJobById(id);
if (!j) {
await new Promise(r => setTimeout(r, 50));
continue;
}
if (['completed','failed','cancelled','expired'].includes(j.state)) return j;
const j = await boss.getJobById(name, id, { includeArchive: true });
if (j && ['completed','failed','cancelled','expired'].includes(j.state)) return j;
await new Promise(r => setTimeout(r, 50));
}
throw new Error(`job ${id} did not finish in ${timeoutMs} ms`);
throw new Error(`job ${name} ${id} did not finish in ${timeoutMs} ms`);
}

View File

@@ -0,0 +1,22 @@
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { resetDb } from '../../helpers/db.js';
import { migrateUp } from '../../../lib/db/migrate.js';
import { stopBoss, waitForJob } from '../../helpers/boss.js';
import * as queue from '../../../lib/jobs/queue.js';
import { registerWorkers } from '../../../lib/jobs/index.js';
beforeEach(async () => {
await resetDb(); await migrateUp();
await queue.start();
await registerWorkers();
});
afterEach(async () => { await stopBoss(); });
describe('echo worker', () => {
it('completes with the expected output', async () => {
const id = await queue.enqueue('echo', { ping: 7 });
const j = await waitForJob('echo', id);
expect(j.state).toBe('completed');
expect(j.output).toEqual({ pong: 7 });
});
});