wallets-api / server /src /index.js
z1amez's picture
v.1
2dddd1f
"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' });
}
});