BillTracker/routes/categories.js

132 lines
4.2 KiB
JavaScript
Raw Normal View History

2026-05-03 19:51:57 -05:00
const express = require('express');
2026-05-09 13:03:36 -05:00
const { standardizeError } = require('../middleware/errorFormatter');
2026-05-03 19:51:57 -05:00
const router = express.Router();
const { getDb, ensureUserDefaultCategories } = require('../db/database');
// GET /api/categories
router.get('/', (req, res) => {
const db = getDb();
ensureUserDefaultCategories(req.user.id);
2026-05-04 16:38:03 -05:00
const categories = db.prepare(`
SELECT id, user_id, name, created_at, updated_at
FROM categories
WHERE user_id = ?
ORDER BY name COLLATE NOCASE ASC
`).all(req.user.id);
const billsByCategory = db.prepare(`
SELECT
b.id,
b.category_id,
b.name,
b.active,
b.expected_amount,
b.due_day,
COUNT(p.id) AS payment_count,
COALESCE(SUM(p.amount), 0) AS total_paid,
MAX(p.paid_date) AS last_paid_date
FROM bills b
LEFT JOIN payments p
ON p.bill_id = b.id
AND p.deleted_at IS NULL
WHERE b.user_id = ?
AND b.category_id = ?
GROUP BY b.id
ORDER BY b.active DESC, b.due_day ASC, b.name COLLATE NOCASE ASC
`);
const shaped = categories.map(category => {
const bills = billsByCategory.all(req.user.id, category.id).map(bill => ({
...bill,
active: !!bill.active,
payment_count: Number(bill.payment_count || 0),
total_paid: Number(bill.total_paid || 0),
last_paid_date: bill.last_paid_date || null,
}));
const activeBillCount = bills.filter(bill => bill.active).length;
const inactiveBillCount = bills.length - activeBillCount;
const paymentCount = bills.reduce((sum, bill) => sum + bill.payment_count, 0);
return {
...category,
bill_count: activeBillCount,
active_bill_count: activeBillCount,
inactive_bill_count: inactiveBillCount,
payment_count: paymentCount,
bill_names: bills.map(bill => bill.name),
bills,
};
});
res.json(shaped);
2026-05-03 19:51:57 -05:00
});
// POST /api/categories
router.post('/', (req, res) => {
const db = getDb();
const { name } = req.body;
2026-05-09 13:03:36 -05:00
if (!name) return res.status(400).json(standardizeError('name is required', 'VALIDATION_ERROR', 'name'));
2026-05-03 19:51:57 -05:00
try {
const result = db.prepare('INSERT INTO categories (user_id, name) VALUES (?, ?)').run(req.user.id, name.trim());
const created = db.prepare('SELECT * FROM categories WHERE id = ?').get(result.lastInsertRowid);
res.status(201).json(created);
} catch (e) {
if (e.message.includes('UNIQUE')) {
2026-05-09 13:03:36 -05:00
return res.status(409).json(standardizeError('Category already exists', 'CONFLICT', 'name'));
2026-05-03 19:51:57 -05:00
}
throw e;
}
});
// PUT /api/categories/:id
router.put('/:id', (req, res) => {
const db = getDb();
const { name } = req.body;
2026-05-09 13:03:36 -05:00
if (!name) return res.status(400).json(standardizeError('name is required', 'VALIDATION_ERROR', 'name'));
2026-05-03 19:51:57 -05:00
const cat = db.prepare('SELECT id FROM categories WHERE id = ? AND user_id = ?').get(req.params.id, req.user.id);
2026-05-09 13:03:36 -05:00
if (!cat) return res.status(404).json(standardizeError('Category not found', 'NOT_FOUND', 'id'));
2026-05-03 19:51:57 -05:00
try {
db.prepare("UPDATE categories SET name = ?, updated_at = datetime('now') WHERE id = ? AND user_id = ?")
.run(name.trim(), req.params.id, req.user.id);
res.json(db.prepare('SELECT * FROM categories WHERE id = ? AND user_id = ?').get(req.params.id, req.user.id));
} catch (e) {
if (e.message.includes('UNIQUE')) {
2026-05-09 13:03:36 -05:00
return res.status(409).json(standardizeError('Category already exists', 'CONFLICT', 'name'));
2026-05-03 19:51:57 -05:00
}
throw e;
}
});
// DELETE /api/categories/:id
router.delete('/:id', (req, res) => {
const db = getDb();
const cat = db.prepare('SELECT id FROM categories WHERE id = ? AND user_id = ?').get(req.params.id, req.user.id);
2026-05-09 13:03:36 -05:00
if (!cat) return res.status(404).json(standardizeError('Category not found', 'NOT_FOUND', 'id'));
2026-05-04 20:12:57 -05:00
const deleteCategory = db.transaction(() => {
const bills = db.prepare(`
UPDATE bills
SET category_id = NULL, updated_at = datetime('now')
WHERE category_id = ? AND user_id = ?
`).run(req.params.id, req.user.id);
const deleted = db.prepare('DELETE FROM categories WHERE id = ? AND user_id = ?')
.run(req.params.id, req.user.id);
return {
deleted: deleted.changes,
uncategorized_bills: bills.changes,
};
});
const result = deleteCategory();
res.json({ success: true, ...result });
2026-05-03 19:51:57 -05:00
});
module.exports = router;