Files
market_screener/tests/MarketCallRepository.test.ts
T
2026-06-05 23:34:25 -04:00

141 lines
5.1 KiB
TypeScript

/**
* Unit tests for MarketCallRepository (SQLite-backed).
* Each test gets its own in-memory database so tests are fully isolated.
*/
import { test } from 'node:test';
import assert from 'node:assert/strict';
import BetterSqlite3 from 'better-sqlite3';
import { MarketCallRepository } from '../server/repositories/MarketCallRepository';
// ── Helpers ───────────────────────────────────────────────────────────────────
const DDL = `
CREATE TABLE IF NOT EXISTS market_calls (
id TEXT PRIMARY KEY,
title TEXT NOT NULL, quarter TEXT NOT NULL, date TEXT NOT NULL,
thesis TEXT NOT NULL, tickers TEXT NOT NULL, snapshot TEXT NOT NULL,
created_at TEXT NOT NULL
);
`;
function makeRepo(): MarketCallRepository {
const db = new BetterSqlite3(':memory:');
db.pragma('journal_mode = WAL');
db.exec(DDL);
return new MarketCallRepository(db);
}
// ── Fixtures ──────────────────────────────────────────────────────────────────
const CALL_INPUT = {
title: 'Rate pivot play',
quarter: 'Q3 2025',
thesis: 'Fed cuts expected — rotate into duration and growth.',
tickers: ['TLT', 'QQQ'],
};
// ── Tests ─────────────────────────────────────────────────────────────────────
test('list() returns empty array on fresh db', () => {
assert.deepEqual(makeRepo().list(), []);
});
test('create() returns call with id, createdAt, and correct fields', () => {
const call = makeRepo().create(CALL_INPUT);
assert.ok(call.id, 'id should be set');
assert.ok(call.createdAt, 'createdAt should be set');
assert.equal(call.title, CALL_INPUT.title);
assert.equal(call.quarter, CALL_INPUT.quarter);
assert.equal(call.thesis, CALL_INPUT.thesis);
assert.deepEqual(call.tickers, CALL_INPUT.tickers);
});
test('create() persists — list() returns the created call', () => {
const repo = makeRepo();
repo.create(CALL_INPUT);
assert.equal(repo.list().length, 1);
assert.equal(repo.list()[0].title, CALL_INPUT.title);
});
test('list() returns calls newest-first', () => {
const repo = makeRepo();
const db = (repo as any).db as BetterSqlite3.Database;
// Insert two rows with distinct created_at values directly
db.prepare(
`INSERT INTO market_calls (id, title, quarter, date, thesis, tickers, snapshot, created_at)
VALUES (?,?,?,?,?,?,?,?)`,
).run('old-id', 'First', 'Q1', '2025-01-01', 'A', '[]', '{}', '2025-01-01T00:00:00.000Z');
db.prepare(
`INSERT INTO market_calls (id, title, quarter, date, thesis, tickers, snapshot, created_at)
VALUES (?,?,?,?,?,?,?,?)`,
).run('new-id', 'Second', 'Q1', '2025-01-02', 'B', '[]', '{}', '2025-01-02T00:00:00.000Z');
const list = repo.list();
assert.equal(list[0].id, 'new-id', 'newer call should be first');
assert.equal(list[1].id, 'old-id', 'older call should be second');
});
test('get() returns the call by id', () => {
const repo = makeRepo();
const call = repo.create(CALL_INPUT);
const found = repo.get(call.id);
assert.ok(found);
assert.equal(found!.id, call.id);
});
test('get() returns null for unknown id', () => {
assert.equal(makeRepo().get('no-such-id'), null);
});
test('delete() removes the call and returns true', () => {
const repo = makeRepo();
const call = repo.create(CALL_INPUT);
assert.equal(repo.delete(call.id), true);
assert.equal(repo.list().length, 0);
assert.equal(repo.get(call.id), null);
});
test('delete() returns false for unknown id', () => {
assert.equal(makeRepo().delete('no-such-id'), false);
});
test('delete() only removes the targeted call', () => {
const repo = makeRepo();
const a = repo.create({ ...CALL_INPUT, title: 'Keep me' });
const b = repo.create({ ...CALL_INPUT, title: 'Delete me' });
repo.delete(b.id);
const list = repo.list();
assert.equal(list.length, 1);
assert.equal(list[0].id, a.id);
});
test('create() stores snapshot when provided', () => {
const repo = makeRepo();
const snapshot = { TLT: { price: 95.5, signal: '✅ Strong Buy' } };
const call = repo.create({ ...CALL_INPUT, snapshot } as any);
assert.deepEqual(repo.get(call.id)!.snapshot, snapshot);
});
test('create() sets default date when not provided', () => {
const call = makeRepo().create(CALL_INPUT);
assert.match(call.date, /^\d{4}-\d{2}-\d{2}$/);
});
test('create() uses provided date', () => {
const call = makeRepo().create({ ...CALL_INPUT, date: '2025-03-15' });
assert.equal(call.date, '2025-03-15');
});
test('concurrent writes: two rapid creates both persist (SQLite WAL is concurrency-safe)', () => {
const repo = makeRepo();
const a = repo.create({ ...CALL_INPUT, title: 'A' });
const b = repo.create({ ...CALL_INPUT, title: 'B' });
const list = repo.list();
assert.equal(list.length, 2);
const ids = new Set(list.map((c) => c.id));
assert.ok(ids.has(a.id));
assert.ok(ids.has(b.id));
});