"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const express_1 = __importDefault(require("express")); const cors_1 = __importDefault(require("cors")); const db_1 = require("./db"); const telegram_1 = require("./telegram"); require("express-async-errors"); const zod_1 = require("zod"); const xlsx = __importStar(require("xlsx")); const app = (0, express_1.default)(); const PORT = 3001; app.use((0, cors_1.default)()); app.use(express_1.default.json()); app.get('/api/rates', (req, res) => { res.json((0, telegram_1.getCurrentRates)()); }); // Initialize Telegram Client in background (0, telegram_1.initTelegramAuth)().catch(console.error); // Add Validation Schemas const transactionSchema = zod_1.z.object({ type: zod_1.z.enum(['income', 'expense', 'transfer']), amount: zod_1.z.number().positive(), currency: zod_1.z.string().min(2), wallet_id: zod_1.z.number().int().positive(), to_wallet_id: zod_1.z.number().int().positive().nullable().optional(), category: zod_1.z.string().optional(), note: zod_1.z.string().optional(), date: zod_1.z.string(), country_id: zod_1.z.string().optional() }); const exchangeSchema = zod_1.z.object({ from_amount: zod_1.z.number().positive(), from_currency: zod_1.z.string().min(2), to_amount: zod_1.z.number().positive(), to_currency: zod_1.z.string().min(2), rate: zod_1.z.number().positive(), note: zod_1.z.string().optional(), date: zod_1.z.string() }); const loanSchema = zod_1.z.object({ person: zod_1.z.string().min(1), type: zod_1.z.enum(['borrowed_from_me', 'owed_by_me']), amount: zod_1.z.number().positive(), currency: zod_1.z.string().min(2), note: zod_1.z.string().optional(), date: zod_1.z.string() }); app.get('/api/wallets', async (req, res) => { const db = await (0, db_1.getDb)(); const wallets = await db.all('SELECT * FROM wallets'); res.json(wallets); }); app.get('/api/transactions', async (req, res) => { const db = await (0, db_1.getDb)(); const transactions = await db.all('SELECT * FROM transactions ORDER BY date DESC'); res.json(transactions); }); app.post('/api/transactions', async (req, res) => { const validatedData = transactionSchema.parse(req.body); const { type, amount, currency, wallet_id, to_wallet_id, category, note, date, country_id } = validatedData; const db = await (0, db_1.getDb)(); const result = await db.run(`INSERT INTO transactions (type, amount, currency, wallet_id, to_wallet_id, category, note, date, country_id) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`, [type, amount, currency, wallet_id, to_wallet_id, category, note, date, country_id]); res.json({ id: result.lastID }); }); app.get('/api/exchanges', async (req, res) => { const db = await (0, db_1.getDb)(); const exchanges = await db.all('SELECT * FROM exchanges ORDER BY date DESC'); res.json(exchanges); }); app.post('/api/exchanges', async (req, res) => { const validatedData = exchangeSchema.parse(req.body); const { from_amount, from_currency, to_amount, to_currency, rate, note, date } = validatedData; const db = await (0, db_1.getDb)(); const result = await db.run(`INSERT INTO exchanges (from_amount, from_currency, to_amount, to_currency, rate, note, date) VALUES (?, ?, ?, ?, ?, ?, ?)`, [from_amount, from_currency, to_amount, to_currency, rate, note, date]); res.json({ id: result.lastID }); }); app.get('/api/loans', async (req, res) => { const db = await (0, db_1.getDb)(); const loans = await db.all('SELECT * FROM loans ORDER BY date DESC'); res.json(loans); }); app.post('/api/loans', async (req, res) => { const validatedData = loanSchema.parse(req.body); const { person, type, amount, currency, note, date } = validatedData; const db = await (0, db_1.getDb)(); const result = await db.run(`INSERT INTO loans (person, type, amount, currency, note, date) VALUES (?, ?, ?, ?, ?, ?)`, [person, type, amount, currency, note, date]); res.json({ id: result.lastID }); }); app.get('/api/export', async (req, res) => { const db = await (0, db_1.getDb)(); const transactions = await db.all('SELECT * FROM transactions'); const wallets = await db.all('SELECT * FROM wallets'); const exchanges = await db.all('SELECT * FROM exchanges'); const loans = await db.all('SELECT * FROM loans'); const wb = xlsx.utils.book_new(); xlsx.utils.book_append_sheet(wb, xlsx.utils.json_to_sheet(transactions), 'Transactions'); xlsx.utils.book_append_sheet(wb, xlsx.utils.json_to_sheet(wallets), 'Wallets'); xlsx.utils.book_append_sheet(wb, xlsx.utils.json_to_sheet(exchanges), 'Exchanges'); xlsx.utils.book_append_sheet(wb, xlsx.utils.json_to_sheet(loans), 'Loans'); const buf = xlsx.write(wb, { type: 'buffer', bookType: 'xlsx' }); res.set('Content-Disposition', 'attachment; filename="backup.xlsx"'); res.type('application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'); res.send(buf); }); app.get('/api/analytics/dashboard', async (req, res) => { const db = await (0, db_1.getDb)(); // Using SQLite to aggregate the balances // Since currency conversion changes, we return the raw grouped data and calculate total dynamically const walletBalancesRaw = await db.all(` SELECT w.id, w.name, w.type, w.currency, COALESCE(SUM(CASE WHEN t.type = 'income' THEN t.amount ELSE 0 END), 0) - COALESCE(SUM(CASE WHEN t.type = 'expense' THEN t.amount ELSE 0 END), 0) - COALESCE(SUM(CASE WHEN t.type = 'transfer' AND t.wallet_id = w.id THEN t.amount ELSE 0 END), 0) + COALESCE(SUM(CASE WHEN t.type = 'transfer' AND t.to_wallet_id = w.id THEN t.amount ELSE 0 END), 0) as balance FROM wallets w LEFT JOIN transactions t ON t.wallet_id = w.id OR t.to_wallet_id = w.id GROUP BY w.id `); // Fetch exchanges to merge into currency pools later on the frontend const exchangesRaw = await db.all('SELECT from_amount, from_currency, to_amount, to_currency FROM exchanges'); // Fetch loans for net worth calculation const loansRaw = await db.all('SELECT type, amount, paid, currency FROM loans'); // We also need transactions for the recent activity feed const recentActivity = await db.all('SELECT * FROM transactions ORDER BY date DESC LIMIT 5'); res.json({ walletBalancesRaw, exchangesRaw, loansRaw, recentActivity }); }); app.listen(PORT, () => { console.log(`Backend running on http://localhost:${PORT}`); }); // Global Error Handler app.use((err, req, res, next) => { if (err instanceof zod_1.z.ZodError) { res.status(400).json({ error: 'Validation Error', details: err.issues }); } else { console.error('Unhandled Error:', err); res.status(500).json({ error: err.message || 'Internal Server Error' }); } });