You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
taskbot/tests/unit/services/response-queue.cleanup.test.ts

125 lines
5.3 KiB
TypeScript

import { describe, test, expect, beforeAll, afterAll, beforeEach } from 'bun:test';
import { Database } from 'bun:sqlite';
import { initializeDatabase } from '../../../src/db';
import { ResponseQueue } from '../../../src/services/response-queue';
let testDb: Database;
let originalDbInstance: Database;
function toIso(dt: Date): string {
return dt.toISOString().replace('T', ' ').replace('Z', '');
}
describe('ResponseQueue cleanup/retention', () => {
beforeAll(() => {
testDb = new Database(':memory:');
initializeDatabase(testDb);
originalDbInstance = (ResponseQueue as any).dbInstance;
(ResponseQueue as any).dbInstance = testDb;
});
afterAll(() => {
(ResponseQueue as any).dbInstance = originalDbInstance;
testDb.close();
});
beforeEach(() => {
testDb.exec('DELETE FROM response_queue');
});
test('does not delete queued or processing items regardless of age', async () => {
const old = toIso(new Date(2000, 0, 1));
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`).run('u1','m1','queued', old);
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`).run('u2','m2','processing', old);
testDb.prepare(`INSERT INTO response_queue (recipient, message, status) VALUES (?,?,?)`).run('u3','m3','queued');
const res = await (ResponseQueue as any).runCleanupOnce(new Date());
expect(res.totalDeleted).toBe(0);
const counts = testDb.query(`SELECT status, COUNT(*) as c FROM response_queue GROUP BY status ORDER BY status`).all() as any[];
const map = Object.fromEntries(counts.map(r => [r.status, r.c]));
expect(map['queued']).toBe(2);
expect(map['processing']).toBe(1);
});
test('deletes sent older than 14 days but keeps recent', async () => {
const now = new Date();
const days14Ago = new Date(now.getTime() - 14 * 24 * 60 * 60 * 1000);
const days13Ago = new Date(now.getTime() - 13 * 24 * 60 * 60 * 1000);
const thresholdExact = toIso(days14Ago); // exact boundary
// exactly at threshold (should NOT delete because comparison is strict <)
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run('u1','m1','sent', thresholdExact);
// older than threshold
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run('u2','m2','sent', toIso(new Date(days14Ago.getTime() - 1000)));
// newer than threshold
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run('u3','m3','sent', toIso(days13Ago));
const res = await (ResponseQueue as any).runCleanupOnce(now);
expect(res.deletedSent).toBe(1);
const rows = testDb.query(`SELECT status, updated_at FROM response_queue WHERE status='sent' ORDER BY updated_at`).all() as any[];
expect(rows.length).toBe(2);
expect(rows[0].updated_at).toBe(thresholdExact);
expect(rows[1].updated_at).toBe(toIso(days13Ago));
});
test('deletes failed older than 30 days but keeps newer', async () => {
const now = new Date();
const days30Ago = new Date(now.getTime() - 30 * 24 * 60 * 60 * 1000);
const days29Ago = new Date(now.getTime() - 29 * 24 * 60 * 60 * 1000);
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run('u1','m1','failed', toIso(new Date(days30Ago.getTime() - 1000)));
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run('u2','m2','failed', toIso(days29Ago));
const res = await (ResponseQueue as any).runCleanupOnce(now);
expect(res.deletedFailed).toBe(1);
const rows = testDb.query(`SELECT COUNT(*) as c FROM response_queue WHERE status='failed'`).get() as any;
expect(rows.c).toBe(1);
});
test('batch deletes large sets in multiple passes', async () => {
(ResponseQueue as any).CLEANUP_BATCH = 500; // reduce for test
const old = toIso(new Date(2000, 0, 1));
const total = 1200;
const insert = testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`);
testDb.transaction(() => {
for (let i = 0; i < total; i++) {
insert.run(`u${i}`, `m${i}`, 'sent', old);
}
})();
const res = await (ResponseQueue as any).runCleanupOnce(new Date());
expect(res.deletedSent).toBe(total);
const count = testDb.query(`SELECT COUNT(*) as c FROM response_queue WHERE status='sent'`).get() as any;
expect(count.c).toBe(0);
});
test('concurrent cleanup calls do not overlap', async () => {
const old = toIso(new Date(2000, 0, 1));
for (let i = 0; i < 50; i++) {
testDb.prepare(`INSERT INTO response_queue (recipient, message, status, updated_at) VALUES (?,?,?,?)`)
.run(`u${i}`, `m${i}`, 'sent', old);
}
// Trigger two cleanups concurrently
const [r1, r2] = await Promise.all([
(ResponseQueue as any).runCleanupOnce(new Date()),
(ResponseQueue as any).runCleanupOnce(new Date()),
]);
const total = (r1.totalDeleted || 0) + (r2.totalDeleted || 0);
expect(total).toBe(50); // no double-deletes
const remain = testDb.query(`SELECT COUNT(*) as c FROM response_queue`).get() as any;
expect(remain.c).toBe(0);
});
});