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.
239 lines
7.5 KiB
239 lines
7.5 KiB
'use strict';
|
|
|
|
const bcrypt = require('bcryptjs');
|
|
const crypto = require('node:crypto');
|
|
const { createServer } = require('node:http');
|
|
const { stat, readFile } = require('node:fs/promises');
|
|
const { join } = 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,
|
|
LANG,
|
|
UI_TRAFFIC_STATS,
|
|
UI_CHART_TYPE,
|
|
} = require('../config');
|
|
|
|
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/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}"`;
|
|
|
|
// Authentication
|
|
.get('/api/session', defineEventHandler((event) => {
|
|
const requiresPassword = !!process.env.PASSWORD;
|
|
const authenticated = requiresPassword
|
|
? !!(event.node.req.session && event.node.req.session.authenticated)
|
|
: true;
|
|
|
|
return {
|
|
requiresPassword,
|
|
authenticated,
|
|
};
|
|
}))
|
|
.post('/api/session', defineEventHandler(async (event) => {
|
|
const { password } = await readBody(event);
|
|
|
|
if (typeof password !== 'string') {
|
|
throw createError({
|
|
status: 401,
|
|
message: 'Missing: Password',
|
|
});
|
|
}
|
|
|
|
if (password !== PASSWORD) {
|
|
throw createError({
|
|
status: 401,
|
|
message: 'Incorrect Password',
|
|
});
|
|
}
|
|
|
|
event.node.req.session.authenticated = true;
|
|
event.node.req.session.save();
|
|
|
|
debug(`New Session: ${event.node.req.session.id}`);
|
|
|
|
return { succcess: true };
|
|
}));
|
|
|
|
// WireGuard
|
|
app.use(
|
|
fromNodeMiddleware((req, res, next) => {
|
|
if (!PASSWORD || !req.url.startsWith('/api/')) {
|
|
return next();
|
|
}
|
|
|
|
if (req.session && req.session.authenticated) {
|
|
return next();
|
|
}
|
|
|
|
if (req.url.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',
|
|
});
|
|
}),
|
|
);
|
|
|
|
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);
|
|
await WireGuard.createClient({ name });
|
|
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/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 };
|
|
}));
|
|
|
|
// Static assets
|
|
const publicDir = 'www';
|
|
app.use(
|
|
defineEventHandler((event) => {
|
|
return serveStatic(event, {
|
|
getContents: (id) => readFile(join(publicDir, id)),
|
|
getMeta: async (id) => {
|
|
const stats = await stat(join(publicDir, id)).catch(() => {});
|
|
|
|
if (!stats || !stats.isFile()) {
|
|
return;
|
|
}
|
|
|
|
return {
|
|
size: stats.size,
|
|
mtime: stats.mtimeMs,
|
|
};
|
|
},
|
|
});
|
|
}),
|
|
);
|
|
|
|
createServer(toNodeListener(app)).listen(PORT, WEBUI_HOST);
|
|
debug(`Listening on http://${WEBUI_HOST}:${PORT}`);
|
|
}
|
|
|
|
};
|
|
|