Files
Web-Dev-For-Beginners/7-bank-project/api/server.js
2025-05-21 16:10:28 +00:00

238 lines
6.8 KiB
JavaScript

const express = require('express');
const bodyParser = require('body-parser');
const cors = require('cors')
const crypto = require('crypto');
const pkg = require('./package.json');
// App constants
const port = process.env.PORT || 5000;
const apiPrefix = '/api';
// Store data in-memory, not suited for production use!
const db = {
test: {
user: 'test',
currency: '$',
description: `Test account`,
balance: 75,
transactions: [
{ id: '1', date: '2020-10-01', object: 'Pocket money', amount: 50 },
{ id: '2', date: '2020-10-03', object: 'Book', amount: -10 },
{ id: '3', date: '2020-10-04', object: 'Sandwich', amount: -5 }
],
}
};
// Create the Express app & setup middlewares
const app = express();
app.use(bodyParser.urlencoded({ extended: true }));
app.use(bodyParser.json());
// SECURITY NOTE: In a production environment, you should specify exact origins
// rather than allowing all localhost and 127.x.x.x addresses
// For more information: https://owasp.org/www-project-cheat-sheets/cheatsheets/Cross-Site_Request_Forgery_Prevention_Cheat_Sheet.html
app.use(cors({
origin: /http:\/\/(127(\.\d){3}|localhost)/,
methods: ['GET', 'POST', 'DELETE'],
allowedHeaders: ['Content-Type', 'Authorization']
}));
app.options('*', cors());
// SECURITY NOTE: In production, you should:
// 1. Enforce HTTPS with a middleware like:
// app.use((req, res, next) => {
// if (!req.secure && req.get('x-forwarded-proto') !== 'https' && process.env.NODE_ENV === 'production') {
// return res.redirect('https://' + req.get('host') + req.url);
// }
// next();
// });
//
// 2. Implement proper authentication with JWT, OAuth, etc.
// For more information: https://cheatsheetseries.owasp.org/cheatsheets/Authentication_Cheat_Sheet.html
//
// 3. Add rate limiting to prevent abuse:
// const rateLimit = require('express-rate-limit');
// app.use(rateLimit({
// windowMs: 15 * 60 * 1000, // 15 minutes
// max: 100 // limit each IP to 100 requests per windowMs
// }));
// ***************************************************************************
// Configure routes
const router = express.Router();
// Get server infos
router.get('/', (req, res) => {
return res.send(`${pkg.description} v${pkg.version}`);
});
// ----------------------------------------------
// Create an account
router.post('/accounts', (req, res) => {
// Check mandatory request parameters
if (!req.body.user || !req.body.currency) {
return res.status(400).json({ error: 'Missing parameters' });
}
// Check if account already exists
if (db[req.body.user]) {
return res.status(409).json({ error: 'User already exists' });
}
// Convert balance to number if needed
let balance = req.body.balance;
if (balance && typeof balance !== 'number') {
balance = parseFloat(balance);
if (isNaN(balance)) {
return res.status(400).json({ error: 'Balance must be a number' });
}
}
// Create account
const account = {
user: req.body.user,
currency: req.body.currency,
description: req.body.description || `${req.body.user}'s budget`,
balance: balance || 0,
transactions: [],
};
db[req.body.user] = account;
return res.status(201).json(account);
});
// ----------------------------------------------
// Get all data for the specified account
router.get('/accounts/:user', (req, res) => {
const account = db[req.params.user];
// Check if account exists
if (!account) {
return res.status(404).json({ error: 'User does not exist' });
}
return res.json(account);
});
// ----------------------------------------------
// Remove specified account
router.delete('/accounts/:user', (req, res) => {
const account = db[req.params.user];
// Check if account exists
if (!account) {
return res.status(404).json({ error: 'User does not exist' });
}
// Removed account
delete db[req.params.user];
res.sendStatus(204);
});
// ----------------------------------------------
// Add a transaction to a specific account
router.post('/accounts/:user/transactions', (req, res) => {
const account = db[req.params.user];
// Check if account exists
if (!account) {
return res.status(404).json({ error: 'User does not exist' });
}
// Check mandatory requests parameters
if (!req.body.date || !req.body.object || !req.body.amount) {
return res.status(400).json({ error: 'Missing parameters' });
}
// SECURITY NOTE: In production, validate date format to prevent injection attacks
// Example: use a library like date-fns to validate proper date format
const dateRegex = /^\d{4}-\d{2}-\d{2}$/;
if (!dateRegex.test(req.body.date)) {
return res.status(400).json({ error: 'Invalid date format. Use YYYY-MM-DD' });
}
// SECURITY NOTE: In production, sanitize object field to prevent XSS
// Example: use a library like DOMPurify
if (typeof req.body.object !== 'string' || req.body.object.length > 100) {
return res.status(400).json({ error: 'Object must be a string with max length of 100' });
}
// Convert amount to number if needed
let amount = req.body.amount;
if (amount && typeof amount !== 'number') {
amount = parseFloat(amount);
}
// Check that amount is a valid number
if (amount && isNaN(amount)) {
return res.status(400).json({ error: 'Amount must be a number' });
}
// Generates an ID for the transaction
const id = crypto
.createHash('md5')
.update(req.body.date + req.body.object + req.body.amount)
.digest('hex');
// Check that transaction does not already exist
if (account.transactions.some((transaction) => transaction.id === id)) {
return res.status(409).json({ error: 'Transaction already exists' });
}
// Add transaction
const transaction = {
id,
date: req.body.date,
object: req.body.object,
amount,
};
account.transactions.push(transaction);
// Update balance
account.balance += transaction.amount;
return res.status(201).json(transaction);
});
// ----------------------------------------------
// Remove specified transaction from account
router.delete('/accounts/:user/transactions/:id', (req, res) => {
const account = db[req.params.user];
// Check if account exists
if (!account) {
return res.status(404).json({ error: 'User does not exist' });
}
const transactionIndex = account.transactions.findIndex(
(transaction) => transaction.id === req.params.id
);
// Check if transaction exists
if (transactionIndex === -1) {
return res.status(404).json({ error: 'Transaction does not exist' });
}
// Remove transaction
account.transactions.splice(transactionIndex, 1);
res.sendStatus(204);
});
// ***************************************************************************
// Add 'api` prefix to all routes
app.use(apiPrefix, router);
// Start the server
app.listen(port, () => {
console.log(`Server listening on port ${port}`);
});