BillTracker/tests/exportRicher.test.js

89 lines
3.6 KiB
JavaScript

'use strict';
// Batch 4: richer export — full JSON assembly (money in dollars) and the payments
// export's date-range vs year filtering.
const test = require('node:test');
const assert = require('node:assert/strict');
const os = require('node:os');
const path = require('node:path');
const fs = require('node:fs');
const dbPath = path.join(os.tmpdir(), `bill-tracker-export-richer-${process.pid}.sqlite`);
process.env.DB_PATH = dbPath;
const { getDb, closeDb } = require('../db/database');
const exportRouter = require('../routes/export');
const { getUserExportData } = exportRouter;
let userId, billId;
function callExport(query) {
const layer = exportRouter.stack.find(l => l.route?.path === '/' && l.route.methods.get);
const handler = layer.route.stack[0].handle;
return new Promise((resolve) => {
const headers = {};
const req = { query, user: { id: userId, role: 'user' } };
const res = {
statusCode: 200,
setHeader(k, v) { headers[k] = v; },
status(c) { this.statusCode = c; return this; },
json(d) { resolve({ status: this.statusCode, headers, json: d }); },
send(body) { resolve({ status: this.statusCode, headers, body }); },
};
handler(req, res);
});
}
test.before(() => {
const db = getDb();
userId = db.prepare("INSERT INTO users (username, password_hash, role, active) VALUES ('export-user','x','user',1)").run().lastInsertRowid;
billId = db.prepare("INSERT INTO bills (user_id, name, due_day, expected_amount, active) VALUES (?, 'Rent', 1, 120000, 1)").run(userId).lastInsertRowid;
const pay = db.prepare("INSERT INTO payments (bill_id, amount, paid_date, payment_source) VALUES (?, ?, ?, 'manual')");
pay.run(billId, 8500, '2025-06-15'); // prior year
pay.run(billId, 9000, '2026-01-10'); // in range
pay.run(billId, 9500, '2026-07-20'); // out of range, same year
});
test.after(() => {
closeDb();
for (const s of ['', '-wal', '-shm']) { try { fs.unlinkSync(dbPath + s); } catch {} }
});
test('getUserExportData assembles user data with money in dollars', () => {
const data = getUserExportData(userId);
assert.equal(data.bills.length, 1);
assert.equal(data.bills[0].expected_amount, 1200, 'expected_amount is dollars (fromCents)');
assert.equal(data.payments.length, 3);
const amounts = data.payments.map(p => p.amount).sort((a, b) => a - b);
assert.deepEqual(amounts, [85, 90, 95], 'payment amounts in dollars');
assert.equal(data.metadata.counts.payments, 3);
});
test('payments export by year returns that year only', async () => {
const { status, json } = await callExport({ year: '2026', format: 'json' });
assert.equal(status, 200);
assert.equal(json.count, 2, 'both 2026 payments');
assert.equal(json.range, '2026');
});
test('payments export by date range filters to the range', async () => {
const { status, json } = await callExport({ from: '2026-01-01', to: '2026-06-30', format: 'json' });
assert.equal(status, 200);
assert.equal(json.count, 1, 'only the Jan 10 payment');
assert.equal(json.payments[0].paid_amount, 90);
});
test('CSV export sets a filename derived from the range', async () => {
const { status, headers, body } = await callExport({ from: '2026-01-01', to: '2026-12-31', format: 'csv' });
assert.equal(status, 200);
assert.match(headers['Content-Disposition'], /bills-2026-01-01_to_2026-12-31\.csv/);
assert.match(body, /^Date,Bill,Category/);
});
test('invalid date range is rejected', async () => {
const bad = await callExport({ from: '2026-12-31', to: '2026-01-01' });
assert.equal(bad.status, 400);
const badFmt = await callExport({ from: 'nope', to: '2026-01-01' });
assert.equal(badFmt.status, 400);
});