From 198b240755216a814dd1be15b756497591ceda7a Mon Sep 17 00:00:00 2001 From: Bernd Storath <32197462+kaaax0815@users.noreply.github.com> Date: Fri, 14 Mar 2025 10:33:02 +0100 Subject: [PATCH] Feat: Suggest IP or Hostname (#1739) * get ip and hostnames * use heroicons * add host field * get private info * unstyled prototype * styled select * add to setup * fix types --- .vscode/settings.json | 3 + docker-compose.dev.yml | 2 +- package.json | 2 +- src/app/components/Admin/SuggestDialog.vue | 40 +++++ src/app/components/Base/Select.vue | 37 +++++ src/app/components/Form/HostField.vue | 48 ++++++ src/app/components/Icons/ArrowDown.vue | 16 +- src/app/components/Icons/ArrowInf.vue | 19 +-- src/app/components/Icons/ArrowLeftCircle.vue | 18 +-- src/app/components/Icons/ArrowRightCircle.vue | 18 +-- src/app/components/Icons/ArrowUp.vue | 16 +- src/app/components/Icons/Chart.vue | 15 +- src/app/components/Icons/CheckCircle.vue | 18 +-- src/app/components/Icons/Close.vue | 18 +-- src/app/components/Icons/Delete.vue | 16 +- src/app/components/Icons/Download.vue | 18 +-- src/app/components/Icons/Edit.vue | 18 +-- src/app/components/Icons/Info.vue | 18 +-- src/app/components/Icons/Language.vue | 18 +-- src/app/components/Icons/Link.vue | 18 +-- src/app/components/Icons/Logout.vue | 18 +-- src/app/components/Icons/Moon.vue | 18 +-- src/app/components/Icons/Plus.vue | 19 +-- src/app/components/Icons/QRCode.vue | 18 +-- src/app/components/Icons/Sparkles.vue | 7 + src/app/components/Icons/Stack.vue | 19 +-- src/app/components/Icons/Sun.vue | 18 +-- src/app/components/Icons/Warning.vue | 20 +-- src/app/pages/admin/config.vue | 3 +- src/app/pages/setup/4.vue | 3 +- src/i18n/locales/en.json | 7 +- src/package.json | 3 +- src/pnpm-lock.yaml | 13 +- src/server/api/admin/ip-info.get.ts | 4 + src/server/api/setup/4.get.ts | 4 + src/server/utils/cache.ts | 29 ++++ src/server/utils/ip.ts | 141 +++++++++++++++++- src/server/utils/release.ts | 27 +--- src/server/utils/session.ts | 5 + 39 files changed, 450 insertions(+), 302 deletions(-) create mode 100644 src/app/components/Admin/SuggestDialog.vue create mode 100644 src/app/components/Base/Select.vue create mode 100644 src/app/components/Form/HostField.vue create mode 100644 src/app/components/Icons/Sparkles.vue create mode 100644 src/server/api/admin/ip-info.get.ts create mode 100644 src/server/api/setup/4.get.ts create mode 100644 src/server/utils/cache.ts diff --git a/.vscode/settings.json b/.vscode/settings.json index 25e14c36..a1205250 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -6,6 +6,9 @@ "nuxtr.vueFiles.style.addStyleTag": false, "nuxtr.piniaFiles.defaultTemplate": "setup", "nuxtr.monorepoMode.DirectoryName": "src", + "editor.codeActionsOnSave": { + "source.fixAll.eslint": "always" + }, "[vue]": { "editor.defaultFormatter": "esbenp.prettier-vscode" }, diff --git a/docker-compose.dev.yml b/docker-compose.dev.yml index 8f9dad6e..9b191b34 100644 --- a/docker-compose.dev.yml +++ b/docker-compose.dev.yml @@ -16,7 +16,7 @@ services: - NET_ADMIN - SYS_MODULE environment: - - INIT_ENABLED=true + - INIT_ENABLED=false - INIT_HOST=test - INIT_PORT=51820 - INIT_USERNAME=testtest diff --git a/package.json b/package.json index 7c76ab77..d85a09df 100644 --- a/package.json +++ b/package.json @@ -7,5 +7,5 @@ "docs:preview": "docker run --rm -it -p 8080:8080 -v ./docs:/docs squidfunk/mkdocs-material serve -a 0.0.0.0:8080", "scripts:version": "bash scripts/version.sh" }, - "packageManager": "pnpm@10.6.2" + "packageManager": "pnpm@10.6.3" } diff --git a/src/app/components/Admin/SuggestDialog.vue b/src/app/components/Admin/SuggestDialog.vue new file mode 100644 index 00000000..6b18612c --- /dev/null +++ b/src/app/components/Admin/SuggestDialog.vue @@ -0,0 +1,40 @@ + + + diff --git a/src/app/components/Base/Select.vue b/src/app/components/Base/Select.vue new file mode 100644 index 00000000..d69fa730 --- /dev/null +++ b/src/app/components/Base/Select.vue @@ -0,0 +1,37 @@ + + + diff --git a/src/app/components/Form/HostField.vue b/src/app/components/Form/HostField.vue new file mode 100644 index 00000000..467da466 --- /dev/null +++ b/src/app/components/Form/HostField.vue @@ -0,0 +1,48 @@ + + + diff --git a/src/app/components/Icons/ArrowDown.vue b/src/app/components/Icons/ArrowDown.vue index a44f7c39..8c3cfa9d 100644 --- a/src/app/components/Icons/ArrowDown.vue +++ b/src/app/components/Icons/ArrowDown.vue @@ -1,13 +1,7 @@ + + diff --git a/src/app/components/Icons/ArrowInf.vue b/src/app/components/Icons/ArrowInf.vue index 0a204f4a..4369d5c5 100644 --- a/src/app/components/Icons/ArrowInf.vue +++ b/src/app/components/Icons/ArrowInf.vue @@ -1,16 +1,7 @@ + + diff --git a/src/app/components/Icons/ArrowLeftCircle.vue b/src/app/components/Icons/ArrowLeftCircle.vue index fd6d9179..f2b53895 100644 --- a/src/app/components/Icons/ArrowLeftCircle.vue +++ b/src/app/components/Icons/ArrowLeftCircle.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/ArrowRightCircle.vue b/src/app/components/Icons/ArrowRightCircle.vue index f43a2afc..bee6aa60 100644 --- a/src/app/components/Icons/ArrowRightCircle.vue +++ b/src/app/components/Icons/ArrowRightCircle.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/ArrowUp.vue b/src/app/components/Icons/ArrowUp.vue index 37a6ad52..24e52124 100644 --- a/src/app/components/Icons/ArrowUp.vue +++ b/src/app/components/Icons/ArrowUp.vue @@ -1,13 +1,7 @@ + + diff --git a/src/app/components/Icons/Chart.vue b/src/app/components/Icons/Chart.vue index 2c9d74e8..511cadc7 100644 --- a/src/app/components/Icons/Chart.vue +++ b/src/app/components/Icons/Chart.vue @@ -1,12 +1,7 @@ + + diff --git a/src/app/components/Icons/CheckCircle.vue b/src/app/components/Icons/CheckCircle.vue index ebdc8a1a..45d1e5bf 100644 --- a/src/app/components/Icons/CheckCircle.vue +++ b/src/app/components/Icons/CheckCircle.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Close.vue b/src/app/components/Icons/Close.vue index 654ca4d4..32491ca8 100644 --- a/src/app/components/Icons/Close.vue +++ b/src/app/components/Icons/Close.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Delete.vue b/src/app/components/Icons/Delete.vue index 43ad5ff2..c29b9970 100644 --- a/src/app/components/Icons/Delete.vue +++ b/src/app/components/Icons/Delete.vue @@ -1,13 +1,7 @@ + + diff --git a/src/app/components/Icons/Download.vue b/src/app/components/Icons/Download.vue index 8fdb7643..a1d7a40e 100644 --- a/src/app/components/Icons/Download.vue +++ b/src/app/components/Icons/Download.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Edit.vue b/src/app/components/Icons/Edit.vue index e13e8453..2a0a89f4 100644 --- a/src/app/components/Icons/Edit.vue +++ b/src/app/components/Icons/Edit.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Info.vue b/src/app/components/Icons/Info.vue index cedf7d45..0902ea40 100644 --- a/src/app/components/Icons/Info.vue +++ b/src/app/components/Icons/Info.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Language.vue b/src/app/components/Icons/Language.vue index 9ddd7b3c..d29a87e4 100644 --- a/src/app/components/Icons/Language.vue +++ b/src/app/components/Icons/Language.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Link.vue b/src/app/components/Icons/Link.vue index d8395c28..2ba02eec 100644 --- a/src/app/components/Icons/Link.vue +++ b/src/app/components/Icons/Link.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Logout.vue b/src/app/components/Icons/Logout.vue index 21656ea4..3a44a3c3 100644 --- a/src/app/components/Icons/Logout.vue +++ b/src/app/components/Icons/Logout.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Moon.vue b/src/app/components/Icons/Moon.vue index 4b271121..ad36b258 100644 --- a/src/app/components/Icons/Moon.vue +++ b/src/app/components/Icons/Moon.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Plus.vue b/src/app/components/Icons/Plus.vue index e3e95b7a..b1cf1d3d 100644 --- a/src/app/components/Icons/Plus.vue +++ b/src/app/components/Icons/Plus.vue @@ -1,16 +1,7 @@ + + diff --git a/src/app/components/Icons/QRCode.vue b/src/app/components/Icons/QRCode.vue index bca1dd33..c749ba80 100644 --- a/src/app/components/Icons/QRCode.vue +++ b/src/app/components/Icons/QRCode.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Sparkles.vue b/src/app/components/Icons/Sparkles.vue new file mode 100644 index 00000000..72777b5f --- /dev/null +++ b/src/app/components/Icons/Sparkles.vue @@ -0,0 +1,7 @@ + + + diff --git a/src/app/components/Icons/Stack.vue b/src/app/components/Icons/Stack.vue index d26b002b..65407eab 100644 --- a/src/app/components/Icons/Stack.vue +++ b/src/app/components/Icons/Stack.vue @@ -1,16 +1,7 @@ + + diff --git a/src/app/components/Icons/Sun.vue b/src/app/components/Icons/Sun.vue index eccf9d16..ea223f50 100644 --- a/src/app/components/Icons/Sun.vue +++ b/src/app/components/Icons/Sun.vue @@ -1,15 +1,7 @@ + + diff --git a/src/app/components/Icons/Warning.vue b/src/app/components/Icons/Warning.vue index 07a27999..1dd19ae0 100644 --- a/src/app/components/Icons/Warning.vue +++ b/src/app/components/Icons/Warning.vue @@ -1,17 +1,7 @@ + + diff --git a/src/app/pages/admin/config.vue b/src/app/pages/admin/config.vue index 80d868bb..05be9649 100644 --- a/src/app/pages/admin/config.vue +++ b/src/app/pages/admin/config.vue @@ -3,11 +3,12 @@ {{ $t('admin.config.connection') }} -
-
diff --git a/src/i18n/locales/en.json b/src/i18n/locales/en.json index a4203365..71b10925 100644 --- a/src/i18n/locales/en.json +++ b/src/i18n/locales/en.json @@ -32,7 +32,8 @@ "port": "Port", "yes": "Yes", "no": "No", - "confirmPassword": "Confirm Password" + "confirmPassword": "Confirm Password", + "loading": "Loading" }, "setup": { "welcome": "Welcome to your first setup of wg-easy", @@ -147,7 +148,9 @@ "allowedIpsDesc": "Allowed IPs clients will use (global config)", "dnsDesc": "DNS server clients will use (global config)", "mtuDesc": "MTU clients will use (only for new clients)", - "persistentKeepaliveDesc": "Interval in seconds to send keepalives to the server. 0 = disabled (only for new clients)" + "persistentKeepaliveDesc": "Interval in seconds to send keepalives to the server. 0 = disabled (only for new clients)", + "suggest": "Suggest", + "suggestDesc": "Choose a IP-Address or Hostname for the Host field" }, "interface": { "cidrSuccess": "Changed CIDR", diff --git a/src/package.json b/src/package.json index 5be4f552..2adb40e6 100644 --- a/src/package.json +++ b/src/package.json @@ -19,6 +19,7 @@ }, "dependencies": { "@eschricht/nuxt-color-mode": "^1.1.5", + "@heroicons/vue": "^2.2.0", "@libsql/client": "^0.14.0", "@nuxtjs/i18n": "^9.3.1", "@nuxtjs/tailwindcss": "^6.13.2", @@ -60,5 +61,5 @@ "typescript": "^5.8.2", "vue-tsc": "^2.2.8" }, - "packageManager": "pnpm@10.6.2" + "packageManager": "pnpm@10.6.3" } diff --git a/src/pnpm-lock.yaml b/src/pnpm-lock.yaml index 15f73d72..37a6bf72 100644 --- a/src/pnpm-lock.yaml +++ b/src/pnpm-lock.yaml @@ -11,6 +11,9 @@ importers: '@eschricht/nuxt-color-mode': specifier: ^1.1.5 version: 1.1.5(magicast@0.3.5) + '@heroicons/vue': + specifier: ^2.2.0 + version: 2.2.0(vue@3.5.13(typescript@5.8.2)) '@libsql/client': specifier: ^0.14.0 version: 0.14.0 @@ -791,6 +794,11 @@ packages: '@floating-ui/vue@1.1.6': resolution: {integrity: sha512-XFlUzGHGv12zbgHNk5FN2mUB7ROul3oG2ENdTpWdE+qMFxyNxWSRmsoyhiEnpmabNm6WnUvR1OvJfUfN4ojC1A==} + '@heroicons/vue@2.2.0': + resolution: {integrity: sha512-G3dbSxoeEKqbi/DFalhRxJU4mTXJn7GwZ7ae8NuEQzd1bqdd0jAbdaBZlHPcvPD2xI1iGzNVB4k20Un2AguYPw==} + peerDependencies: + vue: '>= 3' + '@humanfs/core@0.19.1': resolution: {integrity: sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==} engines: {node: '>=18.18.0'} @@ -3403,7 +3411,6 @@ packages: libsql@0.4.7: resolution: {integrity: sha512-T9eIRCs6b0J1SHKYIvD8+KCJMcWZ900iZyxdnSCdqxN12Z1ijzT+jY5nrk72Jw4B0HGzms2NgpryArlJqvc3Lw==} - cpu: [x64, arm64, wasm32] os: [darwin, linux, win32] lilconfig@3.1.3: @@ -5761,6 +5768,10 @@ snapshots: - '@vue/composition-api' - vue + '@heroicons/vue@2.2.0(vue@3.5.13(typescript@5.8.2))': + dependencies: + vue: 3.5.13(typescript@5.8.2) + '@humanfs/core@0.19.1': {} '@humanfs/node@0.16.6': diff --git a/src/server/api/admin/ip-info.get.ts b/src/server/api/admin/ip-info.get.ts new file mode 100644 index 00000000..1ea6d745 --- /dev/null +++ b/src/server/api/admin/ip-info.get.ts @@ -0,0 +1,4 @@ +export default definePermissionEventHandler('admin', 'any', async () => { + const result = await cachedGetIpInformation(); + return result; +}); diff --git a/src/server/api/setup/4.get.ts b/src/server/api/setup/4.get.ts new file mode 100644 index 00000000..4c2a5186 --- /dev/null +++ b/src/server/api/setup/4.get.ts @@ -0,0 +1,4 @@ +export default defineSetupEventHandler(4, async () => { + const result = await cachedGetIpInformation(); + return result; +}); diff --git a/src/server/utils/cache.ts b/src/server/utils/cache.ts new file mode 100644 index 00000000..96ea8c2d --- /dev/null +++ b/src/server/utils/cache.ts @@ -0,0 +1,29 @@ +type Opts = { + /** + * Expiry time in milliseconds + */ + expiry: number; +}; + +/** + * Cache function for 1 hour + */ +export function cacheFunction(fn: () => T, { expiry }: Opts): () => T { + let cache: { value: T; expiry: number } | null = null; + + return (): T => { + const now = Date.now(); + + if (cache && cache.expiry > now) { + return cache.value; + } + + const result = fn(); + cache = { + value: result, + expiry: now + expiry, + }; + + return result; + }; +} diff --git a/src/server/utils/ip.ts b/src/server/utils/ip.ts index c5661337..c3c097e1 100644 --- a/src/server/utils/ip.ts +++ b/src/server/utils/ip.ts @@ -1,5 +1,7 @@ -import type { parseCidr } from 'cidr-tools'; +import { Resolver } from 'node:dns/promises'; +import { networkInterfaces } from 'node:os'; import { stringifyIp } from 'ip-bigint'; +import type { parseCidr } from 'cidr-tools'; import type { ClientNextIpType } from '#db/repositories/client/types'; @@ -31,3 +33,140 @@ export function nextIP( return address; } + +// use opendns to get public ip +const dnsServers = { + ip4: ['208.67.222.222'], + ip6: ['2620:119:35::35'], + ip: 'myip.opendns.com', +}; + +async function getPublicInformation() { + const ipv4 = await getPublicIpv4(); + const ipv6 = await getPublicIpv6(); + + const ptr4 = ipv4 ? await getReverseDns(ipv4) : []; + const ptr6 = ipv6 ? await getReverseDns(ipv6) : []; + const hostnames = [...new Set([...ptr4, ...ptr6])]; + + return { ipv4, ipv6, hostnames }; +} + +async function getPublicIpv4() { + try { + const resolver = new Resolver(); + resolver.setServers(dnsServers.ip4); + const ipv4 = await resolver.resolve4(dnsServers.ip); + return ipv4[0]; + } catch { + return null; + } +} + +async function getPublicIpv6() { + try { + const resolver = new Resolver(); + resolver.setServers(dnsServers.ip6); + const ipv6 = await resolver.resolve6(dnsServers.ip); + return ipv6[0]; + } catch { + return null; + } +} + +async function getReverseDns(ip: string) { + try { + const resolver = new Resolver(); + resolver.setServers([...dnsServers.ip4, ...dnsServers.ip6]); + const ptr = await resolver.reverse(ip); + return ptr; + } catch { + return []; + } +} + +function getPrivateInformation() { + const interfaces = networkInterfaces(); + + const interfaceNames = Object.keys(interfaces); + + const obj: Record = {}; + + for (const name of interfaceNames) { + if (name === 'wg0') { + continue; + } + + const iface = interfaces[name]; + if (!iface) continue; + + for (const { family, internal, address } of iface) { + if (internal) { + continue; + } + if (!obj[name]) { + obj[name] = { + ipv4: [], + ipv6: [], + }; + } + if (family === 'IPv4') { + obj[name].ipv4.push(address); + } else if (family === 'IPv6') { + obj[name].ipv6.push(address); + } + } + } + + return obj; +} + +async function getIpInformation() { + const results = []; + + const publicInfo = await getPublicInformation(); + if (publicInfo.ipv4) { + results.push({ + value: publicInfo.ipv4, + label: 'IPv4 - Public', + }); + } + if (publicInfo.ipv6) { + results.push({ + value: `[${publicInfo.ipv6}]`, + label: 'IPv6 - Public', + }); + } + for (const hostname of publicInfo.hostnames) { + results.push({ + value: hostname, + label: 'Hostname - Public', + }); + } + + const privateInfo = getPrivateInformation(); + for (const [name, { ipv4, ipv6 }] of Object.entries(privateInfo)) { + for (const ip of ipv4) { + results.push({ + value: ip, + label: `IPv4 - ${name}`, + }); + } + for (const ip of ipv6) { + results.push({ + value: `[${ip}]`, + label: `IPv6 - ${name}`, + }); + } + } + + return results; +} + +/** + * Fetch IP Information + * @cache Response is cached for 15 min + */ +export const cachedGetIpInformation = cacheFunction(getIpInformation, { + expiry: 15 * 60 * 1000, +}); diff --git a/src/server/utils/release.ts b/src/server/utils/release.ts index 29fe762c..f5dfdc46 100644 --- a/src/server/utils/release.ts +++ b/src/server/utils/release.ts @@ -3,29 +3,6 @@ type GithubRelease = { body: string; }; -/** - * Cache function for 1 hour - */ -function cacheFunction(fn: () => T): () => T { - let cache: { value: T; expiry: number } | null = null; - - return (): T => { - const now = Date.now(); - - if (cache && cache.expiry > now) { - return cache.value; - } - - const result = fn(); - cache = { - value: result, - expiry: now + 3600000, - }; - - return result; - }; -} - async function fetchLatestRelease() { try { const response = await $fetch( @@ -53,4 +30,6 @@ async function fetchLatestRelease() { * Fetch latest release from GitHub * @cache Response is cached for 1 hour */ -export const cachedFetchLatestRelease = cacheFunction(fetchLatestRelease); +export const cachedFetchLatestRelease = cacheFunction(fetchLatestRelease, { + expiry: 60 * 60 * 1000, +}); diff --git a/src/server/utils/session.ts b/src/server/utils/session.ts index 9602ded5..1a144cea 100644 --- a/src/server/utils/session.ts +++ b/src/server/utils/session.ts @@ -91,6 +91,11 @@ export async function getCurrentUser(event: H3Event) { }); } user = foundUser; + } else { + throw createError({ + statusCode: 401, + statusMessage: 'Session failed. No Authorization', + }); } if (!user) {