390 lines
13 KiB
JavaScript
390 lines
13 KiB
JavaScript
'use strict';
|
|
|
|
const bcrypt = require('bcryptjs');
|
|
const crypto = require('node:crypto');
|
|
const { createServer } = require('node:http');
|
|
const { stat, readFile } = require('node:fs/promises');
|
|
const { resolve, sep } = require('node:path');
|
|
|
|
const expressSession = require('express-session');
|
|
const debug = require('debug')('Server');
|
|
|
|
const {
|
|
createApp,
|
|
createError,
|
|
createRouter,
|
|
defineEventHandler,
|
|
fromNodeMiddleware,
|
|
getRouterParam,
|
|
toNodeListener,
|
|
readBody,
|
|
setHeader,
|
|
serveStatic,
|
|
} = require('h3');
|
|
|
|
const WireGuard = require('../services/WireGuard');
|
|
|
|
const {
|
|
PORT,
|
|
WEBUI_HOST,
|
|
RELEASE,
|
|
PASSWORD_HASH,
|
|
MAX_AGE,
|
|
LANG,
|
|
UI_TRAFFIC_STATS,
|
|
UI_CHART_TYPE,
|
|
WG_ENABLE_ONE_TIME_LINKS,
|
|
UI_ENABLE_SORT_CLIENTS,
|
|
WG_ENABLE_EXPIRES_TIME,
|
|
} = require('../config');
|
|
|
|
const requiresPassword = !!PASSWORD_HASH;
|
|
|
|
/**
|
|
* Checks if `password` matches the PASSWORD_HASH.
|
|
*
|
|
* If environment variable is not set, the password is always invalid.
|
|
*
|
|
* @param {string} password String to test
|
|
* @returns {boolean} true if matching environment, otherwise false
|
|
*/
|
|
const isPasswordValid = (password) => {
|
|
if (typeof password !== 'string') {
|
|
return false;
|
|
}
|
|
|
|
if (PASSWORD_HASH) {
|
|
return bcrypt.compareSync(password, PASSWORD_HASH);
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
const cronJobEveryMinute = async () => {
|
|
await WireGuard.cronJobEveryMinute();
|
|
setTimeout(cronJobEveryMinute, 60 * 1000);
|
|
};
|
|
|
|
module.exports = class Server {
|
|
|
|
constructor() {
|
|
const app = createApp();
|
|
this.app = app;
|
|
|
|
app.use(fromNodeMiddleware(expressSession({
|
|
secret: crypto.randomBytes(256).toString('hex'),
|
|
resave: true,
|
|
saveUninitialized: true,
|
|
})));
|
|
|
|
const router = createRouter();
|
|
app.use(router);
|
|
|
|
router
|
|
.get('/api/release', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return RELEASE;
|
|
}))
|
|
|
|
.get('/api/lang', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `"${LANG}"`;
|
|
}))
|
|
|
|
.get('/api/remember-me', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return MAX_AGE > 0;
|
|
}))
|
|
|
|
.get('/api/ui-traffic-stats', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `${UI_TRAFFIC_STATS}`;
|
|
}))
|
|
|
|
.get('/api/ui-chart-type', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `"${UI_CHART_TYPE}"`;
|
|
}))
|
|
|
|
.get('/api/wg-enable-one-time-links', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `${WG_ENABLE_ONE_TIME_LINKS}`;
|
|
}))
|
|
|
|
.get('/api/ui-sort-clients', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `${UI_ENABLE_SORT_CLIENTS}`;
|
|
}))
|
|
|
|
.get('/api/wg-enable-expire-time', defineEventHandler((event) => {
|
|
setHeader(event, 'Content-Type', 'application/json');
|
|
return `${WG_ENABLE_EXPIRES_TIME}`;
|
|
}))
|
|
|
|
// Authentication
|
|
.get('/api/session', defineEventHandler((event) => {
|
|
const authenticated = requiresPassword
|
|
? !!(event.node.req.session && event.node.req.session.authenticated)
|
|
: true;
|
|
|
|
return {
|
|
requiresPassword,
|
|
authenticated,
|
|
};
|
|
}))
|
|
.get('/cnf/:clientOneTimeLink', defineEventHandler(async (event) => {
|
|
if (WG_ENABLE_ONE_TIME_LINKS === 'false') {
|
|
throw createError({
|
|
status: 404,
|
|
message: 'Invalid state',
|
|
});
|
|
}
|
|
const clientOneTimeLink = getRouterParam(event, 'clientOneTimeLink');
|
|
const clients = await WireGuard.getClients();
|
|
const client = clients.find((client) => client.oneTimeLink === clientOneTimeLink);
|
|
if (!client) return;
|
|
const clientId = client.id;
|
|
const config = await WireGuard.getClientConfiguration({ clientId });
|
|
await WireGuard.eraseOneTimeLink({ clientId });
|
|
setHeader(event, 'Content-Disposition', `attachment; filename="${clientOneTimeLink}.conf"`);
|
|
setHeader(event, 'Content-Type', 'text/plain');
|
|
return config;
|
|
}))
|
|
.post('/api/session', defineEventHandler(async (event) => {
|
|
const { password, remember } = await readBody(event);
|
|
|
|
if (!requiresPassword) {
|
|
// if no password is required, the API should never be called.
|
|
// Do not automatically authenticate the user.
|
|
throw createError({
|
|
status: 401,
|
|
message: 'Invalid state',
|
|
});
|
|
}
|
|
|
|
if (!isPasswordValid(password)) {
|
|
throw createError({
|
|
status: 401,
|
|
message: 'Incorrect Password',
|
|
});
|
|
}
|
|
|
|
if (MAX_AGE && remember) {
|
|
event.node.req.session.cookie.maxAge = MAX_AGE;
|
|
}
|
|
event.node.req.session.authenticated = true;
|
|
event.node.req.session.save();
|
|
|
|
debug(`New Session: ${event.node.req.session.id}`);
|
|
|
|
return { success: true };
|
|
}));
|
|
|
|
// WireGuard
|
|
app.use(
|
|
fromNodeMiddleware((req, res, next) => {
|
|
if (!requiresPassword || !req.url.startsWith('/api/')) {
|
|
return next();
|
|
}
|
|
|
|
if (req.session && req.session.authenticated) {
|
|
return next();
|
|
}
|
|
|
|
if (req.url.startsWith('/api/') && req.headers['authorization']) {
|
|
if (isPasswordValid(req.headers['authorization'])) {
|
|
return next();
|
|
}
|
|
return res.status(401).json({
|
|
error: 'Incorrect Password',
|
|
});
|
|
}
|
|
|
|
return res.status(401).json({
|
|
error: 'Not Logged In',
|
|
});
|
|
}),
|
|
);
|
|
|
|
const router2 = createRouter();
|
|
app.use(router2);
|
|
|
|
router2
|
|
.delete('/api/session', defineEventHandler((event) => {
|
|
const sessionId = event.node.req.session.id;
|
|
|
|
event.node.req.session.destroy();
|
|
|
|
debug(`Deleted Session: ${sessionId}`);
|
|
return { success: true };
|
|
}))
|
|
.get('/api/wireguard/client', defineEventHandler(() => {
|
|
return WireGuard.getClients();
|
|
}))
|
|
.get('/api/wireguard/client/:clientId/qrcode.svg', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
const svg = await WireGuard.getClientQRCodeSVG({ clientId });
|
|
setHeader(event, 'Content-Type', 'image/svg+xml');
|
|
return svg;
|
|
}))
|
|
.get('/api/wireguard/client/:clientId/configuration', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
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);
|
|
setHeader(event, 'Content-Disposition', `attachment; filename="${configName || clientId}.conf"`);
|
|
setHeader(event, 'Content-Type', 'text/plain');
|
|
return config;
|
|
}))
|
|
.post('/api/wireguard/client', defineEventHandler(async (event) => {
|
|
const { name } = await readBody(event);
|
|
const { expiredDate } = await readBody(event);
|
|
await WireGuard.createClient({ name, expiredDate });
|
|
return { success: true };
|
|
}))
|
|
.delete('/api/wireguard/client/:clientId', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
await WireGuard.deleteClient({ clientId });
|
|
return { success: true };
|
|
}))
|
|
.post('/api/wireguard/client/:clientId/enable', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
await WireGuard.enableClient({ clientId });
|
|
return { success: true };
|
|
}))
|
|
.post('/api/wireguard/client/:clientId/generateOneTimeLink', defineEventHandler(async (event) => {
|
|
if (WG_ENABLE_ONE_TIME_LINKS === 'false') {
|
|
throw createError({
|
|
status: 404,
|
|
message: 'Invalid state',
|
|
});
|
|
}
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
await WireGuard.generateOneTimeLink({ clientId });
|
|
return { success: true };
|
|
}))
|
|
.post('/api/wireguard/client/:clientId/disable', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
await WireGuard.disableClient({ clientId });
|
|
return { success: true };
|
|
}))
|
|
.put('/api/wireguard/client/:clientId/name', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
const { name } = await readBody(event);
|
|
await WireGuard.updateClientName({ clientId, name });
|
|
return { success: true };
|
|
}))
|
|
.put('/api/wireguard/client/:clientId/address', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
const { address } = await readBody(event);
|
|
await WireGuard.updateClientAddress({ clientId, address });
|
|
return { success: true };
|
|
}))
|
|
.put('/api/wireguard/client/:clientId/expireDate', defineEventHandler(async (event) => {
|
|
const clientId = getRouterParam(event, 'clientId');
|
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
|
throw createError({ status: 403 });
|
|
}
|
|
const { expireDate } = await readBody(event);
|
|
await WireGuard.updateClientExpireDate({ clientId, expireDate });
|
|
return { success: true };
|
|
}));
|
|
|
|
const safePathJoin = (base, target) => {
|
|
// Manage web root (edge case)
|
|
if (target === '/') {
|
|
return `${base}${sep}`;
|
|
}
|
|
|
|
// Prepend './' to prevent absolute paths
|
|
const targetPath = `.${sep}${target}`;
|
|
|
|
// Resolve the absolute path
|
|
const resolvedPath = resolve(base, targetPath);
|
|
|
|
// Check if resolvedPath is a subpath of base
|
|
if (resolvedPath.startsWith(`${base}${sep}`)) {
|
|
return resolvedPath;
|
|
}
|
|
|
|
throw createError({
|
|
status: 400,
|
|
message: 'Bad Request',
|
|
});
|
|
};
|
|
|
|
// backup_restore
|
|
const router3 = createRouter();
|
|
app.use(router3);
|
|
|
|
router3
|
|
.get('/api/wireguard/backup', defineEventHandler(async (event) => {
|
|
const config = await WireGuard.backupConfiguration();
|
|
setHeader(event, 'Content-Disposition', 'attachment; filename="wg0.json"');
|
|
setHeader(event, 'Content-Type', 'text/json');
|
|
return config;
|
|
}))
|
|
.put('/api/wireguard/restore', defineEventHandler(async (event) => {
|
|
const { file } = await readBody(event);
|
|
await WireGuard.restoreConfiguration(file);
|
|
return { success: true };
|
|
}));
|
|
|
|
// Static assets
|
|
const publicDir = '/app/www';
|
|
app.use(
|
|
defineEventHandler((event) => {
|
|
return serveStatic(event, {
|
|
getContents: (id) => {
|
|
return readFile(safePathJoin(publicDir, id));
|
|
},
|
|
getMeta: async (id) => {
|
|
const filePath = safePathJoin(publicDir, id);
|
|
|
|
const stats = await stat(filePath).catch(() => {});
|
|
if (!stats || !stats.isFile()) {
|
|
return;
|
|
}
|
|
|
|
if (id.endsWith('.html')) setHeader(event, 'Content-Type', 'text/html');
|
|
if (id.endsWith('.js')) setHeader(event, 'Content-Type', 'application/javascript');
|
|
if (id.endsWith('.json')) setHeader(event, 'Content-Type', 'application/json');
|
|
if (id.endsWith('.css')) setHeader(event, 'Content-Type', 'text/css');
|
|
if (id.endsWith('.png')) setHeader(event, 'Content-Type', 'image/png');
|
|
|
|
return {
|
|
size: stats.size,
|
|
mtime: stats.mtimeMs,
|
|
};
|
|
},
|
|
});
|
|
}),
|
|
);
|
|
|
|
createServer(toNodeListener(app)).listen(PORT, WEBUI_HOST);
|
|
debug(`Listening on http://${WEBUI_HOST}:${PORT}`);
|
|
|
|
cronJobEveryMinute();
|
|
}
|
|
|
|
};
|