Merge remote-tracking branch 'wg/master'
# Conflicts: # .github/CODEOWNERS # .github/PULL_REQUEST_TEMPLATE.md # .github/dependabot.yml # .github/workflows/deploy-development.yml # .github/workflows/deploy-nightly.yml # .github/workflows/deploy-pr.yml # .github/workflows/lint.yml # Dockerfile # How_to_generate_an_bcrypt_hash.md # README.md # docker-compose.dev.yml # docker-compose.yml # src/config.js # src/lib/Server.js # src/lib/WireGuard.js # src/www/js/api.js # src/www/js/app.js
This commit is contained in:
commit
7b1e606bc2
|
@ -0,0 +1,23 @@
|
||||||
|
# http://editorconfig.org
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
end_of_line = lf
|
||||||
|
charset = utf-8
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
# The JSON files contain newlines inconsistently
|
||||||
|
[*.json]
|
||||||
|
insert_final_newline = ignore
|
||||||
|
|
||||||
|
# Minified JavaScript files shouldn't be changed
|
||||||
|
[**.min.js]
|
||||||
|
indent_style = ignore
|
||||||
|
insert_final_newline = ignore
|
||||||
|
|
||||||
|
[*.md]
|
||||||
|
trim_trailing_whitespace = false
|
||||||
|
|
|
@ -46,4 +46,4 @@ ENV DEBUG=Server,WireGuard
|
||||||
|
|
||||||
# Run Web UI
|
# Run Web UI
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
CMD ["/usr/bin/dumb-init", "node", "server.js"]
|
CMD ["/usr/bin/dumb-init", "node", "server.js"]
|
||||||
|
|
|
@ -12,11 +12,17 @@
|
||||||
To generate a bcrypt password hash using docker, run the following command :
|
To generate a bcrypt password hash using docker, run the following command :
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
docker run ghcr.io/w0rng/amnezia-wg-easy wgpw YOUR_PASSWORD
|
docker run -it ghcr.io/w0rng/amnezia-wg-easy wgpw YOUR_PASSWORD
|
||||||
PASSWORD_HASH='$2b$12$coPqCsPtcFO.Ab99xylBNOW4.Iu7OOA2/ZIboHN6/oyxca3MWo7fW' // literally YOUR_PASSWORD
|
PASSWORD_HASH='$2b$12$coPqCsPtcFO.Ab99xylBNOW4.Iu7OOA2/ZIboHN6/oyxca3MWo7fW' // literally YOUR_PASSWORD
|
||||||
```
|
```
|
||||||
|
If a password is not provided, the tool will prompt you for one :
|
||||||
|
```sh
|
||||||
|
docker run -it ghcr.io/wg-easy/wg-easy wgpw
|
||||||
|
Enter your password: // hidden prompt, type in your password
|
||||||
|
PASSWORD_HASH='$2b$12$coPqCsPtcFO.Ab99xylBNOW4.Iu7OOA2/ZIboHN6/oyxca3MWo7fW'
|
||||||
|
```
|
||||||
|
|
||||||
*Important* : make sure to enclose your password in **single quotes** when you run `docker run` command :
|
**Important** : make sure to enclose your password in **single quotes** when you run `docker run` command :
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
$ echo $2b$12$coPqCsPtcF <-- not correct
|
$ echo $2b$12$coPqCsPtcF <-- not correct
|
||||||
|
@ -26,3 +32,11 @@ b2
|
||||||
$ echo '$2b$12$coPqCsPtcF' <-- correct
|
$ echo '$2b$12$coPqCsPtcF' <-- correct
|
||||||
$2b$12$coPqCsPtcF
|
$2b$12$coPqCsPtcF
|
||||||
```
|
```
|
||||||
|
|
||||||
|
**Important** : Please note: don't wrap the generated hash password in single quotes when you use `docker-compose.yml`. Instead, replace each `$` symbol with two `$$` symbols. For example:
|
||||||
|
|
||||||
|
``` yaml
|
||||||
|
- PASSWORD_HASH=$$2y$$10$$hBCoykrB95WSzuV4fafBzOHWKu9sbyVa34GJr8VV5R/pIelfEMYyG
|
||||||
|
```
|
||||||
|
|
||||||
|
This hash is for the password 'foobar123', obtained using the command `docker run ghcr.io/wg-easy/wg-easy wgpw foobar123` and then inserted an additional `$` before each existing `$` symbol.
|
||||||
|
|
81
README.md
81
README.md
|
@ -18,7 +18,10 @@ You have found the easiest way to install & manage WireGuard on any Linux host!
|
||||||
* Gravatar support or random avatars.
|
* Gravatar support or random avatars.
|
||||||
* Automatic Light / Dark Mode
|
* Automatic Light / Dark Mode
|
||||||
* Multilanguage Support
|
* Multilanguage Support
|
||||||
* UI_TRAFFIC_STATS (default off)
|
* Traffic Stats (default off)
|
||||||
|
* One Time Links (default off)
|
||||||
|
* Client Expiry (default off)
|
||||||
|
* Prometheus metrics support
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
|
@ -69,44 +72,52 @@ To automatically install & run wg-easy, simply run:
|
||||||
|
|
||||||
The Web UI will now be available on `http://0.0.0.0:51821`.
|
The Web UI will now be available on `http://0.0.0.0:51821`.
|
||||||
|
|
||||||
|
The Prometheus metrics will now be available on `http://0.0.0.0:51821/metrics`. Grafana dashboard [21733](https://grafana.com/grafana/dashboards/21733-wireguard/)
|
||||||
|
|
||||||
> 💡 Your configuration files will be saved in `~/.amnezia-wg-easy`
|
> 💡 Your configuration files will be saved in `~/.amnezia-wg-easy`
|
||||||
|
|
||||||
## Options
|
## Options
|
||||||
|
|
||||||
These options can be configured by setting environment variables using `-e KEY="VALUE"` in the `docker run` command.
|
These options can be configured by setting environment variables using `-e KEY="VALUE"` in the `docker run` command.
|
||||||
|
|
||||||
| Env | Default | Example | Description |
|
| Env | Default | Example | Description |
|
||||||
|---------------------------|-------------------|--------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|
|-------------------------------|-------------------|--------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|
||||||
| `PORT` | `51821` | `6789` | TCP port for Web UI. |
|
| `PORT` | `51821` | `6789` | TCP port for Web UI. |
|
||||||
| `WEBUI_HOST` | `0.0.0.0` | `localhost` | IP address web UI binds to. |
|
| `WEBUI_HOST` | `0.0.0.0` | `localhost` | IP address web UI binds to. |
|
||||||
| `PASSWORD_HASH` | - | `$2y$05$Ci...` | When set, requires a password when logging in to the Web UI. See [How to generate an bcrypt hash.md]("https://github.com/wg-easy/wg-easy/blob/master/How_to_generate_an_bcrypt_hash.md") for know how generate the hash. |
|
| `PASSWORD_HASH` | - | `$2y$05$Ci...` | When set, requires a password when logging in to the Web UI. See [How to generate an bcrypt hash.md]("https://github.com/wg-easy/wg-easy/blob/master/How_to_generate_an_bcrypt_hash.md") for know how generate the hash. |
|
||||||
| `WG_HOST` | - | `vpn.myserver.com` | The public hostname of your VPN server. |
|
| `WG_HOST` | - | `vpn.myserver.com` | The public hostname of your VPN server. |
|
||||||
| `WG_DEVICE` | `eth0` | `ens6f0` | Ethernet device the wireguard traffic should be forwarded through. |
|
| `WG_DEVICE` | `eth0` | `ens6f0` | Ethernet device the wireguard traffic should be forwarded through. |
|
||||||
| `WG_PORT` | `51820` | `12345` | The public UDP port of your VPN server. WireGuard will listen on that (othwise default) inside the Docker container. |
|
| `WG_PORT` | `51820` | `12345` | The public UDP port of your VPN server. WireGuard will listen on that (othwise default) inside the Docker container. |
|
||||||
| `WG_CONFIG_PORT` | `51820` | `12345` | The UDP port used on [Home Assistant Plugin](https://github.com/adriy-be/homeassistant-addons-jdeath/tree/main/wgeasy)
|
| `WG_CONFIG_PORT` | `51820` | `12345` | The UDP port used on [Home Assistant Plugin](https://github.com/adriy-be/homeassistant-addons-jdeath/tree/main/wgeasy) |
|
||||||
| `WG_MTU` | `null` | `1420` | The MTU the clients will use. Server uses default WG MTU. |
|
| `WG_MTU` | `null` | `1420` | The MTU the clients will use. Server uses default WG MTU. |
|
||||||
| `WG_PERSISTENT_KEEPALIVE` | `0` | `25` | Value in seconds to keep the "connection" open. If this value is 0, then connections won't be kept alive. |
|
| `WG_PERSISTENT_KEEPALIVE` | `0` | `25` | Value in seconds to keep the "connection" open. If this value is 0, then connections won't be kept alive. |
|
||||||
| `WG_DEFAULT_ADDRESS` | `10.8.0.x` | `10.6.0.x` | Clients IP address range. |
|
| `WG_DEFAULT_ADDRESS` | `10.8.0.x` | `10.6.0.x` | Clients IP address range. |
|
||||||
| `WG_DEFAULT_DNS` | `1.1.1.1` | `8.8.8.8, 8.8.4.4` | DNS server clients will use. If set to blank value, clients will not use any DNS. |
|
| `WG_DEFAULT_DNS` | `1.1.1.1` | `8.8.8.8, 8.8.4.4` | DNS server clients will use. If set to blank value, clients will not use any DNS. |
|
||||||
| `WG_ALLOWED_IPS` | `0.0.0.0/0, ::/0` | `192.168.15.0/24, 10.0.1.0/24` | Allowed IPs clients will use. |
|
| `WG_ALLOWED_IPS` | `0.0.0.0/0, ::/0` | `192.168.15.0/24, 10.0.1.0/24` | Allowed IPs clients will use. |
|
||||||
| `WG_PRE_UP` | `...` | - | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L19) for the default value. |
|
| `WG_PRE_UP` | `...` | - | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L19) for the default value. |
|
||||||
| `WG_POST_UP` | `...` | `iptables ...` | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L20) for the default value. |
|
| `WG_POST_UP` | `...` | `iptables ...` | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L20) for the default value. |
|
||||||
| `WG_PRE_DOWN` | `...` | - | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L27) for the default value. |
|
| `WG_PRE_DOWN` | `...` | - | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L27) for the default value. |
|
||||||
| `WG_POST_DOWN` | `...` | `iptables ...` | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L28) for the default value. |
|
| `WG_POST_DOWN` | `...` | `iptables ...` | See [config.js](https://github.com/wg-easy/wg-easy/blob/master/src/config.js#L28) for the default value. |
|
||||||
| `LANG` | `en` | `de` | Web UI language (Supports: en, ua, ru, tr, no, pl, fr, de, ca, es, ko, vi, nl, is, pt, chs, cht, it, th, hi). |
|
| `WG_ENABLE_EXPIRES_TIME` | `false` | `true` | Enable expire time for clients |
|
||||||
| `UI_TRAFFIC_STATS` | `false` | `true` | Enable detailed RX / TX client stats in Web UI |
|
| `LANG` | `en` | `de` | Web UI language (Supports: en, ua, ru, tr, no, pl, fr, de, ca, es, ko, vi, nl, is, pt, chs, cht, it, th, hi). |
|
||||||
| `UI_CHART_TYPE` | `0` | `1` | UI_CHART_TYPE=0 # Charts disabled, UI_CHART_TYPE=1 # Line chart, UI_CHART_TYPE=2 # Area chart, UI_CHART_TYPE=3 # Bar chart |
|
| `UI_TRAFFIC_STATS` | `false` | `true` | Enable detailed RX / TX client stats in Web UI |
|
||||||
| `DICEBEAR_TYPE` | `false` | `bottts` | see [dicebear types](https://www.dicebear.com/styles/) |
|
| `UI_CHART_TYPE` | `0` | `1` | UI_CHART_TYPE=0 # Charts disabled, UI_CHART_TYPE=1 # Line chart, UI_CHART_TYPE=2 # Area chart, UI_CHART_TYPE=3 # Bar chart |
|
||||||
| `USE_GRAVATAR` | `false` | `true` | Use or not GRAVATAR service |
|
| `DICEBEAR_TYPE` | `false` | `bottts` | see [dicebear types](https://www.dicebear.com/styles/) |
|
||||||
| `JC` | `random` | `5` | Junk packet count — number of packets with random data that are sent before the start of the session. |
|
| `USE_GRAVATAR` | `false` | `true` | Use or not GRAVATAR service |
|
||||||
| `JMIN` | `50` | `25` | Junk packet minimum size — minimum packet size for Junk packet. That is, all randomly generated packets will have a size no smaller than Jmin. |
|
| `WG_ENABLE_ONE_TIME_LINKS` | `false` | `true` | Enable display and generation of short one time download links (expire after 5 minutes) |
|
||||||
| `JMAX` | `1000` | `250` | Junk packet maximum size — maximum size for Junk packets. |
|
| `MAX_AGE` | `0` | `1440` | The maximum age of Web UI sessions in minutes. `0` means that the session will exist until the browser is closed. |
|
||||||
| `S1` | `random` | `75` | Init packet junk size — the size of random data that will be added to the init packet, the size of which is initially fixed. |
|
| `UI_ENABLE_SORT_CLIENTS` | `false` | `true` | Enable UI sort clients by name |
|
||||||
| `S2` | `random` | `75` | Response packet junk size — the size of random data that will be added to the response packet, the size of which is initially fixed. |
|
| `ENABLE_PROMETHEUS_METRICS` | `false` | `true` | Enable Prometheus metrics `http://0.0.0.0:51821/metrics` and `http://0.0.0.0:51821/metrics/json` |
|
||||||
| `H1` | `random` | `1234567891` | Init packet magic header — the header of the first byte of the handshake. Must be < uint_max. |
|
| `PROMETHEUS_METRICS_PASSWORD` | - | `$2y$05$Ci...` | If set, Basic Auth is required when requesting metrics. See [How to generate an bcrypt hash.md]("https://github.com/wg-easy/wg-easy/blob/master/How_to_generate_an_bcrypt_hash.md") for know how generate the hash. |
|
||||||
| `H2` | `random` | `1234567892` | Response packet magic header — header of the first byte of the handshake response. Must be < uint_max. |
|
| `JC` | `random` | `5` | Junk packet count — number of packets with random data that are sent before the start of the session. |
|
||||||
| `H3` | `random` | `1234567893` | Underload packet magic header — UnderLoad packet header. Must be < uint_max. |
|
| `JMIN` | `50` | `25` | Junk packet minimum size — minimum packet size for Junk packet. That is, all randomly generated packets will have a size no smaller than Jmin. |
|
||||||
| `H4` | `random` | `1234567894` | Transport packet magic header — header of the packet of the data packet. Must be < uint_max. |
|
| `JMAX` | `1000` | `250` | Junk packet maximum size — maximum size for Junk packets. |
|
||||||
|
| `S1` | `random` | `75` | Init packet junk size — the size of random data that will be added to the init packet, the size of which is initially fixed. |
|
||||||
|
| `S2` | `random` | `75` | Response packet junk size — the size of random data that will be added to the response packet, the size of which is initially fixed. |
|
||||||
|
| `H1` | `random` | `1234567891` | Init packet magic header — the header of the first byte of the handshake. Must be < uint_max. |
|
||||||
|
| `H2` | `random` | `1234567892` | Response packet magic header — header of the first byte of the handshake response. Must be < uint_max. |
|
||||||
|
| `H3` | `random` | `1234567893` | Underload packet magic header — UnderLoad packet header. Must be < uint_max. |
|
||||||
|
| `H4` | `random` | `1234567894` | Transport packet magic header — header of the packet of the data packet. Must be < uint_max. |
|
||||||
|
|
||||||
> If you change `WG_PORT`, make sure to also change the exposed port.
|
> If you change `WG_PORT`, make sure to also change the exposed port.
|
||||||
|
|
||||||
|
@ -125,6 +136,4 @@ And then run the `docker run -d \ ...` command above again.
|
||||||
## Thanks
|
## Thanks
|
||||||
|
|
||||||
Based on [wg-easy](https://github.com/wg-easy/wg-easy) by Emile Nijssen.
|
Based on [wg-easy](https://github.com/wg-easy/wg-easy) by Emile Nijssen.
|
||||||
Use integrations with AmneziaWg from [
|
Use integrations with AmneziaWg from [amnezia-wg-easy](https://github.com/spcfox/amnezia-wg-easy) by Viktor Yudov.
|
||||||
amnezia-wg-easy
|
|
||||||
](https://github.com/spcfox/amnezia-wg-easy) by Viktor Yudov.
|
|
|
@ -16,6 +16,7 @@ services:
|
||||||
cap_add:
|
cap_add:
|
||||||
- NET_ADMIN
|
- NET_ADMIN
|
||||||
- SYS_MODULE
|
- SYS_MODULE
|
||||||
|
# - NET_RAW # ⚠️ Uncomment if using Podman
|
||||||
sysctls:
|
sysctls:
|
||||||
- net.ipv4.ip_forward=1
|
- net.ipv4.ip_forward=1
|
||||||
- net.ipv4.conf.all.src_valid_mark=1
|
- net.ipv4.conf.all.src_valid_mark=1
|
||||||
|
|
|
@ -6,6 +6,7 @@ module.exports.RELEASE = version;
|
||||||
module.exports.PORT = process.env.PORT || '51821';
|
module.exports.PORT = process.env.PORT || '51821';
|
||||||
module.exports.WEBUI_HOST = process.env.WEBUI_HOST || '0.0.0.0';
|
module.exports.WEBUI_HOST = process.env.WEBUI_HOST || '0.0.0.0';
|
||||||
module.exports.PASSWORD_HASH = process.env.PASSWORD_HASH;
|
module.exports.PASSWORD_HASH = process.env.PASSWORD_HASH;
|
||||||
|
module.exports.MAX_AGE = parseInt(process.env.MAX_AGE, 10) * 1000 * 60 || 0;
|
||||||
module.exports.WG_PATH = process.env.WG_PATH || '/etc/wireguard/';
|
module.exports.WG_PATH = process.env.WG_PATH || '/etc/wireguard/';
|
||||||
module.exports.WG_DEVICE = process.env.WG_DEVICE || 'eth0';
|
module.exports.WG_DEVICE = process.env.WG_DEVICE || 'eth0';
|
||||||
module.exports.WG_HOST = process.env.WG_HOST;
|
module.exports.WG_HOST = process.env.WG_HOST;
|
||||||
|
@ -37,6 +38,11 @@ iptables -D FORWARD -o wg0 -j ACCEPT;
|
||||||
module.exports.LANG = process.env.LANG || 'en';
|
module.exports.LANG = process.env.LANG || 'en';
|
||||||
module.exports.UI_TRAFFIC_STATS = process.env.UI_TRAFFIC_STATS || 'false';
|
module.exports.UI_TRAFFIC_STATS = process.env.UI_TRAFFIC_STATS || 'false';
|
||||||
module.exports.UI_CHART_TYPE = process.env.UI_CHART_TYPE || 0;
|
module.exports.UI_CHART_TYPE = process.env.UI_CHART_TYPE || 0;
|
||||||
|
module.exports.WG_ENABLE_ONE_TIME_LINKS = process.env.WG_ENABLE_ONE_TIME_LINKS || 'false';
|
||||||
|
module.exports.UI_ENABLE_SORT_CLIENTS = process.env.UI_ENABLE_SORT_CLIENTS || 'false';
|
||||||
|
module.exports.WG_ENABLE_EXPIRES_TIME = process.env.WG_ENABLE_EXPIRES_TIME || 'false';
|
||||||
|
module.exports.ENABLE_PROMETHEUS_METRICS = process.env.ENABLE_PROMETHEUS_METRICS || 'false';
|
||||||
|
module.exports.PROMETHEUS_METRICS_PASSWORD = process.env.PROMETHEUS_METRICS_PASSWORD;
|
||||||
|
|
||||||
module.exports.DICEBEAR_TYPE = process.env.DICEBEAR_TYPE || false;
|
module.exports.DICEBEAR_TYPE = process.env.DICEBEAR_TYPE || false;
|
||||||
module.exports.USE_GRAVATAR = process.env.USE_GRAVATAR || false;
|
module.exports.USE_GRAVATAR = process.env.USE_GRAVATAR || false;
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
const bcrypt = require('bcryptjs');
|
const bcrypt = require('bcryptjs');
|
||||||
const crypto = require('node:crypto');
|
const crypto = require('node:crypto');
|
||||||
|
const basicAuth = require('basic-auth');
|
||||||
const { createServer } = require('node:http');
|
const { createServer } = require('node:http');
|
||||||
const { stat, readFile } = require('node:fs/promises');
|
const { stat, readFile } = require('node:fs/promises');
|
||||||
const { resolve, sep } = require('node:path');
|
const { resolve, sep } = require('node:path');
|
||||||
|
@ -29,14 +30,21 @@ const {
|
||||||
WEBUI_HOST,
|
WEBUI_HOST,
|
||||||
RELEASE,
|
RELEASE,
|
||||||
PASSWORD_HASH,
|
PASSWORD_HASH,
|
||||||
|
MAX_AGE,
|
||||||
LANG,
|
LANG,
|
||||||
UI_TRAFFIC_STATS,
|
UI_TRAFFIC_STATS,
|
||||||
UI_CHART_TYPE,
|
UI_CHART_TYPE,
|
||||||
|
WG_ENABLE_ONE_TIME_LINKS,
|
||||||
|
UI_ENABLE_SORT_CLIENTS,
|
||||||
|
WG_ENABLE_EXPIRES_TIME,
|
||||||
|
ENABLE_PROMETHEUS_METRICS,
|
||||||
|
PROMETHEUS_METRICS_PASSWORD,
|
||||||
DICEBEAR_TYPE,
|
DICEBEAR_TYPE,
|
||||||
USE_GRAVATAR,
|
USE_GRAVATAR,
|
||||||
} = require('../config');
|
} = require('../config');
|
||||||
|
|
||||||
const requiresPassword = !!PASSWORD_HASH;
|
const requiresPassword = !!PASSWORD_HASH;
|
||||||
|
const requiresPrometheusPassword = !!PROMETHEUS_METRICS_PASSWORD;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if `password` matches the PASSWORD_HASH.
|
* Checks if `password` matches the PASSWORD_HASH.
|
||||||
|
@ -46,18 +54,22 @@ const requiresPassword = !!PASSWORD_HASH;
|
||||||
* @param {string} password String to test
|
* @param {string} password String to test
|
||||||
* @returns {boolean} true if matching environment, otherwise false
|
* @returns {boolean} true if matching environment, otherwise false
|
||||||
*/
|
*/
|
||||||
const isPasswordValid = (password) => {
|
const isPasswordValid = (password, hash) => {
|
||||||
if (typeof password !== 'string') {
|
if (typeof password !== 'string') {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
if (hash) {
|
||||||
if (PASSWORD_HASH) {
|
return bcrypt.compareSync(password, hash);
|
||||||
return bcrypt.compareSync(password, PASSWORD_HASH);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const cronJobEveryMinute = async () => {
|
||||||
|
await WireGuard.cronJobEveryMinute();
|
||||||
|
setTimeout(cronJobEveryMinute, 60 * 1000);
|
||||||
|
};
|
||||||
|
|
||||||
module.exports = class Server {
|
module.exports = class Server {
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
|
@ -84,9 +96,14 @@ module.exports = class Server {
|
||||||
return `"${LANG}"`;
|
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) => {
|
.get('/api/ui-traffic-stats', defineEventHandler((event) => {
|
||||||
setHeader(event, 'Content-Type', 'application/json');
|
setHeader(event, 'Content-Type', 'application/json');
|
||||||
return `"${UI_TRAFFIC_STATS}"`;
|
return `${UI_TRAFFIC_STATS}`;
|
||||||
}))
|
}))
|
||||||
|
|
||||||
.get('/api/ui-chart-type', defineEventHandler((event) => {
|
.get('/api/ui-chart-type', defineEventHandler((event) => {
|
||||||
|
@ -94,6 +111,21 @@ module.exports = class Server {
|
||||||
return `"${UI_CHART_TYPE}"`;
|
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}`;
|
||||||
|
}))
|
||||||
|
|
||||||
.get('/api/ui-avatar-settings', defineEventHandler((event) => {
|
.get('/api/ui-avatar-settings', defineEventHandler((event) => {
|
||||||
setHeader(event, 'Content-Type', 'application/json');
|
setHeader(event, 'Content-Type', 'application/json');
|
||||||
return {
|
return {
|
||||||
|
@ -113,8 +145,26 @@ module.exports = class Server {
|
||||||
authenticated,
|
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) => {
|
.post('/api/session', defineEventHandler(async (event) => {
|
||||||
const { password } = await readBody(event);
|
const { password, remember } = await readBody(event);
|
||||||
|
|
||||||
if (!requiresPassword) {
|
if (!requiresPassword) {
|
||||||
// if no password is required, the API should never be called.
|
// if no password is required, the API should never be called.
|
||||||
|
@ -125,13 +175,16 @@ module.exports = class Server {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isPasswordValid(password)) {
|
if (!isPasswordValid(password, PASSWORD_HASH)) {
|
||||||
throw createError({
|
throw createError({
|
||||||
status: 401,
|
status: 401,
|
||||||
message: 'Incorrect Password',
|
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.authenticated = true;
|
||||||
event.node.req.session.save();
|
event.node.req.session.save();
|
||||||
|
|
||||||
|
@ -152,7 +205,7 @@ module.exports = class Server {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (req.url.startsWith('/api/') && req.headers['authorization']) {
|
if (req.url.startsWith('/api/') && req.headers['authorization']) {
|
||||||
if (isPasswordValid(req.headers['authorization'])) {
|
if (isPasswordValid(req.headers['authorization'], PASSWORD_HASH)) {
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
return res.status(401).json({
|
return res.status(401).json({
|
||||||
|
@ -202,7 +255,8 @@ module.exports = class Server {
|
||||||
}))
|
}))
|
||||||
.post('/api/wireguard/client', defineEventHandler(async (event) => {
|
.post('/api/wireguard/client', defineEventHandler(async (event) => {
|
||||||
const { name } = await readBody(event);
|
const { name } = await readBody(event);
|
||||||
await WireGuard.createClient({ name });
|
const { expiredDate } = await readBody(event);
|
||||||
|
await WireGuard.createClient({ name, expiredDate });
|
||||||
return { success: true };
|
return { success: true };
|
||||||
}))
|
}))
|
||||||
.delete('/api/wireguard/client/:clientId', defineEventHandler(async (event) => {
|
.delete('/api/wireguard/client/:clientId', defineEventHandler(async (event) => {
|
||||||
|
@ -218,6 +272,20 @@ module.exports = class Server {
|
||||||
await WireGuard.enableClient({ clientId });
|
await WireGuard.enableClient({ clientId });
|
||||||
return { success: true };
|
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) => {
|
.post('/api/wireguard/client/:clientId/disable', defineEventHandler(async (event) => {
|
||||||
const clientId = getRouterParam(event, 'clientId');
|
const clientId = getRouterParam(event, 'clientId');
|
||||||
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
if (clientId === '__proto__' || clientId === 'constructor' || clientId === 'prototype') {
|
||||||
|
@ -243,6 +311,15 @@ module.exports = class Server {
|
||||||
const { address } = await readBody(event);
|
const { address } = await readBody(event);
|
||||||
await WireGuard.updateClientAddress({ clientId, address });
|
await WireGuard.updateClientAddress({ clientId, address });
|
||||||
return { success: true };
|
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) => {
|
const safePathJoin = (base, target) => {
|
||||||
|
@ -268,6 +345,51 @@ module.exports = class Server {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Prometheus Metrics API
|
||||||
|
const routerPrometheusMetrics = createRouter();
|
||||||
|
app.use(routerPrometheusMetrics);
|
||||||
|
|
||||||
|
// Check Prometheus credentials
|
||||||
|
app.use(
|
||||||
|
fromNodeMiddleware((req, res, next) => {
|
||||||
|
if (!requiresPrometheusPassword || !req.url.startsWith('/metrics')) {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
const user = basicAuth(req);
|
||||||
|
if (requiresPrometheusPassword && !user) {
|
||||||
|
res.statusCode = 401;
|
||||||
|
return { error: 'Not Logged In' };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (user.pass) {
|
||||||
|
if (isPasswordValid(user.pass, PROMETHEUS_METRICS_PASSWORD)) {
|
||||||
|
return next();
|
||||||
|
}
|
||||||
|
res.statusCode = 401;
|
||||||
|
return { error: 'Incorrect Password' };
|
||||||
|
}
|
||||||
|
res.statusCode = 401;
|
||||||
|
return { error: 'Not Logged In' };
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
// Prometheus Routes
|
||||||
|
routerPrometheusMetrics
|
||||||
|
.get('/metrics', defineEventHandler(async (event) => {
|
||||||
|
setHeader(event, 'Content-Type', 'text/plain');
|
||||||
|
if (ENABLE_PROMETHEUS_METRICS === 'true') {
|
||||||
|
return WireGuard.getMetrics();
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
}))
|
||||||
|
.get('/metrics/json', defineEventHandler(async (event) => {
|
||||||
|
setHeader(event, 'Content-Type', 'application/json');
|
||||||
|
if (ENABLE_PROMETHEUS_METRICS === 'true') {
|
||||||
|
return WireGuard.getMetricsJSON();
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
}));
|
||||||
|
|
||||||
// backup_restore
|
// backup_restore
|
||||||
const router3 = createRouter();
|
const router3 = createRouter();
|
||||||
app.use(router3);
|
app.use(router3);
|
||||||
|
@ -319,6 +441,8 @@ module.exports = class Server {
|
||||||
|
|
||||||
createServer(toNodeListener(app)).listen(PORT, WEBUI_HOST);
|
createServer(toNodeListener(app)).listen(PORT, WEBUI_HOST);
|
||||||
debug(`Listening on http://${WEBUI_HOST}:${PORT}`);
|
debug(`Listening on http://${WEBUI_HOST}:${PORT}`);
|
||||||
|
|
||||||
|
cronJobEveryMinute();
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -5,6 +5,7 @@ const path = require('path');
|
||||||
const debug = require('debug')('WireGuard');
|
const debug = require('debug')('WireGuard');
|
||||||
const crypto = require('node:crypto');
|
const crypto = require('node:crypto');
|
||||||
const QRCode = require('qrcode');
|
const QRCode = require('qrcode');
|
||||||
|
const CRC32 = require('crc-32');
|
||||||
|
|
||||||
const Util = require('./Util');
|
const Util = require('./Util');
|
||||||
const ServerError = require('./ServerError');
|
const ServerError = require('./ServerError');
|
||||||
|
@ -23,6 +24,8 @@ const {
|
||||||
WG_POST_UP,
|
WG_POST_UP,
|
||||||
WG_PRE_DOWN,
|
WG_PRE_DOWN,
|
||||||
WG_POST_DOWN,
|
WG_POST_DOWN,
|
||||||
|
WG_ENABLE_EXPIRES_TIME,
|
||||||
|
WG_ENABLE_ONE_TIME_LINKS,
|
||||||
JC,
|
JC,
|
||||||
JMIN,
|
JMIN,
|
||||||
JMAX,
|
JMAX,
|
||||||
|
@ -182,12 +185,18 @@ ${client.preSharedKey ? `PresharedKey = ${client.preSharedKey}\n` : ''
|
||||||
publicKey: client.publicKey,
|
publicKey: client.publicKey,
|
||||||
createdAt: new Date(client.createdAt),
|
createdAt: new Date(client.createdAt),
|
||||||
updatedAt: new Date(client.updatedAt),
|
updatedAt: new Date(client.updatedAt),
|
||||||
|
expiredAt: client.expiredAt !== null
|
||||||
|
? new Date(client.expiredAt)
|
||||||
|
: null,
|
||||||
allowedIPs: client.allowedIPs,
|
allowedIPs: client.allowedIPs,
|
||||||
|
oneTimeLink: client.oneTimeLink ?? null,
|
||||||
|
oneTimeLinkExpiresAt: client.oneTimeLinkExpiresAt ?? null,
|
||||||
downloadableConfig: 'privateKey' in client,
|
downloadableConfig: 'privateKey' in client,
|
||||||
persistentKeepalive: null,
|
persistentKeepalive: null,
|
||||||
latestHandshakeAt: null,
|
latestHandshakeAt: null,
|
||||||
transferRx: null,
|
transferRx: null,
|
||||||
transferTx: null,
|
transferTx: null,
|
||||||
|
endpoint: null,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Loop WireGuard status
|
// Loop WireGuard status
|
||||||
|
@ -216,6 +225,7 @@ ${client.preSharedKey ? `PresharedKey = ${client.preSharedKey}\n` : ''
|
||||||
client.latestHandshakeAt = latestHandshakeAt === '0'
|
client.latestHandshakeAt = latestHandshakeAt === '0'
|
||||||
? null
|
? null
|
||||||
: new Date(Number(`${latestHandshakeAt}000`));
|
: new Date(Number(`${latestHandshakeAt}000`));
|
||||||
|
client.endpoint = endpoint === '(none)' ? null : endpoint;
|
||||||
client.transferRx = Number(transferRx);
|
client.transferRx = Number(transferRx);
|
||||||
client.transferTx = Number(transferTx);
|
client.transferTx = Number(transferTx);
|
||||||
client.persistentKeepalive = persistentKeepalive;
|
client.persistentKeepalive = persistentKeepalive;
|
||||||
|
@ -270,7 +280,7 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async createClient({ name }) {
|
async createClient({ name, expiredDate }) {
|
||||||
if (!name) {
|
if (!name) {
|
||||||
throw new Error('Missing: Name');
|
throw new Error('Missing: Name');
|
||||||
}
|
}
|
||||||
|
@ -299,7 +309,6 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
if (!address) {
|
if (!address) {
|
||||||
throw new Error('Maximum number of clients reached.');
|
throw new Error('Maximum number of clients reached.');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create Client
|
// Create Client
|
||||||
const id = crypto.randomUUID();
|
const id = crypto.randomUUID();
|
||||||
const client = {
|
const client = {
|
||||||
|
@ -312,10 +321,15 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
|
|
||||||
createdAt: new Date(),
|
createdAt: new Date(),
|
||||||
updatedAt: new Date(),
|
updatedAt: new Date(),
|
||||||
|
expiredAt: null,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
};
|
};
|
||||||
|
if (expiredDate) {
|
||||||
|
client.expiredAt = new Date(expiredDate);
|
||||||
|
client.expiredAt.setHours(23);
|
||||||
|
client.expiredAt.setMinutes(59);
|
||||||
|
client.expiredAt.setSeconds(59);
|
||||||
|
}
|
||||||
config.clients[id] = client;
|
config.clients[id] = client;
|
||||||
|
|
||||||
await this.saveConfig();
|
await this.saveConfig();
|
||||||
|
@ -341,6 +355,23 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
await this.saveConfig();
|
await this.saveConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async generateOneTimeLink({ clientId }) {
|
||||||
|
const client = await this.getClient({ clientId });
|
||||||
|
const key = `${clientId}-${Math.floor(Math.random() * 1000)}`;
|
||||||
|
client.oneTimeLink = Math.abs(CRC32.str(key)).toString(16);
|
||||||
|
client.oneTimeLinkExpiresAt = new Date(Date.now() + 5 * 60 * 1000);
|
||||||
|
client.updatedAt = new Date();
|
||||||
|
await this.saveConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
async eraseOneTimeLink({ clientId }) {
|
||||||
|
const client = await this.getClient({ clientId });
|
||||||
|
client.oneTimeLink = null;
|
||||||
|
client.oneTimeLinkExpiresAt = null;
|
||||||
|
client.updatedAt = new Date();
|
||||||
|
await this.saveConfig();
|
||||||
|
}
|
||||||
|
|
||||||
async disableClient({ clientId }) {
|
async disableClient({ clientId }) {
|
||||||
const client = await this.getClient({ clientId });
|
const client = await this.getClient({ clientId });
|
||||||
|
|
||||||
|
@ -372,6 +403,22 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
await this.saveConfig();
|
await this.saveConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async updateClientExpireDate({ clientId, expireDate }) {
|
||||||
|
const client = await this.getClient({ clientId });
|
||||||
|
|
||||||
|
if (expireDate) {
|
||||||
|
client.expiredAt = new Date(expireDate);
|
||||||
|
client.expiredAt.setHours(23);
|
||||||
|
client.expiredAt.setMinutes(59);
|
||||||
|
client.expiredAt.setSeconds(59);
|
||||||
|
} else {
|
||||||
|
client.expiredAt = null;
|
||||||
|
}
|
||||||
|
client.updatedAt = new Date();
|
||||||
|
|
||||||
|
await this.saveConfig();
|
||||||
|
}
|
||||||
|
|
||||||
async __reloadConfig() {
|
async __reloadConfig() {
|
||||||
await this.__buildConfig();
|
await this.__buildConfig();
|
||||||
await this.__syncConfig();
|
await this.__syncConfig();
|
||||||
|
@ -398,4 +445,107 @@ Endpoint = ${WG_HOST}:${WG_CONFIG_PORT}`;
|
||||||
await Util.exec('wg-quick down wg0').catch(() => {});
|
await Util.exec('wg-quick down wg0').catch(() => {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async cronJobEveryMinute() {
|
||||||
|
const config = await this.getConfig();
|
||||||
|
let needSaveConfig = false;
|
||||||
|
// Expires Feature
|
||||||
|
if (WG_ENABLE_EXPIRES_TIME === 'true') {
|
||||||
|
for (const client of Object.values(config.clients)) {
|
||||||
|
if (client.enabled !== true) continue;
|
||||||
|
if (client.expiredAt !== null && new Date() > new Date(client.expiredAt)) {
|
||||||
|
debug(`Client ${client.id} expired.`);
|
||||||
|
needSaveConfig = true;
|
||||||
|
client.enabled = false;
|
||||||
|
client.updatedAt = new Date();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// One Time Link Feature
|
||||||
|
if (WG_ENABLE_ONE_TIME_LINKS === 'true') {
|
||||||
|
for (const client of Object.values(config.clients)) {
|
||||||
|
if (client.oneTimeLink !== null && new Date() > new Date(client.oneTimeLinkExpiresAt)) {
|
||||||
|
debug(`Client ${client.id} One Time Link expired.`);
|
||||||
|
needSaveConfig = true;
|
||||||
|
client.oneTimeLink = null;
|
||||||
|
client.oneTimeLinkExpiresAt = null;
|
||||||
|
client.updatedAt = new Date();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (needSaveConfig) {
|
||||||
|
await this.saveConfig();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getMetrics() {
|
||||||
|
const clients = await this.getClients();
|
||||||
|
let wireguardPeerCount = 0;
|
||||||
|
let wireguardEnabledPeersCount = 0;
|
||||||
|
let wireguardConnectedPeersCount = 0;
|
||||||
|
let wireguardSentBytes = '';
|
||||||
|
let wireguardReceivedBytes = '';
|
||||||
|
let wireguardLatestHandshakeSeconds = '';
|
||||||
|
for (const client of Object.values(clients)) {
|
||||||
|
wireguardPeerCount++;
|
||||||
|
if (client.enabled === true) {
|
||||||
|
wireguardEnabledPeersCount++;
|
||||||
|
}
|
||||||
|
if (client.endpoint !== null) {
|
||||||
|
wireguardConnectedPeersCount++;
|
||||||
|
}
|
||||||
|
wireguardSentBytes += `wireguard_sent_bytes{interface="wg0",enabled="${client.enabled}",address="${client.address}",name="${client.name}"} ${Number(client.transferTx)}\n`;
|
||||||
|
wireguardReceivedBytes += `wireguard_received_bytes{interface="wg0",enabled="${client.enabled}",address="${client.address}",name="${client.name}"} ${Number(client.transferRx)}\n`;
|
||||||
|
wireguardLatestHandshakeSeconds += `wireguard_latest_handshake_seconds{interface="wg0",enabled="${client.enabled}",address="${client.address}",name="${client.name}"} ${client.latestHandshakeAt ? (new Date().getTime() - new Date(client.latestHandshakeAt).getTime()) / 1000 : 0}\n`;
|
||||||
|
}
|
||||||
|
|
||||||
|
let returnText = '# HELP wg-easy and wireguard metrics\n';
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_configured_peers\n';
|
||||||
|
returnText += '# TYPE wireguard_configured_peers gauge\n';
|
||||||
|
returnText += `wireguard_configured_peers{interface="wg0"} ${Number(wireguardPeerCount)}\n`;
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_enabled_peers\n';
|
||||||
|
returnText += '# TYPE wireguard_enabled_peers gauge\n';
|
||||||
|
returnText += `wireguard_enabled_peers{interface="wg0"} ${Number(wireguardEnabledPeersCount)}\n`;
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_connected_peers\n';
|
||||||
|
returnText += '# TYPE wireguard_connected_peers gauge\n';
|
||||||
|
returnText += `wireguard_connected_peers{interface="wg0"} ${Number(wireguardConnectedPeersCount)}\n`;
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_sent_bytes Bytes sent to the peer\n';
|
||||||
|
returnText += '# TYPE wireguard_sent_bytes counter\n';
|
||||||
|
returnText += `${wireguardSentBytes}`;
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_received_bytes Bytes received from the peer\n';
|
||||||
|
returnText += '# TYPE wireguard_received_bytes counter\n';
|
||||||
|
returnText += `${wireguardReceivedBytes}`;
|
||||||
|
|
||||||
|
returnText += '\n# HELP wireguard_latest_handshake_seconds UNIX timestamp seconds of the last handshake\n';
|
||||||
|
returnText += '# TYPE wireguard_latest_handshake_seconds gauge\n';
|
||||||
|
returnText += `${wireguardLatestHandshakeSeconds}`;
|
||||||
|
|
||||||
|
return returnText;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getMetricsJSON() {
|
||||||
|
const clients = await this.getClients();
|
||||||
|
let wireguardPeerCount = 0;
|
||||||
|
let wireguardEnabledPeersCount = 0;
|
||||||
|
let wireguardConnectedPeersCount = 0;
|
||||||
|
for (const client of Object.values(clients)) {
|
||||||
|
wireguardPeerCount++;
|
||||||
|
if (client.enabled === true) {
|
||||||
|
wireguardEnabledPeersCount++;
|
||||||
|
}
|
||||||
|
if (client.endpoint !== null) {
|
||||||
|
wireguardConnectedPeersCount++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
wireguard_configured_peers: Number(wireguardPeerCount),
|
||||||
|
wireguard_enabled_peers: Number(wireguardEnabledPeersCount),
|
||||||
|
wireguard_connected_peers: Number(wireguardConnectedPeersCount),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -9,16 +9,19 @@
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"license": "CC BY-NC-SA 4.0",
|
"license": "CC BY-NC-SA 4.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"basic-auth": "^2.0.1",
|
||||||
"bcryptjs": "^2.4.3",
|
"bcryptjs": "^2.4.3",
|
||||||
"debug": "^4.3.6",
|
"crc-32": "^1.2.2",
|
||||||
|
"debug": "^4.3.7",
|
||||||
"express-session": "^1.18.0",
|
"express-session": "^1.18.0",
|
||||||
"h3": "^1.12.0",
|
"h3": "^1.12.0",
|
||||||
"qrcode": "^1.5.4"
|
"qrcode": "^1.5.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@tailwindcss/forms": "^0.5.9",
|
||||||
"eslint-config-athom": "^3.1.3",
|
"eslint-config-athom": "^3.1.3",
|
||||||
"nodemon": "^3.1.4",
|
"nodemon": "^3.1.4",
|
||||||
"tailwindcss": "^3.4.9"
|
"tailwindcss": "^3.4.10"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
|
@ -451,6 +454,26 @@
|
||||||
"node": ">=14"
|
"node": ">=14"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@rtsao/scc": {
|
||||||
|
"version": "1.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz",
|
||||||
|
"integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/@tailwindcss/forms": {
|
||||||
|
"version": "0.5.9",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tailwindcss/forms/-/forms-0.5.9.tgz",
|
||||||
|
"integrity": "sha512-tM4XVr2+UVTxXJzey9Twx48c1gcxFStqn1pQz0tRsX8o3DvxhN5oY5pvyAbUx7VTaZxpej4Zzvc6h+1RJBzpIg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"mini-svg-data-uri": "^1.2.3"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"tailwindcss": ">=3.0.0 || >= 3.0.0-alpha.1 || >= 4.0.0-alpha.20"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/json-schema": {
|
"node_modules/@types/json-schema": {
|
||||||
"version": "7.0.15",
|
"version": "7.0.15",
|
||||||
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
|
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
|
||||||
|
@ -977,6 +1000,18 @@
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/basic-auth": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"safe-buffer": "5.1.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/bcryptjs": {
|
"node_modules/bcryptjs": {
|
||||||
"version": "2.4.3",
|
"version": "2.4.3",
|
||||||
"resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
|
"resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
|
||||||
|
@ -1209,6 +1244,18 @@
|
||||||
"integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==",
|
"integrity": "sha512-NXdYc3dLr47pBkpUCHtKSwIOQXLVn8dZEuywboCOJY/osA0wFSLlSawr3KN8qXJEyX66FcONTH8EIlVuK0yyFA==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/crc-32": {
|
||||||
|
"version": "1.2.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz",
|
||||||
|
"integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"bin": {
|
||||||
|
"crc32": "bin/crc32.njs"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/cross-spawn": {
|
"node_modules/cross-spawn": {
|
||||||
"version": "7.0.3",
|
"version": "7.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
|
||||||
|
@ -1306,12 +1353,11 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/debug": {
|
"node_modules/debug": {
|
||||||
"version": "4.3.6",
|
"version": "4.3.7",
|
||||||
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.6.tgz",
|
"resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz",
|
||||||
"integrity": "sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==",
|
"integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==",
|
||||||
"license": "MIT",
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"ms": "2.1.2"
|
"ms": "^2.1.3"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=6.0"
|
"node": ">=6.0"
|
||||||
|
@ -1749,9 +1795,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/eslint-module-utils": {
|
"node_modules/eslint-module-utils": {
|
||||||
"version": "2.8.1",
|
"version": "2.9.0",
|
||||||
"resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.1.tgz",
|
"resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.9.0.tgz",
|
||||||
"integrity": "sha512-rXDXR3h7cs7dy9RNpUlQf80nX31XWJEyGq1tRMo+6GsO5VmTe4UTwtmonAD4ZkAsrfMVDA2wlGJ3790Ys+D49Q==",
|
"integrity": "sha512-McVbYmwA3NEKwRQY5g4aWMdcZE5xZxV8i8l7CqJSrameuGSQJtSWaL/LxTEzSKKaCcOhlpDR8XEfYXWPrdo/ZQ==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -1810,27 +1856,28 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/eslint-plugin-import": {
|
"node_modules/eslint-plugin-import": {
|
||||||
"version": "2.29.1",
|
"version": "2.30.0",
|
||||||
"resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.29.1.tgz",
|
"resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.30.0.tgz",
|
||||||
"integrity": "sha512-BbPC0cuExzhiMo4Ff1BTVwHpjjv28C5R+btTOGaCRC7UEz801up0JadwkeSk5Ued6TG34uaczuVuH6qyy5YUxw==",
|
"integrity": "sha512-/mHNE9jINJfiD2EKkg1BKyPyUk4zdnT54YgbOgfjSakWT5oyX/qQLVNTkehyfpcMxZXMy1zyonZ2v7hZTX43Yw==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"array-includes": "^3.1.7",
|
"@rtsao/scc": "^1.1.0",
|
||||||
"array.prototype.findlastindex": "^1.2.3",
|
"array-includes": "^3.1.8",
|
||||||
|
"array.prototype.findlastindex": "^1.2.5",
|
||||||
"array.prototype.flat": "^1.3.2",
|
"array.prototype.flat": "^1.3.2",
|
||||||
"array.prototype.flatmap": "^1.3.2",
|
"array.prototype.flatmap": "^1.3.2",
|
||||||
"debug": "^3.2.7",
|
"debug": "^3.2.7",
|
||||||
"doctrine": "^2.1.0",
|
"doctrine": "^2.1.0",
|
||||||
"eslint-import-resolver-node": "^0.3.9",
|
"eslint-import-resolver-node": "^0.3.9",
|
||||||
"eslint-module-utils": "^2.8.0",
|
"eslint-module-utils": "^2.9.0",
|
||||||
"hasown": "^2.0.0",
|
"hasown": "^2.0.2",
|
||||||
"is-core-module": "^2.13.1",
|
"is-core-module": "^2.15.1",
|
||||||
"is-glob": "^4.0.3",
|
"is-glob": "^4.0.3",
|
||||||
"minimatch": "^3.1.2",
|
"minimatch": "^3.1.2",
|
||||||
"object.fromentries": "^2.0.7",
|
"object.fromentries": "^2.0.8",
|
||||||
"object.groupby": "^1.0.1",
|
"object.groupby": "^1.0.3",
|
||||||
"object.values": "^1.1.7",
|
"object.values": "^1.2.0",
|
||||||
"semver": "^6.3.1",
|
"semver": "^6.3.1",
|
||||||
"tsconfig-paths": "^3.15.0"
|
"tsconfig-paths": "^3.15.0"
|
||||||
},
|
},
|
||||||
|
@ -2159,6 +2206,26 @@
|
||||||
"integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
|
"integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/express-session/node_modules/safe-buffer": {
|
||||||
|
"version": "5.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
|
||||||
|
"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/fast-deep-equal": {
|
"node_modules/fast-deep-equal": {
|
||||||
"version": "3.1.3",
|
"version": "3.1.3",
|
||||||
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
|
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
|
||||||
|
@ -2658,9 +2725,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/ignore": {
|
"node_modules/ignore": {
|
||||||
"version": "5.3.1",
|
"version": "5.3.2",
|
||||||
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz",
|
||||||
"integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==",
|
"integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"engines": {
|
"engines": {
|
||||||
|
@ -2822,9 +2889,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/is-core-module": {
|
"node_modules/is-core-module": {
|
||||||
"version": "2.15.0",
|
"version": "2.15.1",
|
||||||
"resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.0.tgz",
|
"resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.1.tgz",
|
||||||
"integrity": "sha512-Dd+Lb2/zvk9SKy1TGCt1wFJFo/MWBPMX5x7KcvLajWTGuomczdQX61PvY5yK6SVACwpoexWo81IfFyoKY2QnTA==",
|
"integrity": "sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -3223,9 +3290,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/micromatch": {
|
"node_modules/micromatch": {
|
||||||
"version": "4.0.7",
|
"version": "4.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.7.tgz",
|
"resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
|
||||||
"integrity": "sha512-LPP/3KorzCwBxfeUuZmaR6bG2kdeHSbe0P2tY3FLRU4vYrjYz5hI4QZwV0njUx3jeuKe67YukQ1LSPZBKDqO/Q==",
|
"integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -3248,6 +3315,16 @@
|
||||||
"node": ">=10.0.0"
|
"node": ">=10.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/mini-svg-data-uri": {
|
||||||
|
"version": "1.4.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/mini-svg-data-uri/-/mini-svg-data-uri-1.4.4.tgz",
|
||||||
|
"integrity": "sha512-r9deDe9p5FJUPZAk3A59wGH7Ii9YrjjWw0jmw/liSbHl2CHiyXj6FcDXDu2K3TjVAXqiJdaw3xxwlZZr9E6nHg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"bin": {
|
||||||
|
"mini-svg-data-uri": "cli.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/minimatch": {
|
"node_modules/minimatch": {
|
||||||
"version": "3.1.2",
|
"version": "3.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
|
||||||
|
@ -3282,10 +3359,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/ms": {
|
"node_modules/ms": {
|
||||||
"version": "2.1.2",
|
"version": "2.1.3",
|
||||||
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
|
||||||
"integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==",
|
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
|
||||||
"license": "MIT"
|
|
||||||
},
|
},
|
||||||
"node_modules/mz": {
|
"node_modules/mz": {
|
||||||
"version": "2.7.0",
|
"version": "2.7.0",
|
||||||
|
@ -3704,9 +3780,9 @@
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/picocolors": {
|
"node_modules/picocolors": {
|
||||||
"version": "1.0.1",
|
"version": "1.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.0.tgz",
|
||||||
"integrity": "sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==",
|
"integrity": "sha512-TQ92mBOW0l3LeMeyLV6mzy/kWr8lkd/hp3mTg7wYK7zJhuBStmGMBG0BdeDZS/dZx1IukaX6Bk11zcln25o1Aw==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
|
@ -3763,9 +3839,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/postcss": {
|
"node_modules/postcss": {
|
||||||
"version": "8.4.41",
|
"version": "8.4.45",
|
||||||
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.41.tgz",
|
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.45.tgz",
|
||||||
"integrity": "sha512-TesUflQ0WKZqAvg52PWL6kHgLKP6xB6heTOdoYM0Wt2UHyxNa4K25EZZMgKns3BH1RLVbZCREPpLY0rhnNoHVQ==",
|
"integrity": "sha512-7KTLTdzdZZYscUc65XmjFiB73vBhBfbPztCYdUNvlaso9PrzjzcmjqBPR0lNGkcVlcO4BjiO5rK/qNz+XAen1Q==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
|
@ -3905,9 +3981,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/postcss-selector-parser": {
|
"node_modules/postcss-selector-parser": {
|
||||||
"version": "6.1.1",
|
"version": "6.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz",
|
||||||
"integrity": "sha512-b4dlw/9V8A71rLIDsSwVmak9z2DuBUB7CA1/wSdelNEzqsjoSPeADTWNO09lpH49Diy3/JIZ2bSPB1dI3LJCHg==",
|
"integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -4208,23 +4284,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/safe-buffer": {
|
"node_modules/safe-buffer": {
|
||||||
"version": "5.2.1",
|
"version": "5.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
|
"resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
|
||||||
"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
|
"integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==",
|
||||||
"funding": [
|
|
||||||
{
|
|
||||||
"type": "github",
|
|
||||||
"url": "https://github.com/sponsors/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "patreon",
|
|
||||||
"url": "https://www.patreon.com/feross"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"type": "consulting",
|
|
||||||
"url": "https://feross.org/support"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/safe-regex-test": {
|
"node_modules/safe-regex-test": {
|
||||||
|
@ -4687,9 +4749,9 @@
|
||||||
"peer": true
|
"peer": true
|
||||||
},
|
},
|
||||||
"node_modules/tailwindcss": {
|
"node_modules/tailwindcss": {
|
||||||
"version": "3.4.9",
|
"version": "3.4.10",
|
||||||
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.9.tgz",
|
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.10.tgz",
|
||||||
"integrity": "sha512-1SEOvRr6sSdV5IDf9iC+NU4dhwdqzF4zKKq3sAbasUWHEM6lsMhX+eNN5gkPx1BvLFEnZQEUFbXnGj8Qlp83Pg==",
|
"integrity": "sha512-KWZkVPm7yJRhdu4SRSl9d4AK2wM3a50UsvgHZO7xY77NQr2V+fIrEuoDGQcbvswWvFGbS2f6e+jC/6WJm1Dl0w==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -5134,9 +5196,9 @@
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
"node_modules/yaml": {
|
"node_modules/yaml": {
|
||||||
"version": "2.5.0",
|
"version": "2.5.1",
|
||||||
"resolved": "https://registry.npmjs.org/yaml/-/yaml-2.5.0.tgz",
|
"resolved": "https://registry.npmjs.org/yaml/-/yaml-2.5.1.tgz",
|
||||||
"integrity": "sha512-2wWLbGbYDiSqqIKoPjar3MPgB94ErzCtrNE1FdqGuaO0pi2JGjmE8aW8TDZwzU7vuxcGRdL/4gPQwQ7hD5AMSw==",
|
"integrity": "sha512-bLQOjaX/ADgQ20isPJRvF0iRUHIxVhYvr53Of7wGcWlO2jvtUlH5m87DsmulFVxRpNLOnI4tB6p/oh8D7kpn9Q==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"bin": {
|
"bin": {
|
||||||
|
|
|
@ -15,16 +15,19 @@
|
||||||
"author": "Emile Nijssen",
|
"author": "Emile Nijssen",
|
||||||
"license": "CC BY-NC-SA 4.0",
|
"license": "CC BY-NC-SA 4.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"basic-auth": "^2.0.1",
|
||||||
"bcryptjs": "^2.4.3",
|
"bcryptjs": "^2.4.3",
|
||||||
"debug": "^4.3.6",
|
"crc-32": "^1.2.2",
|
||||||
|
"debug": "^4.3.7",
|
||||||
"express-session": "^1.18.0",
|
"express-session": "^1.18.0",
|
||||||
"h3": "^1.12.0",
|
"h3": "^1.12.0",
|
||||||
"qrcode": "^1.5.4"
|
"qrcode": "^1.5.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@tailwindcss/forms": "^0.5.9",
|
||||||
"eslint-config-athom": "^3.1.3",
|
"eslint-config-athom": "^3.1.3",
|
||||||
"nodemon": "^3.1.4",
|
"nodemon": "^3.1.4",
|
||||||
"tailwindcss": "^3.4.9"
|
"tailwindcss": "^3.4.10"
|
||||||
},
|
},
|
||||||
"nodemonConfig": {
|
"nodemonConfig": {
|
||||||
"ignore": [
|
"ignore": [
|
||||||
|
|
30
src/wgpw.mjs
30
src/wgpw.mjs
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
// Import needed libraries
|
// Import needed libraries
|
||||||
import bcrypt from 'bcryptjs';
|
import bcrypt from 'bcryptjs';
|
||||||
|
import { Writable } from 'stream';
|
||||||
|
import readline from 'readline';
|
||||||
|
|
||||||
// Function to generate hash
|
// Function to generate hash
|
||||||
const generateHash = async (password) => {
|
const generateHash = async (password) => {
|
||||||
|
@ -31,12 +33,35 @@ const comparePassword = async (password, hash) => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const readStdinPassword = () => {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
process.stdout.write('Enter your password: ');
|
||||||
|
|
||||||
|
const rl = readline.createInterface({
|
||||||
|
input: process.stdin,
|
||||||
|
output: new Writable({
|
||||||
|
write(_chunk, _encoding, callback) {
|
||||||
|
callback();
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
terminal: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
rl.question('', (answer) => {
|
||||||
|
rl.close();
|
||||||
|
// Print a new line after password prompt
|
||||||
|
process.stdout.write('\n');
|
||||||
|
resolve(answer);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
(async () => {
|
(async () => {
|
||||||
try {
|
try {
|
||||||
// Retrieve command line arguments
|
// Retrieve command line arguments
|
||||||
const args = process.argv.slice(2); // Ignore the first two arguments
|
const args = process.argv.slice(2); // Ignore the first two arguments
|
||||||
if (args.length > 2) {
|
if (args.length > 2) {
|
||||||
throw new Error('Usage : wgpw YOUR_PASSWORD [HASH]');
|
throw new Error('Usage : wgpw [YOUR_PASSWORD] [HASH]');
|
||||||
}
|
}
|
||||||
|
|
||||||
const [password, hash] = args;
|
const [password, hash] = args;
|
||||||
|
@ -44,6 +69,9 @@ const comparePassword = async (password, hash) => {
|
||||||
await comparePassword(password, hash);
|
await comparePassword(password, hash);
|
||||||
} else if (password) {
|
} else if (password) {
|
||||||
await generateHash(password);
|
await generateHash(password);
|
||||||
|
} else {
|
||||||
|
const password = await readStdinPassword();
|
||||||
|
await generateHash(password);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
! tailwindcss v3.4.9 | MIT License | https://tailwindcss.com
|
! tailwindcss v3.4.10 | MIT License | https://tailwindcss.com
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -714,6 +714,10 @@ video {
|
||||||
margin-bottom: 2.5rem;
|
margin-bottom: 2.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.mb-2 {
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
.mb-4 {
|
.mb-4 {
|
||||||
margin-bottom: 1rem;
|
margin-bottom: 1rem;
|
||||||
}
|
}
|
||||||
|
@ -1160,6 +1164,10 @@ video {
|
||||||
fill: #4b5563;
|
fill: #4b5563;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-0 {
|
||||||
|
padding: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
.p-1 {
|
.p-1 {
|
||||||
padding: 0.25rem;
|
padding: 0.25rem;
|
||||||
}
|
}
|
||||||
|
@ -1465,6 +1473,10 @@ video {
|
||||||
cursor: default;
|
cursor: default;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-0 {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
.last\:border-b-0:last-child {
|
.last\:border-b-0:last-child {
|
||||||
border-bottom-width: 0px;
|
border-bottom-width: 0px;
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,7 +43,7 @@
|
||||||
<path stroke-linecap="round" stroke-linejoin="round"
|
<path stroke-linecap="round" stroke-linejoin="round"
|
||||||
d="M21.752 15.002A9.72 9.72 0 0 1 18 15.75c-5.385 0-9.75-4.365-9.75-9.75 0-1.33.266-2.597.748-3.752A9.753 9.753 0 0 0 3 11.25C3 16.635 7.365 21 12.75 21a9.753 9.753 0 0 0 9.002-5.998Z" />
|
d="M21.752 15.002A9.72 9.72 0 0 1 18 15.75c-5.385 0-9.75-4.365-9.75-9.75 0-1.33.266-2.597.748-3.752A9.753 9.753 0 0 0 3 11.25C3 16.635 7.365 21 12.75 21a9.753 9.753 0 0 0 9.002-5.998Z" />
|
||||||
</svg>
|
</svg>
|
||||||
<svg v-else xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 24 24"
|
<svg v-else xmlns="http://www.w3.org/2000/svg" fill="currentColor" viewBox="0 0 24 24"
|
||||||
class="w-5 h-5 fill-gray-600 dark:fill-neutral-400">
|
class="w-5 h-5 fill-gray-600 dark:fill-neutral-400">
|
||||||
<path
|
<path
|
||||||
d="M12,2.2c-5.4,0-9.8,4.4-9.8,9.8s4.4,9.8,9.8,9.8s9.8-4.4,9.8-9.8S17.4,2.2,12,2.2z M3.8,12c0-4.5,3.7-8.2,8.2-8.2v16.5C7.5,20.2,3.8,16.5,3.8,12z" />
|
d="M12,2.2c-5.4,0-9.8,4.4-9.8,9.8s4.4,9.8,9.8,9.8s9.8-4.4,9.8-9.8S17.4,2.2,12,2.2z M3.8,12c0-4.5,3.7-8.2,8.2-8.2v16.5C7.5,20.2,3.8,16.5,3.8,12z" />
|
||||||
|
@ -112,8 +112,22 @@
|
||||||
</svg>
|
</svg>
|
||||||
<span class="max-md:hidden text-sm">{{$t("backup")}}</span>
|
<span class="max-md:hidden text-sm">{{$t("backup")}}</span>
|
||||||
</a>
|
</a>
|
||||||
|
<!-- Sort client -->
|
||||||
|
<button v-if="enableSortClient" @click="sortClient = !sortClient;"
|
||||||
|
class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 dark:text-neutral-200 max-md:border-x-0 border-2 border-gray-100 dark:border-neutral-600 py-2 px-4 md:rounded inline-flex items-center transition">
|
||||||
|
<svg v-if="sortClient === true" inline class="w-4 md:mr-2" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="size-6">
|
||||||
|
<path d="M12 19.75C11.9015 19.7504 11.8038 19.7312 11.7128 19.6934C11.6218 19.6557 11.5392 19.6001 11.47 19.53L5.47 13.53C5.33752 13.3878 5.2654 13.1997 5.26882 13.0054C5.27225 12.8111 5.35096 12.6258 5.48838 12.4883C5.62579 12.3509 5.81118 12.2722 6.00548 12.2688C6.19978 12.2654 6.38782 12.3375 6.53 12.47L12 17.94L17.47 12.47C17.6122 12.3375 17.8002 12.2654 17.9945 12.2688C18.1888 12.2722 18.3742 12.3509 18.5116 12.4883C18.649 12.6258 18.7277 12.8111 18.7312 13.0054C18.7346 13.1997 18.6625 13.3878 18.53 13.53L12.53 19.53C12.4608 19.6001 12.3782 19.6557 12.2872 19.6934C12.1962 19.7312 12.0985 19.7504 12 19.75Z" fill="#000000"/>
|
||||||
|
<path d="M12 19.75C11.8019 19.7474 11.6126 19.6676 11.4725 19.5275C11.3324 19.3874 11.2526 19.1981 11.25 19V5C11.25 4.80109 11.329 4.61032 11.4697 4.46967C11.6103 4.32902 11.8011 4.25 12 4.25C12.1989 4.25 12.3897 4.32902 12.5303 4.46967C12.671 4.61032 12.75 4.80109 12.75 5V19C12.7474 19.1981 12.6676 19.3874 12.5275 19.5275C12.3874 19.6676 12.1981 19.7474 12 19.75Z" fill="#000000"/>
|
||||||
|
</svg>
|
||||||
|
<svg v-if="sortClient === false" inline class="w-4 md:mr-2" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" stroke-width="1.5" stroke="currentColor" class="size-6">
|
||||||
|
<path d="M18 11.75C17.9015 11.7505 17.8038 11.7313 17.7128 11.6935C17.6218 11.6557 17.5392 11.6001 17.47 11.53L12 6.06001L6.53 11.53C6.38782 11.6625 6.19978 11.7346 6.00548 11.7312C5.81118 11.7278 5.62579 11.649 5.48838 11.5116C5.35096 11.3742 5.27225 11.1888 5.26882 10.9945C5.2654 10.8002 5.33752 10.6122 5.47 10.47L11.47 4.47001C11.6106 4.32956 11.8012 4.25067 12 4.25067C12.1987 4.25067 12.3894 4.32956 12.53 4.47001L18.53 10.47C18.6705 10.6106 18.7493 10.8013 18.7493 11C18.7493 11.1988 18.6705 11.3894 18.53 11.53C18.4608 11.6001 18.3782 11.6557 18.2872 11.6935C18.1962 11.7313 18.0985 11.7505 18 11.75Z" fill="#000000"/>
|
||||||
|
<path d="M12 19.75C11.8019 19.7474 11.6126 19.6676 11.4725 19.5275C11.3324 19.3874 11.2526 19.1981 11.25 19V5C11.25 4.80109 11.329 4.61032 11.4697 4.46967C11.6103 4.32902 11.8011 4.25 12 4.25C12.1989 4.25 12.3897 4.32902 12.5303 4.46967C12.671 4.61032 12.75 4.80109 12.75 5V19C12.7474 19.1981 12.6676 19.3874 12.5275 19.5275C12.3874 19.6676 12.1981 19.7474 12 19.75Z" fill="#000000"/>
|
||||||
|
</svg>
|
||||||
|
<span class="max-md:hidden text-sm">{{$t("sort")}}</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
<!-- New client -->
|
<!-- New client -->
|
||||||
<button @click="clientCreate = true; clientCreateName = '';"
|
<button @click="clientCreate = true; clientCreateName = ''; clientExpiredDate = '';"
|
||||||
class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 dark:text-neutral-200 max-md:border-l-0 border-2 border-gray-100 dark:border-neutral-600 py-2 px-4 rounded-r-full md:rounded inline-flex items-center transition">
|
class="hover:bg-red-800 hover:border-red-800 hover:text-white text-gray-700 dark:text-neutral-200 max-md:border-l-0 border-2 border-gray-100 dark:border-neutral-600 py-2 px-4 rounded-r-full md:rounded inline-flex items-center transition">
|
||||||
<svg class="w-4 md:mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
|
<svg class="w-4 md:mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
|
||||||
stroke="currentColor">
|
stroke="currentColor">
|
||||||
|
@ -225,7 +239,7 @@
|
||||||
</svg>
|
</svg>
|
||||||
{{client.transferTxCurrent | bytes}}/s
|
{{client.transferTxCurrent | bytes}}/s
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<!-- Inline Transfer RX -->
|
<!-- Inline Transfer RX -->
|
||||||
<span v-if="!uiTrafficStats && client.transferRx" class="whitespace-nowrap" :title="$t('totalUpload') + bytes(client.transferRx)">
|
<span v-if="!uiTrafficStats && client.transferRx" class="whitespace-nowrap" :title="$t('totalUpload') + bytes(client.transferRx)">
|
||||||
·
|
·
|
||||||
|
@ -242,6 +256,35 @@
|
||||||
{{!uiTrafficStats ? " · " : ""}}{{new Date(client.latestHandshakeAt) | timeago}}
|
{{!uiTrafficStats ? " · " : ""}}{{new Date(client.latestHandshakeAt) | timeago}}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
<div v-if="enableOneTimeLinks && client.oneTimeLink !== null && client.oneTimeLink !== ''" :ref="'client-' + client.id + '-link'" class="text-gray-400 text-xs">
|
||||||
|
<a :href="'./cnf/' + client.oneTimeLink + ''">{{document.location.protocol}}//{{document.location.host}}/cnf/{{client.oneTimeLink}}</a>
|
||||||
|
</div>
|
||||||
|
<!-- Expire Date -->
|
||||||
|
<div v-show="enableExpireTime" class=" block md:inline-block pb-1 md:pb-0 text-gray-500 dark:text-neutral-400 text-xs">
|
||||||
|
<span class="group">
|
||||||
|
<!-- Show -->
|
||||||
|
<input v-show="clientEditExpireDateId === client.id" v-model="clientEditExpireDate"
|
||||||
|
v-on:keyup.enter="updateClientExpireDate(client, clientEditExpireDate); clientEditExpireDate = null; clientEditExpireDateId = null;"
|
||||||
|
v-on:keyup.escape="clientEditExpireDate = null; clientEditExpireDateId = null;"
|
||||||
|
:ref="'client-' + client.id + '-expire'"
|
||||||
|
type="text"
|
||||||
|
class="rounded border-2 dark:bg-neutral-700 border-gray-100 dark:border-neutral-600 focus:border-gray-200 dark:focus:border-neutral-500 outline-none w-70 text-black dark:text-neutral-300 dark:placeholder:text-neutral-500 text-xs p-0" />
|
||||||
|
<span v-show="clientEditExpireDateId !== client.id"
|
||||||
|
class="inline-block ">{{client.expiredAt | expiredDateFormat}}</span>
|
||||||
|
|
||||||
|
<!-- Edit -->
|
||||||
|
<span v-show="clientEditExpireDateId !== client.id"
|
||||||
|
@click="clientEditExpireDate = client.expiredAt ? client.expiredAt.toISOString().slice(0, 10) : 'yyyy-mm-dd'; clientEditExpireDateId = client.id; setTimeout(() => $refs['client-' + client.id + '-expire'][0].select(), 1);"
|
||||||
|
class="cursor-pointer opacity-0 group-hover:opacity-100 transition-opacity">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg"
|
||||||
|
class="h-4 w-4 inline align-middle opacity-25 hover:opacity-100" fill="none"
|
||||||
|
viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
|
||||||
|
d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z" />
|
||||||
|
</svg>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Info -->
|
<!-- Info -->
|
||||||
|
@ -341,6 +384,22 @@
|
||||||
</svg>
|
</svg>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
|
<!-- Short OneTime Link -->
|
||||||
|
<button v-if="enableOneTimeLinks" :disabled="!client.downloadableConfig"
|
||||||
|
class="align-middle inline-block bg-gray-100 dark:bg-neutral-600 dark:text-neutral-300 p-2 rounded transition"
|
||||||
|
:class="{
|
||||||
|
'hover:bg-red-800 dark:hover:bg-red-800 hover:text-white dark:hover:text-white': client.downloadableConfig,
|
||||||
|
'is-disabled': !client.downloadableConfig
|
||||||
|
}"
|
||||||
|
:title="!client.downloadableConfig ? $t('noPrivKey') : $t('OneTimeLink')"
|
||||||
|
@click="if(client.downloadableConfig) { showOneTimeLink(client); }">
|
||||||
|
<svg class="w-5" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
|
||||||
|
stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
|
||||||
|
d="M13.213 9.787a3.391 3.391 0 0 0-4.795 0l-3.425 3.426a3.39 3.39 0 0 0 4.795 4.794l.321-.304m-.321-4.49a3.39 3.39 0 0 0 4.795 0l3.424-3.426a3.39 3.39 0 0 0-4.794-4.795l-1.028.961"/>
|
||||||
|
</svg>
|
||||||
|
</button>
|
||||||
|
|
||||||
<!-- Delete -->
|
<!-- Delete -->
|
||||||
|
|
||||||
<button
|
<button
|
||||||
|
@ -361,7 +420,7 @@
|
||||||
<div v-if="clients && clients.length === 0">
|
<div v-if="clients && clients.length === 0">
|
||||||
<p class="text-center m-10 text-gray-400 dark:text-neutral-400 text-sm">
|
<p class="text-center m-10 text-gray-400 dark:text-neutral-400 text-sm">
|
||||||
{{$t("noClients")}}<br /><br />
|
{{$t("noClients")}}<br /><br />
|
||||||
<button @click="clientCreate = true; clientCreateName = '';"
|
<button @click="clientCreate = true; clientCreateName = ''; clientExpiredDate = '';"
|
||||||
class="bg-red-800 hover:bg-red-700 text-white border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
|
class="bg-red-800 hover:bg-red-700 text-white border-2 border-none py-2 px-4 rounded inline-flex items-center transition">
|
||||||
<svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
|
<svg class="w-4 mr-2" inline xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24"
|
||||||
stroke="currentColor">
|
stroke="currentColor">
|
||||||
|
@ -453,6 +512,16 @@
|
||||||
type="text" v-model.trim="clientCreateName" :placeholder="$t('name')" />
|
type="text" v-model.trim="clientCreateName" :placeholder="$t('name')" />
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="mt-2" v-show="enableExpireTime">
|
||||||
|
<p class="text-sm text-gray-500">
|
||||||
|
<label class="block text-gray-900 dark:text-neutral-200 text-sm font-bold mb-2" for="expireDate">
|
||||||
|
{{$t("ExpireDate")}}
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
class="rounded p-2 border-2 dark:bg-neutral-700 dark:text-neutral-200 border-gray-100 dark:border-neutral-600 focus:border-gray-200 focus:dark:border-neutral-500 dark:placeholder:text-neutral-400 outline-none w-full"
|
||||||
|
type="date" v-model.trim="clientExpiredDate" :placeholder="$t('expireDate')" name="expireDate"/>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -563,6 +632,24 @@
|
||||||
<input type="password" name="password" :placeholder="$t('password')" v-model="password" autocomplete="current-password"
|
<input type="password" name="password" :placeholder="$t('password')" v-model="password" autocomplete="current-password"
|
||||||
class="px-3 py-2 text-sm dark:bg-neutral-700 text-gray-500 dark:text-gray-500 mb-5 border-2 border-gray-100 dark:border-neutral-800 rounded-lg w-full focus:border-red-800 dark:focus:border-red-800 dark:placeholder:text-neutral-400 outline-none" />
|
class="px-3 py-2 text-sm dark:bg-neutral-700 text-gray-500 dark:text-gray-500 mb-5 border-2 border-gray-100 dark:border-neutral-800 rounded-lg w-full focus:border-red-800 dark:focus:border-red-800 dark:placeholder:text-neutral-400 outline-none" />
|
||||||
|
|
||||||
|
<!-- Remember me -->
|
||||||
|
<label v-if="rememberMeEnabled"
|
||||||
|
class="inline-block mb-5 cursor-pointer whitespace-nowrap" :title="$t('titleRememberMe')">
|
||||||
|
<input type="checkbox" class="sr-only" v-model="remember">
|
||||||
|
|
||||||
|
<div v-if="remember"
|
||||||
|
class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-red-800 cursor-pointer hover:bg-red-700 transition-all">
|
||||||
|
<div class="rounded-full w-4 h-4 m-1 ml-5 bg-white"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="!remember"
|
||||||
|
class="inline-block align-middle rounded-full w-10 h-6 mr-1 bg-gray-200 dark:bg-neutral-400 cursor-pointer hover:bg-gray-300 dark:hover:bg-neutral-500 transition-all">
|
||||||
|
<div class="rounded-full w-4 h-4 m-1 bg-white"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span class="text-sm">{{$t("rememberMe")}}</span>
|
||||||
|
</label>
|
||||||
|
|
||||||
<button v-if="authenticating"
|
<button v-if="authenticating"
|
||||||
class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed">
|
class="bg-red-800 dark:bg-red-800 w-full rounded shadow py-2 text-sm text-white dark:text-white cursor-not-allowed">
|
||||||
<svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
|
<svg class="w-5 animate-spin mx-auto" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"
|
||||||
|
|
|
@ -43,6 +43,13 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getRememberMeEnabled() {
|
||||||
|
return this.call({
|
||||||
|
method: 'get',
|
||||||
|
path: '/remember-me',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async getuiTrafficStats() {
|
async getuiTrafficStats() {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'get',
|
method: 'get',
|
||||||
|
@ -57,6 +64,20 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getWGEnableOneTimeLinks() {
|
||||||
|
return this.call({
|
||||||
|
method: 'get',
|
||||||
|
path: '/wg-enable-one-time-links',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async getWGEnableExpireTime() {
|
||||||
|
return this.call({
|
||||||
|
method: 'get',
|
||||||
|
path: '/wg-enable-expire-time',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async getAvatarSettings() {
|
async getAvatarSettings() {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'get',
|
method: 'get',
|
||||||
|
@ -71,11 +92,11 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async createSession({ password }) {
|
async createSession({ password, remember }) {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'post',
|
method: 'post',
|
||||||
path: '/session',
|
path: '/session',
|
||||||
body: { password },
|
body: { password, remember },
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -94,17 +115,20 @@ class API {
|
||||||
...client,
|
...client,
|
||||||
createdAt: new Date(client.createdAt),
|
createdAt: new Date(client.createdAt),
|
||||||
updatedAt: new Date(client.updatedAt),
|
updatedAt: new Date(client.updatedAt),
|
||||||
|
expiredAt: client.expiredAt !== null
|
||||||
|
? new Date(client.expiredAt)
|
||||||
|
: null,
|
||||||
latestHandshakeAt: client.latestHandshakeAt !== null
|
latestHandshakeAt: client.latestHandshakeAt !== null
|
||||||
? new Date(client.latestHandshakeAt)
|
? new Date(client.latestHandshakeAt)
|
||||||
: null,
|
: null,
|
||||||
})));
|
})));
|
||||||
}
|
}
|
||||||
|
|
||||||
async createClient({ name }) {
|
async createClient({ name, expiredDate }) {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'post',
|
method: 'post',
|
||||||
path: '/wireguard/client',
|
path: '/wireguard/client',
|
||||||
body: { name },
|
body: { name, expiredDate },
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -115,6 +139,13 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async showOneTimeLink({ clientId }) {
|
||||||
|
return this.call({
|
||||||
|
method: 'post',
|
||||||
|
path: `/wireguard/client/${clientId}/generateOneTimeLink`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async enableClient({ clientId }) {
|
async enableClient({ clientId }) {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'post',
|
method: 'post',
|
||||||
|
@ -145,6 +176,14 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async updateClientExpireDate({ clientId, expireDate }) {
|
||||||
|
return this.call({
|
||||||
|
method: 'put',
|
||||||
|
path: `/wireguard/client/${clientId}/expireDate/`,
|
||||||
|
body: { expireDate },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async restoreConfiguration(file) {
|
async restoreConfiguration(file) {
|
||||||
return this.call({
|
return this.call({
|
||||||
method: 'put',
|
method: 'put',
|
||||||
|
@ -153,4 +192,11 @@ class API {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getUiSortClients() {
|
||||||
|
return this.call({
|
||||||
|
method: 'get',
|
||||||
|
path: '/ui-sort-clients',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,22 @@ function bytes(bytes, decimals, kib, maxunit) {
|
||||||
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`;
|
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sorts an array of objects by a specified property in ascending or descending order.
|
||||||
|
*
|
||||||
|
* @param {Array} array - The array of objects to be sorted.
|
||||||
|
* @param {string} property - The property to sort the array by.
|
||||||
|
* @param {boolean} [sort=true] - Whether to sort the array in ascending (default) or descending order.
|
||||||
|
* @return {Array} - The sorted array of objects.
|
||||||
|
*/
|
||||||
|
function sortByProperty(array, property, sort = true) {
|
||||||
|
if (sort) {
|
||||||
|
return array.sort((a, b) => (typeof a[property] === 'string' ? a[property].localeCompare(b[property]) : a[property] - b[property]));
|
||||||
|
}
|
||||||
|
|
||||||
|
return array.sort((a, b) => (typeof a[property] === 'string' ? b[property].localeCompare(a[property]) : b[property] - a[property]));
|
||||||
|
}
|
||||||
|
|
||||||
const i18n = new VueI18n({
|
const i18n = new VueI18n({
|
||||||
locale: localStorage.getItem('lang') || 'en',
|
locale: localStorage.getItem('lang') || 'en',
|
||||||
fallbackLocale: 'en',
|
fallbackLocale: 'en',
|
||||||
|
@ -53,16 +69,21 @@ new Vue({
|
||||||
authenticating: false,
|
authenticating: false,
|
||||||
password: null,
|
password: null,
|
||||||
requiresPassword: null,
|
requiresPassword: null,
|
||||||
|
remember: false,
|
||||||
|
rememberMeEnabled: false,
|
||||||
|
|
||||||
clients: null,
|
clients: null,
|
||||||
clientsPersist: {},
|
clientsPersist: {},
|
||||||
clientDelete: null,
|
clientDelete: null,
|
||||||
clientCreate: null,
|
clientCreate: null,
|
||||||
clientCreateName: '',
|
clientCreateName: '',
|
||||||
|
clientExpiredDate: '',
|
||||||
clientEditName: null,
|
clientEditName: null,
|
||||||
clientEditNameId: null,
|
clientEditNameId: null,
|
||||||
clientEditAddress: null,
|
clientEditAddress: null,
|
||||||
clientEditAddressId: null,
|
clientEditAddressId: null,
|
||||||
|
clientEditExpireDate: null,
|
||||||
|
clientEditExpireDateId: null,
|
||||||
qrcode: null,
|
qrcode: null,
|
||||||
|
|
||||||
currentRelease: null,
|
currentRelease: null,
|
||||||
|
@ -75,6 +96,11 @@ new Vue({
|
||||||
'dicebear': null,
|
'dicebear': null,
|
||||||
'gravatar': false,
|
'gravatar': false,
|
||||||
},
|
},
|
||||||
|
enableOneTimeLinks: false,
|
||||||
|
enableSortClient: false,
|
||||||
|
sortClient: true, // Sort clients by name, true = asc, false = desc
|
||||||
|
enableExpireTime: false,
|
||||||
|
|
||||||
uiShowCharts: localStorage.getItem('uiShowCharts') === '1',
|
uiShowCharts: localStorage.getItem('uiShowCharts') === '1',
|
||||||
uiTheme: localStorage.theme || 'auto',
|
uiTheme: localStorage.theme || 'auto',
|
||||||
prefersDarkScheme: window.matchMedia('(prefers-color-scheme: dark)'),
|
prefersDarkScheme: window.matchMedia('(prefers-color-scheme: dark)'),
|
||||||
|
@ -159,6 +185,7 @@ new Vue({
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
dateTime: (value) => {
|
dateTime: (value) => {
|
||||||
|
@ -235,6 +262,10 @@ new Vue({
|
||||||
|
|
||||||
return client;
|
return client;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (this.enableSortClient) {
|
||||||
|
this.clients = sortByProperty(this.clients, 'name', this.sortClient);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
login(e) {
|
login(e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
@ -245,6 +276,7 @@ new Vue({
|
||||||
this.authenticating = true;
|
this.authenticating = true;
|
||||||
this.api.createSession({
|
this.api.createSession({
|
||||||
password: this.password,
|
password: this.password,
|
||||||
|
remember: this.remember,
|
||||||
})
|
})
|
||||||
.then(async () => {
|
.then(async () => {
|
||||||
const session = await this.api.getSession();
|
const session = await this.api.getSession();
|
||||||
|
@ -274,9 +306,10 @@ new Vue({
|
||||||
},
|
},
|
||||||
createClient() {
|
createClient() {
|
||||||
const name = this.clientCreateName;
|
const name = this.clientCreateName;
|
||||||
|
const expiredDate = this.clientExpiredDate;
|
||||||
if (!name) return;
|
if (!name) return;
|
||||||
|
|
||||||
this.api.createClient({ name })
|
this.api.createClient({ name, expiredDate })
|
||||||
.catch((err) => alert(err.message || err.toString()))
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
.finally(() => this.refresh().catch(console.error));
|
.finally(() => this.refresh().catch(console.error));
|
||||||
},
|
},
|
||||||
|
@ -285,6 +318,11 @@ new Vue({
|
||||||
.catch((err) => alert(err.message || err.toString()))
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
.finally(() => this.refresh().catch(console.error));
|
.finally(() => this.refresh().catch(console.error));
|
||||||
},
|
},
|
||||||
|
showOneTimeLink(client) {
|
||||||
|
this.api.showOneTimeLink({ clientId: client.id })
|
||||||
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
|
.finally(() => this.refresh().catch(console.error));
|
||||||
|
},
|
||||||
enableClient(client) {
|
enableClient(client) {
|
||||||
this.api.enableClient({ clientId: client.id })
|
this.api.enableClient({ clientId: client.id })
|
||||||
.catch((err) => alert(err.message || err.toString()))
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
|
@ -305,6 +343,11 @@ new Vue({
|
||||||
.catch((err) => alert(err.message || err.toString()))
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
.finally(() => this.refresh().catch(console.error));
|
.finally(() => this.refresh().catch(console.error));
|
||||||
},
|
},
|
||||||
|
updateClientExpireDate(client, expireDate) {
|
||||||
|
this.api.updateClientExpireDate({ clientId: client.id, expireDate })
|
||||||
|
.catch((err) => alert(err.message || err.toString()))
|
||||||
|
.finally(() => this.refresh().catch(console.error));
|
||||||
|
},
|
||||||
restoreConfig(e) {
|
restoreConfig(e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
const file = e.currentTarget.files.item(0);
|
const file = e.currentTarget.files.item(0);
|
||||||
|
@ -348,6 +391,15 @@ new Vue({
|
||||||
timeago: (value) => {
|
timeago: (value) => {
|
||||||
return timeago.format(value, i18n.locale);
|
return timeago.format(value, i18n.locale);
|
||||||
},
|
},
|
||||||
|
expiredDateFormat: (value) => {
|
||||||
|
if (value === null) return i18n.t('Permanent');
|
||||||
|
const dateTime = new Date(value);
|
||||||
|
const options = { year: 'numeric', month: 'long', day: 'numeric' };
|
||||||
|
return dateTime.toLocaleDateString(i18n.locale, options);
|
||||||
|
},
|
||||||
|
expiredDateEditFormat: (value) => {
|
||||||
|
if (value === null) return 'yyyy-MM-dd';
|
||||||
|
},
|
||||||
},
|
},
|
||||||
mounted() {
|
mounted() {
|
||||||
this.prefersDarkScheme.addListener(this.handlePrefersChange);
|
this.prefersDarkScheme.addListener(this.handlePrefersChange);
|
||||||
|
@ -368,6 +420,11 @@ new Vue({
|
||||||
alert(err.message || err.toString());
|
alert(err.message || err.toString());
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.api.getRememberMeEnabled()
|
||||||
|
.then((rememberMeEnabled) => {
|
||||||
|
this.rememberMeEnabled = rememberMeEnabled;
|
||||||
|
});
|
||||||
|
|
||||||
setInterval(() => {
|
setInterval(() => {
|
||||||
this.refresh({
|
this.refresh({
|
||||||
updateCharts: this.updateCharts,
|
updateCharts: this.updateCharts,
|
||||||
|
@ -390,6 +447,30 @@ new Vue({
|
||||||
this.uiChartType = 0;
|
this.uiChartType = 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.api.getWGEnableOneTimeLinks()
|
||||||
|
.then((res) => {
|
||||||
|
this.enableOneTimeLinks = res;
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
this.enableOneTimeLinks = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.api.getUiSortClients()
|
||||||
|
.then((res) => {
|
||||||
|
this.enableSortClient = res;
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
this.enableSortClient = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.api.getWGEnableExpireTime()
|
||||||
|
.then((res) => {
|
||||||
|
this.enableExpireTime = res;
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
this.enableExpireTime = false;
|
||||||
|
});
|
||||||
|
|
||||||
this.api.getAvatarSettings()
|
this.api.getAvatarSettings()
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
this.avatarSettings = res;
|
this.avatarSettings = res;
|
||||||
|
|
|
@ -34,6 +34,12 @@ const messages = { // eslint-disable-line no-unused-vars
|
||||||
backup: 'Backup',
|
backup: 'Backup',
|
||||||
titleRestoreConfig: 'Restore your configuration',
|
titleRestoreConfig: 'Restore your configuration',
|
||||||
titleBackupConfig: 'Backup your configuration',
|
titleBackupConfig: 'Backup your configuration',
|
||||||
|
rememberMe: 'Remember me',
|
||||||
|
titleRememberMe: 'Stay logged after closing the browser',
|
||||||
|
sort: 'Sort',
|
||||||
|
ExpireDate: 'Expire Date',
|
||||||
|
Permanent: 'Permanent',
|
||||||
|
OneTimeLink: 'Generate short one time link',
|
||||||
},
|
},
|
||||||
ua: {
|
ua: {
|
||||||
name: 'Ім`я',
|
name: 'Ім`я',
|
||||||
|
@ -102,6 +108,12 @@ const messages = { // eslint-disable-line no-unused-vars
|
||||||
backup: 'Резервная копия',
|
backup: 'Резервная копия',
|
||||||
titleRestoreConfig: 'Восстановить конфигурацию',
|
titleRestoreConfig: 'Восстановить конфигурацию',
|
||||||
titleBackupConfig: 'Создать резервную копию конфигурации',
|
titleBackupConfig: 'Создать резервную копию конфигурации',
|
||||||
|
rememberMe: 'Запомнить меня',
|
||||||
|
titleRememberMe: 'Оставаться в системе после закрытия браузера',
|
||||||
|
sort: 'Сортировка',
|
||||||
|
ExpireDate: 'Дата истечения срока',
|
||||||
|
Permanent: 'Бессрочно',
|
||||||
|
OneTimeLink: 'Создать короткую одноразовую ссылку',
|
||||||
},
|
},
|
||||||
tr: { // Müslüm Barış Korkmazer @babico
|
tr: { // Müslüm Barış Korkmazer @babico
|
||||||
name: 'İsim',
|
name: 'İsim',
|
||||||
|
@ -340,8 +352,14 @@ const messages = { // eslint-disable-line no-unused-vars
|
||||||
downloadConfig: '구성 다운로드',
|
downloadConfig: '구성 다운로드',
|
||||||
madeBy: '만든 사람',
|
madeBy: '만든 사람',
|
||||||
donate: '기부',
|
donate: '기부',
|
||||||
|
toggleCharts: '차트 표시/숨기기',
|
||||||
|
theme: { dark: '어두운 테마', light: '밝은 테마', auto: '자동 테마' },
|
||||||
|
restore: '복원',
|
||||||
|
backup: '백업',
|
||||||
|
titleRestoreConfig: '구성 파일 복원',
|
||||||
|
titleBackupConfig: '구성 파일 백업',
|
||||||
},
|
},
|
||||||
vi: {
|
vi: { // https://github.com/hoangneeee
|
||||||
name: 'Tên',
|
name: 'Tên',
|
||||||
password: 'Mật khẩu',
|
password: 'Mật khẩu',
|
||||||
signIn: 'Đăng nhập',
|
signIn: 'Đăng nhập',
|
||||||
|
@ -367,6 +385,13 @@ const messages = { // eslint-disable-line no-unused-vars
|
||||||
downloadConfig: 'Tải xuống cấu hình',
|
downloadConfig: 'Tải xuống cấu hình',
|
||||||
madeBy: 'Được tạo bởi',
|
madeBy: 'Được tạo bởi',
|
||||||
donate: 'Ủng hộ',
|
donate: 'Ủng hộ',
|
||||||
|
toggleCharts: 'Mở/Ẩn Biểu đồ',
|
||||||
|
theme: { dark: 'Dark theme', light: 'Light theme', auto: 'Auto theme' },
|
||||||
|
restore: 'Khôi phục',
|
||||||
|
backup: 'Sao lưu',
|
||||||
|
titleRestoreConfig: 'Khôi phục cấu hình của bạn',
|
||||||
|
titleBackupConfig: 'Sao lưu cấu hình của bạn',
|
||||||
|
sort: 'Sắp xếp',
|
||||||
},
|
},
|
||||||
nl: {
|
nl: {
|
||||||
name: 'Naam',
|
name: 'Naam',
|
||||||
|
|
|
@ -1,3 +1,7 @@
|
||||||
@tailwind base;
|
@tailwind base;
|
||||||
@tailwind components;
|
@tailwind components;
|
||||||
@tailwind utilities;
|
@tailwind utilities;
|
||||||
|
|
||||||
|
.p-0 {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue