mirror of
https://github.com/h44z/wg-portal.git
synced 2025-10-04 07:26:18 +00:00
250 lines
11 KiB
Vue
250 lines
11 KiB
Vue
<script setup>
|
|
import Modal from "./Modal.vue";
|
|
import { peerStore } from "@/stores/peers";
|
|
import { interfaceStore } from "@/stores/interfaces";
|
|
import { computed, ref, watch } from "vue";
|
|
import { useI18n } from "vue-i18n";
|
|
import { freshInterface, freshPeer, freshStats } from '@/helpers/models';
|
|
import Prism from "vue-prism-component";
|
|
import { notify } from "@kyvg/vue3-notification";
|
|
import { settingsStore } from "@/stores/settings";
|
|
import { profileStore } from "@/stores/profile";
|
|
import { base64_url_encode } from '@/helpers/encoding';
|
|
import { apiWrapper } from "@/helpers/fetch-wrapper";
|
|
|
|
const { t } = useI18n()
|
|
|
|
const settings = settingsStore()
|
|
const peers = peerStore()
|
|
const interfaces = interfaceStore()
|
|
const profile = profileStore()
|
|
|
|
const props = defineProps({
|
|
peerId: String,
|
|
visible: Boolean,
|
|
})
|
|
|
|
const emit = defineEmits(['close'])
|
|
|
|
function close() {
|
|
emit('close')
|
|
}
|
|
|
|
const configString = ref("")
|
|
|
|
const selectedPeer = computed(() => {
|
|
let p = peers.Find(props.peerId)
|
|
|
|
if (!p) {
|
|
if (!!props.peerId || props.peerId.length) {
|
|
p = profile.peers.find((p) => p.Identifier === props.peerId)
|
|
} else {
|
|
p = freshPeer() // dummy peer to avoid 'undefined' exceptions
|
|
}
|
|
}
|
|
return p
|
|
})
|
|
|
|
const selectedStats = computed(() => {
|
|
let s = peers.Statistics(props.peerId)
|
|
|
|
if (!s) {
|
|
if (!!props.peerId || props.peerId.length) {
|
|
s = profile.Statistics(props.peerId)
|
|
} else {
|
|
s = freshStats() // dummy stats to avoid 'undefined' exceptions
|
|
}
|
|
|
|
}
|
|
return s
|
|
})
|
|
|
|
const selectedInterface = computed(() => {
|
|
let i = interfaces.GetSelected;
|
|
|
|
if (!i) {
|
|
i = freshInterface() // dummy interface to avoid 'undefined' exceptions
|
|
}
|
|
return i
|
|
})
|
|
|
|
const title = computed(() => {
|
|
if (!props.visible) {
|
|
return "" // otherwise interfaces.GetSelected will die...
|
|
}
|
|
if (selectedInterface.value.Mode === "server") {
|
|
return t("modals.peer-view.headline-peer") + " " + selectedPeer.value.DisplayName
|
|
} else {
|
|
return t("modals.peer-view.headline-endpoint") + " " + selectedPeer.value.DisplayName
|
|
}
|
|
})
|
|
|
|
const configStyle = ref("wgquick")
|
|
|
|
watch(() => props.visible, async (newValue, oldValue) => {
|
|
if (oldValue === false && newValue === true) { // if modal is shown
|
|
await peers.LoadPeerConfig(selectedPeer.value.Identifier, configStyle.value)
|
|
configString.value = peers.configuration
|
|
}
|
|
})
|
|
|
|
watch(() => configStyle.value, async () => {
|
|
await peers.LoadPeerConfig(selectedPeer.value.Identifier, configStyle.value)
|
|
configString.value = peers.configuration
|
|
})
|
|
|
|
function download() {
|
|
// credit: https://www.bitdegree.org/learn/javascript-download
|
|
let text = configString.value
|
|
|
|
let element = document.createElement('a')
|
|
element.setAttribute('href', 'data:application/octet-stream;charset=utf-8,' + encodeURIComponent(text))
|
|
element.setAttribute('download', selectedPeer.value.Filename)
|
|
|
|
element.style.display = 'none'
|
|
document.body.appendChild(element)
|
|
|
|
element.click()
|
|
document.body.removeChild(element)
|
|
}
|
|
|
|
function email() {
|
|
peers.MailPeerConfig(settings.Setting("MailLinkOnly"), configStyle.value, [selectedPeer.value.Identifier]).catch(e => {
|
|
notify({
|
|
title: "Failed to send mail with peer configuration!",
|
|
text: e.toString(),
|
|
type: 'error',
|
|
})
|
|
})
|
|
}
|
|
|
|
function ConfigQrUrl() {
|
|
if (props.peerId.length) {
|
|
return apiWrapper.url(`/peer/config-qr/${base64_url_encode(props.peerId)}?style=${configStyle.value}`)
|
|
}
|
|
return ''
|
|
}
|
|
|
|
</script>
|
|
|
|
<template>
|
|
<Modal :title="title" :visible="visible" @close="close">
|
|
<template #default>
|
|
<div class="d-flex justify-content-end align-items-center mb-1" v-if="selectedInterface.Mode !== 'client'">
|
|
<span class="me-2">{{ $t('modals.peer-view.style-label') }}: </span>
|
|
<div class="btn-group btn-switch-group" role="group" aria-label="Configuration Style">
|
|
<input type="radio" class="btn-check" name="configstyle" id="raw" value="raw" autocomplete="off" checked="" v-model="configStyle">
|
|
<label class="btn btn-outline-dark btn-sm" for="raw">Raw</label>
|
|
<input type="radio" class="btn-check" name="configstyle" id="wgquick" value="wgquick" autocomplete="off" checked="" v-model="configStyle">
|
|
<label class="btn btn-outline-dark btn-sm" for="wgquick">WG-Quick</label>
|
|
</div>
|
|
</div>
|
|
<div class="accordion" id="peerInformation">
|
|
<div class="accordion-item">
|
|
<h2 class="accordion-header">
|
|
<button class="accordion-button" type="button" data-bs-toggle="collapse" data-bs-target="#collapseDetails"
|
|
aria-expanded="true" aria-controls="collapseDetails">
|
|
{{ $t('modals.peer-view.section-info') }}
|
|
</button>
|
|
</h2>
|
|
<div id="collapseDetails" class="accordion-collapse collapse show" aria-labelledby="headingDetails"
|
|
data-bs-parent="#peerInformation" style="">
|
|
<div class="accordion-body">
|
|
<div class="row">
|
|
<div :class="{ 'col-md-8': selectedInterface.Mode !== 'client', 'col-md-12': selectedInterface.Mode !== 'server' }" class="col-md-8">
|
|
<ul>
|
|
<li v-if="selectedInterface.Mode !== 'client'"><strong>{{ $t('modals.peer-view.identifier') }}</strong>: {{ selectedPeer.PublicKey }}</li>
|
|
<li v-if="selectedInterface.Mode !== 'server'"><strong>{{ $t('modals.peer-view.endpoint-key') }}</strong>: {{ selectedPeer.PublicKey }}</li>
|
|
<li v-if="selectedInterface.Mode !== 'server'"><strong>{{ $t('modals.peer-view.endpoint') }}</strong>: {{ selectedPeer.Endpoint.Value }}</li>
|
|
<li v-if="selectedInterface.Mode !== 'client'"><strong>{{ $t('modals.peer-view.ip') }}</strong>: <span v-for="ip in selectedPeer.Addresses" :key="ip"
|
|
class="badge rounded-pill bg-light">{{ ip }}</span></li>
|
|
<li v-if="selectedInterface.Mode === 'server'"><strong>{{ $t('modals.peer-view.extra-allowed-ip') }}</strong>: <span v-for="ip in selectedPeer.ExtraAllowedIPs" :key="ip"
|
|
class="badge rounded-pill bg-light">{{ ip }}</span></li>
|
|
<li v-if="selectedInterface.Mode !== 'server' && selectedPeer.AllowedIPs.Value"><strong>{{ $t('modals.peer-view.allowed-ip') }}</strong>: <span v-for="ip in selectedPeer.AllowedIPs.Value" :key="ip"
|
|
class="badge rounded-pill bg-light">{{ ip }}</span></li>
|
|
<li v-if="selectedInterface.Mode !== 'server'"><strong>{{ $t('modals.peer-view.keepalive') }}</strong>: {{ selectedPeer.PersistentKeepalive.Value }}</li>
|
|
<li v-if="selectedPeer.UserDisplayName"><strong>{{ $t('modals.peer-view.user') }}</strong>: {{ selectedPeer.UserDisplayName }} ({{ selectedPeer.UserIdentifier }})</li>
|
|
<li v-else><strong>{{ $t('modals.peer-view.user') }}</strong>: {{ selectedPeer.UserIdentifier }}</li>
|
|
<li v-if="selectedPeer.Notes"><strong>{{ $t('modals.peer-view.notes') }}</strong>: {{ selectedPeer.Notes }}</li>
|
|
<li v-if="selectedPeer.ExpiresAt"><strong>{{ $t('modals.peer-view.expiry-status') }}</strong>: {{
|
|
selectedPeer.ExpiresAt }}</li>
|
|
<li v-if="selectedPeer.Disabled"><strong>{{ $t('modals.peer-view.disabled-status') }}</strong>: {{
|
|
selectedPeer.DisabledReason }}</li>
|
|
</ul>
|
|
</div>
|
|
<div class="col-md-4" v-if="selectedInterface.Mode !== 'client'">
|
|
<img class="config-qr-img" :src="ConfigQrUrl()" loading="lazy" alt="Configuration QR Code">
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="accordion-item">
|
|
<h2 class="accordion-header" id="headingStatus">
|
|
<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse"
|
|
data-bs-target="#collapseStatus" aria-expanded="false" aria-controls="collapseStatus">
|
|
{{ $t('modals.peer-view.section-status') }}
|
|
</button>
|
|
</h2>
|
|
<div id="collapseStatus" class="accordion-collapse collapse" aria-labelledby="headingStatus"
|
|
data-bs-parent="#peerInformation" style="">
|
|
<div class="accordion-body">
|
|
<div class="row">
|
|
<div class="col-md-12">
|
|
<h4>{{ $t('modals.peer-view.traffic') }}</h4>
|
|
<p><i class="fas fa-long-arrow-alt-down" :title="$t('modals.peer-view.download')"></i> {{
|
|
selectedStats.BytesReceived }} Bytes / <i class="fas fa-long-arrow-alt-up"
|
|
:title="$t('modals.peer-view.upload')"></i> {{ selectedStats.BytesTransmitted }} Bytes</p>
|
|
<h4>{{ $t('modals.peer-view.connection-status') }}</h4>
|
|
<ul>
|
|
<li>{{ $t('modals.peer-view.pingable') }}: {{ selectedStats.IsPingable }}</li>
|
|
<li>{{ $t('modals.peer-view.handshake') }}: {{ selectedStats.LastHandshake }}</li>
|
|
<li>{{ $t('modals.peer-view.connected-since') }}: {{ selectedStats.LastSessionStart }}</li>
|
|
<li>{{ $t('modals.peer-view.endpoint') }}: {{ selectedStats.EndpointAddress }}</li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div v-if="selectedInterface.Mode !== 'client'" class="accordion-item">
|
|
<h2 class="accordion-header" id="headingConfig">
|
|
<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse"
|
|
data-bs-target="#collapseConfig" aria-expanded="false" aria-controls="collapseConfig">
|
|
{{ $t('modals.peer-view.section-config') }}
|
|
</button>
|
|
</h2>
|
|
<div id="collapseConfig" class="accordion-collapse collapse" aria-labelledby="headingConfig"
|
|
data-bs-parent="#peerInformation" style="">
|
|
<div class="accordion-body">
|
|
<Prism language="ini" :code="configString"></Prism>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
<template #footer>
|
|
<div class="flex-fill text-start">
|
|
<button v-if="selectedInterface.Mode !== 'client'" @click.prevent="download" type="button" class="btn btn-primary me-1">{{
|
|
$t('modals.peer-view.button-download') }}</button>
|
|
<button v-if="selectedInterface.Mode !== 'client'" @click.prevent="email" type="button" class="btn btn-primary me-1">{{
|
|
$t('modals.peer-view.button-email') }}</button>
|
|
</div>
|
|
<button @click.prevent="close" type="button" class="btn btn-secondary">{{ $t('general.close') }}</button>
|
|
|
|
|
|
</template>
|
|
</Modal></template>
|
|
|
|
<style>
|
|
.config-qr-img {
|
|
max-width: 100%;
|
|
}
|
|
|
|
.btn-switch-group .btn {
|
|
border-width: 1px;
|
|
padding: 5px;
|
|
line-height: 1;
|
|
}
|
|
</style>
|