archy/neode-ui/src/views/settings/VpnStatusSection.vue
Dorian 0c02d06a66 feat: deploy-to-target supports .253 + mesh/federation/VPN updates
- Add deploy_secondary() function for deploying to multiple LAN nodes
- --both now deploys to .198 and .253 (previously .198 only)
- Fleet deploy updated for 3 LAN nodes
- Mesh DM fixes: protocol frame format, DM-via-channel routing
- Federation pending requests, discover modal
- VPN status UI improvements
- Image versions and container specs updates

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-18 11:07:08 -04:00

96 lines
3.6 KiB
Vue

<script setup lang="ts">
import { ref, onMounted } from 'vue'
import { rpcClient } from '@/api/rpc-client'
interface VpnStatus {
connected: boolean
provider: string | null
interface: string | null
ip_address: string | null
hostname: string | null
peers_connected: number
bytes_in: number
bytes_out: number
}
const vpnStatus = ref<VpnStatus | null>(null)
const loading = ref(true)
const error = ref('')
async function fetchVpnStatus() {
try {
loading.value = true
error.value = ''
const result = await rpcClient.call({ method: 'vpn.status', params: {} })
vpnStatus.value = result as VpnStatus
} catch (e) {
error.value = e instanceof Error ? e.message : 'Failed to get VPN status'
} finally {
loading.value = false
}
}
function formatBytes(bytes: number): string {
if (bytes === 0) return '0 B'
const units = ['B', 'KB', 'MB', 'GB']
const i = Math.floor(Math.log(bytes) / Math.log(1024))
return `${(bytes / Math.pow(1024, i)).toFixed(1)} ${units[i]}`
}
onMounted(fetchVpnStatus)
</script>
<template>
<div class="glass-card px-6 py-6 mb-6">
<div class="flex items-center justify-between mb-4">
<div class="flex items-center gap-3">
<div class="w-10 h-10 rounded-xl bg-purple-500/20 flex items-center justify-center">
<svg class="w-5 h-5 text-purple-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 15v2m-6 4h12a2 2 0 002-2v-6a2 2 0 00-2-2H6a2 2 0 00-2 2v6a2 2 0 002 2zm10-10V7a4 4 0 00-8 0v4h8z" />
</svg>
</div>
<div>
<h3 class="text-base font-semibold text-white/96">Nostr VPN</h3>
<p class="text-sm text-white/50">Mesh VPN with Nostr signaling</p>
</div>
</div>
<div v-if="vpnStatus" class="flex items-center gap-2">
<span
class="w-2.5 h-2.5 rounded-full"
:class="vpnStatus.connected ? 'bg-green-400 animate-pulse' : 'bg-white/30'"
/>
<span class="text-sm" :class="vpnStatus.connected ? 'text-green-400' : 'text-white/50'">
{{ vpnStatus.connected ? 'Connected' : 'Inactive' }}
</span>
</div>
</div>
<div v-if="loading" class="text-sm text-white/50">Loading VPN status...</div>
<div v-else-if="vpnStatus?.connected" class="grid grid-cols-2 gap-3">
<div class="bg-white/5 rounded-lg px-3 py-2">
<div class="text-xs text-white/50 mb-1">Provider</div>
<div class="text-sm font-medium text-white/90">{{ vpnStatus.provider || 'wireguard' }}</div>
</div>
<div class="bg-white/5 rounded-lg px-3 py-2">
<div class="text-xs text-white/50 mb-1">Peers</div>
<div class="text-sm font-medium text-white/90">{{ vpnStatus.peers_connected }}</div>
</div>
<div v-if="vpnStatus.ip_address" class="bg-white/5 rounded-lg px-3 py-2">
<div class="text-xs text-white/50 mb-1">VPN Address</div>
<div class="text-sm font-mono text-white/90">{{ vpnStatus.ip_address }}</div>
</div>
<div v-if="vpnStatus.bytes_in || vpnStatus.bytes_out" class="bg-white/5 rounded-lg px-3 py-2">
<div class="text-xs text-white/50 mb-1">Traffic</div>
<div class="text-sm text-white/90">{{ formatBytes(vpnStatus.bytes_in) }} / {{ formatBytes(vpnStatus.bytes_out) }}</div>
</div>
</div>
<div v-else-if="!vpnStatus?.connected" class="text-sm text-white/50">
VPN will activate automatically when peers are discovered via Nostr relays.
</div>
<div v-if="error" class="text-sm text-red-400 mt-2">{{ error }}</div>
</div>
</template>