mirror of https://github.com/wg-easy/wg-easy
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
177 lines
5.7 KiB
177 lines
5.7 KiB
'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,
|
|
} = 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;
|
|
})))
|
|
|
|
// 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']) {
|
|
const authorizationHash = bcrypt.createHash('bcrypt')
|
|
.update(req.headers['authorization'])
|
|
.digest('hex');
|
|
const passwordHash = bcrypt.createHash('bcrypt')
|
|
.update(PASSWORD)
|
|
.digest('hex');
|
|
if (bcrypt.timingSafeEqual(Buffer.from(authorizationHash), Buffer.from(passwordHash))) {
|
|
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}`);
|
|
});
|
|
}
|
|
|
|
};
|
|
|