'use strict'; const path = require('path'); const bcrypt = require('bcryptjs'); const crypto = require('node:crypto'); const express = require('express'); const expressSession = require('express-session'); const debug = require('debug')('Server'); const Util = require('./Util'); const ServerError = require('./ServerError'); const WireGuard = require('../services/WireGuard'); const { PORT, WEBUI_HOST, RELEASE, PASSWORD, LANG, UI_DETAILED_STATS, } = require('../config'); module.exports = class Server { constructor() { // Express this.app = express() .disable('etag') .use('/', express.static(path.join(__dirname, '..', 'www'))) .use(express.json()) .use(expressSession({ secret: crypto.randomBytes(256).toString('hex'), resave: true, saveUninitialized: true, cookie: { httpOnly: true, }, })) .get('/api/release', (Util.promisify(async () => { return RELEASE; }))) .get('/api/lang', (Util.promisify(async () => { return LANG; }))) .get('/api/ui-detailed-stats', (Util.promisify(async () => { return UI_DETAILED_STATS === 'true'; }))) // Authentication .get('/api/session', Util.promisify(async (req) => { const requiresPassword = !!process.env.PASSWORD; const authenticated = requiresPassword ? !!(req.session && req.session.authenticated) : true; return { requiresPassword, authenticated, }; })) .post('/api/session', Util.promisify(async (req) => { const { password, } = req.body; if (typeof password !== 'string') { throw new ServerError('Missing: Password', 401); } if (password !== PASSWORD) { throw new ServerError('Incorrect Password', 401); } req.session.authenticated = true; req.session.save(); debug(`New Session: ${req.session.id}`); })) // WireGuard .use((req, res, next) => { if (!PASSWORD) { return next(); } if (req.session && req.session.authenticated) { return next(); } if (req.path.startsWith('/api/') && req.headers['authorization']) { if (bcrypt.compareSync(req.headers['authorization'], bcrypt.hashSync(PASSWORD, 10))) { return next(); } return res.status(401).json({ error: 'Incorrect Password', }); } return res.status(401).json({ error: 'Not Logged In', }); }) .delete('/api/session', Util.promisify(async (req) => { const sessionId = req.session.id; req.session.destroy(); debug(`Deleted Session: ${sessionId}`); })) .get('/api/wireguard/client', Util.promisify(async (req) => { return WireGuard.getClients(); })) .get('/api/wireguard/client/:clientId/qrcode.svg', Util.promisify(async (req, res) => { const { clientId } = req.params; const svg = await WireGuard.getClientQRCodeSVG({ clientId }); res.header('Content-Type', 'image/svg+xml'); res.send(svg); })) .get('/api/wireguard/client/:clientId/configuration', Util.promisify(async (req, res) => { const { clientId } = req.params; const client = await WireGuard.getClient({ clientId }); const config = await WireGuard.getClientConfiguration({ clientId }); const configName = client.name .replace(/[^a-zA-Z0-9_=+.-]/g, '-') .replace(/(-{2,}|-$)/g, '-') .replace(/-$/, '') .substring(0, 32); res.header('Content-Disposition', `attachment; filename="${configName || clientId}.conf"`); res.header('Content-Type', 'text/plain'); res.send(config); })) .post('/api/wireguard/client', Util.promisify(async (req) => { const { name } = req.body; return WireGuard.createClient({ name }); })) .delete('/api/wireguard/client/:clientId', Util.promisify(async (req) => { const { clientId } = req.params; return WireGuard.deleteClient({ clientId }); })) .post('/api/wireguard/client/:clientId/enable', Util.promisify(async (req, res) => { const { clientId } = req.params; if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') { res.end(403); } return WireGuard.enableClient({ clientId }); })) .post('/api/wireguard/client/:clientId/disable', Util.promisify(async (req, res) => { const { clientId } = req.params; if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') { res.end(403); } return WireGuard.disableClient({ clientId }); })) .put('/api/wireguard/client/:clientId/name', Util.promisify(async (req, res) => { const { clientId } = req.params; if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') { res.end(403); } const { name } = req.body; return WireGuard.updateClientName({ clientId, name }); })) .put('/api/wireguard/client/:clientId/address', Util.promisify(async (req, res) => { const { clientId } = req.params; if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') { res.end(403); } const { address } = req.body; return WireGuard.updateClientAddress({ clientId, address }); })) .listen(PORT, WEBUI_HOST, () => { debug(`Listening on http://${WEBUI_HOST}:${PORT}`); }); } };