This commit is contained in:
Emile Nijssen 2021-05-22 22:40:11 +02:00
parent d7bb645470
commit d8f45476da
6 changed files with 104 additions and 63 deletions

View File

@ -6,3 +6,4 @@ module.exports.WG_PATH = process.env.WG_PATH || '/etc/wireguard/';
module.exports.WG_HOST = process.env.WG_HOST || '127.0.0.1'; module.exports.WG_HOST = process.env.WG_HOST || '127.0.0.1';
module.exports.WG_PORT = process.env.WG_PORT || 51820; module.exports.WG_PORT = process.env.WG_PORT || 51820;
module.exports.WG_DEFAULT_ADDRESS = process.env.WG_DEFAULT_ADDRESS || '10.6.0.1/32'; module.exports.WG_DEFAULT_ADDRESS = process.env.WG_DEFAULT_ADDRESS || '10.6.0.1/32';
module.exports.WG_DEFAULT_DNS = process.env.WG_DEFAULT_DNS || '1.1.1.1';

View File

@ -15,11 +15,13 @@ const {
PASSWORD, PASSWORD,
} = require('../config'); } = require('../config');
WireGuard.getClients().then(console.log);
module.exports = class Server { module.exports = class Server {
constructor() { constructor() {
// Express // Express
this.app = express() this.app = express()
.disable('etag')
.use('/', express.static(path.join(__dirname, '..', 'www'))) .use('/', express.static(path.join(__dirname, '..', 'www')))
.use(express.json()) .use(express.json())
.use(expressSession({ .use(expressSession({

View File

@ -5,58 +5,58 @@ const path = require('path');
const QRCode = require('qrcode'); const QRCode = require('qrcode');
const Util = require('./Util');
const ServerError = require('./ServerError'); const ServerError = require('./ServerError');
const { const {
WG_PATH, WG_PATH,
WG_HOST, WG_HOST,
WG_PORT, WG_PORT,
WG_DEFAULT_DNS,
WG_DEFAULT_ADDRESS, WG_DEFAULT_ADDRESS,
} = require('../config'); } = require('../config');
module.exports = class WireGuard { module.exports = class WireGuard {
constructor() { async getConfig() {
Promise.resolve().then(async () => { if (!this.__configPromise) {
try { this.__configPromise = Promise.resolve().then(async () => {
const config = await fs.readFile(path.join(WG_PATH, 'wg0.json'), 'utf8'); let config;
this.config = JSON.parse(config); try {
} catch (err) { config = await fs.readFile(path.join(WG_PATH, 'wg0.json'), 'utf8');
this.config = { config = JSON.parse(config);
// TODO: Generate new config } catch (err) {
server: { config = {
address: WG_DEFAULT_ADDRESS, server: {
}, // TODO: Generate new config
clients: {}, address: WG_DEFAULT_ADDRESS,
}; dns: WG_DEFAULT_DNS,
// TODO: Save JSON config },
} clients: {},
};
}
console.log('this.config', this.config); return config;
});
}
await this.saveConfig(); return this.__configPromise;
}).catch(err => {
// eslint-disable-next-line no-console
console.error(err);
// eslint-disable-next-line no-process-exit
process.exit(1);
});
} }
async saveConfig() { async saveConfig() {
const config = await this.getConfig();
let result = ` let result = `
# Note: Do not edit this file directly. # Note: Do not edit this file directly.
# Your changes will be overwritten! # Your changes will be overwritten!
# Server # Server
[Interface] [Interface]
PrivateKey = ${this.config.server.privateKey} PrivateKey = ${config.server.privateKey}
Address = ${this.config.server.address} Address = ${config.server.address}
ListenPort = ${this.config.server.port} ListenPort = ${config.server.port}
DNS = ${this.config.server.dns}`; DNS = ${config.server.dns}`;
for (const [clientId, client] of Object.entries(this.config.clients)) { for (const [clientId, client] of Object.entries(config.clients)) {
if (!client.enabled) continue; if (!client.enabled) continue;
result += ` result += `
@ -68,44 +68,70 @@ PresharedKey = ${client.preSharedKey}
AllowedIPs = ${client.allowedIPs}`; AllowedIPs = ${client.allowedIPs}`;
} }
await fs.writeFile(path.join(WG_PATH, 'wg0.json'), JSON.stringify(this.config, false, 2)); await fs.writeFile(path.join(WG_PATH, 'wg0.json'), JSON.stringify(config, false, 2));
await fs.writeFile(path.join(WG_PATH, 'wg0.conf'), result); await fs.writeFile(path.join(WG_PATH, 'wg0.conf'), result);
} }
async getClients() { async getClients() {
return Object.entries(this.config.clients).map(([clientId, client]) => ({ const config = await this.getConfig();
const clients = Object.entries(config.clients).map(([clientId, client]) => ({
id: clientId, id: clientId,
name: client.name, name: client.name,
enabled: client.enabled, enabled: client.enabled,
publicKey: client.publicKey, publicKey: client.publicKey,
createdAt: client.createdAt, createdAt: new Date(client.createdAt),
updatedAt: client.updatedAt, updatedAt: new Date(client.updatedAt),
allowedIPs: client.allowedIPs, allowedIPs: client.allowedIPs,
// TODO: persistentKeepalive: null,
latestHandshake: new Date(), latestHandshakeAt: null,
transferRx: 0, transferRx: null,
transferTx: 0, transferTx: null,
})); }));
// const { stdout } = await Util.exec('sudo cat /etc/wireguard/configs/clients.txt');
// return stdout // Loop WireGuard status
// .trim() // const clientsDump = await Util.exec('wg show wg0 dump');
// .split('\n') const clientsDump = `iOQJS7OUUGPYATsX6nqlL+sOODoiWiN5IOE8Msfw/0o= BkdntwYazhYZzEEHhcYayq6TGw9/YUDQ251s+5bTgC0= 51820 off
// .filter(line => { i8xWKqicnDkNL14I4B+I1zlB8od/booA1joIosWn7X4= MzplKtOQ44/IaAKri2VKqCoIlg4XiVH7TCp5bcYRTQU= 172.17.0.1:60475 10.8.0.2/32 1621679257 7920 7440 off`;
// return line.length > 0; clientsDump
// }) .trim()
// .map(line => { .split('\n')
// const [ name, publicKey, createdAt ] = line.split(' '); .slice(1)
// return { .forEach(line => {
// name, const [
// publicKey, publicKey,
// createdAt: new Date(Number(createdAt + '000')), preSharedKey,
// }; endpoint,
// }); allowedIps,
latestHandshakeAt,
transferRx,
transferTx,
persistentKeepalive,
] = line.split('\t');
const client = clients.find(client => client.publicKey === publicKey);
console.log({ publicKey, client });
if (!client) return;
client.endpoint = endpoint === '(none)'
? null
: endpoint;
client.latestHandshakeAt = latestHandshakeAt === '0'
? null
: new Date(Number(`${latestHandshakeAt}000`));
client.transferRx = Number(transferRx);
client.transferTx = Number(transferTx);
client.persistentKeepalive = persistentKeepalive;
});
console.log('clients', clients);
return clients;
} }
async getClient({ clientId }) { async getClient({ clientId }) {
const client = this.config.clients[clientId]; const config = await this.getConfig();
const client = config.clients[clientId];
if (!client) { if (!client) {
throw new ServerError(`Client Not Found: ${clientId}`, 404); throw new ServerError(`Client Not Found: ${clientId}`, 404);
} }
@ -114,13 +140,14 @@ AllowedIPs = ${client.allowedIPs}`;
} }
async getClientConfiguration({ clientId }) { async getClientConfiguration({ clientId }) {
const config = await this.getConfig();
const client = await this.getClient({ clientId }); const client = await this.getClient({ clientId });
return ` return `
[Interface] [Interface]
PrivateKey = ${client.privateKey} PrivateKey = ${client.privateKey}
Address = ${client.address} Address = ${client.address}
DNS = ${this.config.server.dns} DNS = ${config.server.dns}
[Peer] [Peer]
PublicKey = ${client.publicKey} PublicKey = ${client.publicKey}
@ -160,8 +187,10 @@ Endpoint = ${WG_HOST}:${WG_PORT}`;
} }
async deleteClient({ clientId }) { async deleteClient({ clientId }) {
if (this.config.clients[clientId]) { const config = await this.getConfig();
delete this.config.clients[clientId];
if (config.clients[clientId]) {
delete config.clients[clientId];
await this.saveConfig(); await this.saveConfig();
} }
} }

View File

@ -51,7 +51,8 @@
</svg> </svg>
<img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" /> <img v-if="client.avatar" :src="client.avatar" class="w-10 rounded-full absolute top-0 left-0" />
<div v-if="client.latestHandshake && ((new Date() - new Date(client.latestHandshake) < 1000 * 60 * 10))"> <div
v-if="client.latestHandshakeAt && ((new Date() - new Date(client.latestHandshakeAt) < 1000 * 60 * 10))">
<div class="animate-ping w-4 h-4 p-1 bg-green-100 rounded-full absolute -bottom-1 -right-1"></div> <div class="animate-ping w-4 h-4 p-1 bg-green-100 rounded-full absolute -bottom-1 -right-1"></div>
<div class="w-2 h-2 bg-green-300 rounded-full absolute bottom-0 right-0"></div> <div class="w-2 h-2 bg-green-300 rounded-full absolute bottom-0 right-0"></div>
</div> </div>
@ -60,8 +61,7 @@
<div class="flex-grow"> <div class="flex-grow">
<div class="text-gray-700" :title="'Created at ' + dateTime(new Date(client.createdAt))">{{client.name}} <div class="text-gray-700" :title="'Created at ' + dateTime(new Date(client.createdAt))">{{client.name}}
</div> </div>
<div v-if="client.allowedIps" class="text-gray-300 text-xs">{{client.iface}} <div v-if="client.allowedIPs" class="text-gray-300 text-xs">{{client.allowedIPs.split('/')[0]}}
· {{client.allowedIps.split('/')[0]}}
<span v-if="client.transferTx" title="Download"> <span v-if="client.transferTx" title="Download">
· ·
<svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20" <svg class="align-middle h-3 inline" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"
@ -82,9 +82,9 @@
</svg> </svg>
{{client.transferRx | bytes}} {{client.transferRx | bytes}}
</span> </span>
<span v-if="client.latestHandshake" <span v-if="client.latestHandshakeAt"
:title="'Last seen at ' + dateTime(new Date(client.latestHandshake))"> :title="'Last seen at ' + dateTime(new Date(client.latestHandshakeAt))">
· {{new Date(client.latestHandshake) | timeago}} · {{new Date(client.latestHandshakeAt) | timeago}}
</span> </span>
</div> </div>
</div> </div>

View File

@ -55,7 +55,14 @@ class API {
return this.call({ return this.call({
method: 'get', method: 'get',
path: '/wireguard/client', path: '/wireguard/client',
}); }).then(clients => clients.map(client => ({
...client,
createdAt: new Date(client.createdAt),
updatedAt: new Date(client.updatedAt),
latestHandshakeAt: client.latestHandshakeAt !== null
? new Date(client.latestHandshakeAt)
: null,
})));
} }
async createClient({ name }) { async createClient({ name }) {

View File

@ -39,6 +39,8 @@ new Vue({
return client; return client;
}); });
console.log(clients);
}, },
login(e) { login(e) {
e.preventDefault(); e.preventDefault();