Server endpoints were stored only in browser sessionStorage, lost on tab close/refresh. Now endpoints are saved in a dedicated UCI config file (wireguard_dashboard) with RPCD methods to manage them. The wizard auto-saves the endpoint after tunnel creation, and peers/QR views use a dropdown of saved endpoints instead of requiring manual re-entry. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
330 lines
8.4 KiB
JavaScript
330 lines
8.4 KiB
JavaScript
'use strict';
|
|
'require baseclass';
|
|
'require rpc';
|
|
|
|
/**
|
|
* WireGuard Dashboard API
|
|
* Package: luci-app-wireguard-dashboard
|
|
* RPCD object: luci.wireguard-dashboard
|
|
*/
|
|
|
|
// Version: 0.4.0
|
|
|
|
var callStatus = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'status',
|
|
expect: { }
|
|
});
|
|
|
|
var callGetPeers = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'peers',
|
|
expect: { peers: [] }
|
|
});
|
|
|
|
var callGetInterfaces = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'interfaces',
|
|
expect: { interfaces: [] }
|
|
});
|
|
|
|
var callGenerateKeys = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'generate_keys',
|
|
expect: { }
|
|
});
|
|
|
|
var callCreateInterface = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'create_interface',
|
|
params: ['name', 'private_key', 'listen_port', 'addresses', 'mtu'],
|
|
expect: { }
|
|
});
|
|
|
|
var callAddPeer = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'add_peer',
|
|
params: ['interface', 'name', 'allowed_ips', 'public_key', 'preshared_key', 'endpoint', 'persistent_keepalive', 'private_key'],
|
|
expect: { }
|
|
});
|
|
|
|
var callRemovePeer = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'remove_peer',
|
|
params: ['interface', 'public_key'],
|
|
expect: { }
|
|
});
|
|
|
|
var callGetConfig = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'config',
|
|
expect: { }
|
|
});
|
|
|
|
var callGenerateConfig = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'generate_config',
|
|
params: ['interface', 'peer', 'private_key', 'endpoint'],
|
|
expect: { }
|
|
});
|
|
|
|
var callGenerateQR = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'generate_qr',
|
|
params: ['interface', 'peer', 'private_key', 'endpoint'],
|
|
expect: { }
|
|
});
|
|
|
|
var callGetTraffic = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'traffic',
|
|
expect: { }
|
|
});
|
|
|
|
var callInterfaceControl = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'interface_control',
|
|
params: ['interface', 'action'],
|
|
expect: { }
|
|
});
|
|
|
|
var callPeerDescriptions = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'peer_descriptions',
|
|
expect: { descriptions: {} }
|
|
});
|
|
|
|
var callBandwidthRates = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'bandwidth_rates',
|
|
expect: { rates: [] }
|
|
});
|
|
|
|
var callPingPeer = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'ping_peer',
|
|
params: ['ip'],
|
|
expect: { reachable: false }
|
|
});
|
|
|
|
var callGetEndpoints = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'get_endpoints',
|
|
expect: { }
|
|
});
|
|
|
|
var callSetEndpoint = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'set_endpoint',
|
|
params: ['id', 'name', 'address'],
|
|
expect: { }
|
|
});
|
|
|
|
var callSetDefaultEndpoint = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'set_default_endpoint',
|
|
params: ['id'],
|
|
expect: { }
|
|
});
|
|
|
|
var callDeleteEndpoint = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'delete_endpoint',
|
|
params: ['id'],
|
|
expect: { }
|
|
});
|
|
|
|
function buildEndpointSelector(endpointData, inputId) {
|
|
var endpoints = (endpointData || {}).endpoints || [];
|
|
var defaultId = (endpointData || {})['default'] || '';
|
|
|
|
if (endpoints.length === 0) {
|
|
// No saved endpoints - return a plain text input
|
|
return E('input', {
|
|
'type': 'text',
|
|
'id': inputId,
|
|
'class': 'cbi-input-text',
|
|
'placeholder': 'vpn.example.com or 203.0.113.1',
|
|
'data-mode': 'text'
|
|
});
|
|
}
|
|
|
|
var container = E('div', { 'style': 'display: flex; flex-direction: column; gap: 8px;' });
|
|
|
|
var options = endpoints.map(function(ep) {
|
|
return E('option', {
|
|
'value': ep.address,
|
|
'selected': (ep.id === defaultId) ? '' : null,
|
|
'data-id': ep.id
|
|
}, (ep.name || ep.id) + ' (' + ep.address + ')');
|
|
});
|
|
|
|
options.push(E('option', { 'value': '__custom__' }, _('Custom...')));
|
|
|
|
var select = E('select', {
|
|
'id': inputId,
|
|
'class': 'cbi-input-select',
|
|
'data-mode': 'select',
|
|
'change': function() {
|
|
var customInput = container.querySelector('.wg-custom-endpoint');
|
|
if (this.value === '__custom__') {
|
|
customInput.style.display = '';
|
|
customInput.focus();
|
|
} else {
|
|
customInput.style.display = 'none';
|
|
}
|
|
}
|
|
}, options);
|
|
|
|
var customInput = E('input', {
|
|
'type': 'text',
|
|
'class': 'cbi-input-text wg-custom-endpoint',
|
|
'placeholder': 'vpn.example.com or 203.0.113.1',
|
|
'style': 'display: none; margin-top: 4px;'
|
|
});
|
|
|
|
container.appendChild(select);
|
|
container.appendChild(customInput);
|
|
|
|
return container;
|
|
}
|
|
|
|
function getEndpointValue(inputId) {
|
|
var el = document.getElementById(inputId);
|
|
if (!el) return '';
|
|
|
|
if (el.dataset.mode === 'text') {
|
|
return el.value.trim();
|
|
}
|
|
|
|
// select mode
|
|
if (el.value === '__custom__') {
|
|
var container = el.closest('div');
|
|
var customInput = container ? container.querySelector('.wg-custom-endpoint') : null;
|
|
return customInput ? customInput.value.trim() : '';
|
|
}
|
|
|
|
return el.value;
|
|
}
|
|
|
|
function formatBytes(bytes) {
|
|
if (bytes === 0) return '0 B';
|
|
var k = 1024;
|
|
var sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
var i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
|
}
|
|
|
|
function formatLastHandshake(timestamp) {
|
|
if (!timestamp) return 'Never';
|
|
var now = Math.floor(Date.now() / 1000);
|
|
var diff = now - timestamp;
|
|
if (diff < 60) return diff + 's ago';
|
|
if (diff < 3600) return Math.floor(diff / 60) + 'm ago';
|
|
if (diff < 86400) return Math.floor(diff / 3600) + 'h ago';
|
|
return Math.floor(diff / 86400) + 'd ago';
|
|
}
|
|
|
|
function getPeerStatusClass(status) {
|
|
if (status === 'active') return 'active';
|
|
if (status === 'idle') return 'idle';
|
|
return 'inactive';
|
|
}
|
|
|
|
function shortenKey(key, length) {
|
|
if (!key) return 'N/A';
|
|
length = length || 8;
|
|
return key.substring(0, length) + '...';
|
|
}
|
|
|
|
function formatHandshake(seconds) {
|
|
if (!seconds || seconds === 0) return 'Never';
|
|
if (seconds < 60) return seconds + 's ago';
|
|
if (seconds < 3600) return Math.floor(seconds / 60) + 'm ago';
|
|
if (seconds < 86400) return Math.floor(seconds / 3600) + 'h ago';
|
|
return Math.floor(seconds / 86400) + 'd ago';
|
|
}
|
|
|
|
function formatRate(bytesPerSec) {
|
|
if (!bytesPerSec || bytesPerSec === 0) return '0 B/s';
|
|
var k = 1024;
|
|
var sizes = ['B/s', 'KB/s', 'MB/s', 'GB/s'];
|
|
var i = Math.floor(Math.log(bytesPerSec) / Math.log(k));
|
|
if (i >= sizes.length) i = sizes.length - 1;
|
|
return parseFloat((bytesPerSec / Math.pow(k, i)).toFixed(1)) + ' ' + sizes[i];
|
|
}
|
|
|
|
return baseclass.extend({
|
|
getStatus: callStatus,
|
|
getPeers: callGetPeers,
|
|
getInterfaces: callGetInterfaces,
|
|
getConfig: callGetConfig,
|
|
getTraffic: callGetTraffic,
|
|
generateKeys: callGenerateKeys,
|
|
createInterface: callCreateInterface,
|
|
addPeer: callAddPeer,
|
|
removePeer: callRemovePeer,
|
|
generateConfig: callGenerateConfig,
|
|
generateQR: callGenerateQR,
|
|
interfaceControl: callInterfaceControl,
|
|
getPeerDescriptions: callPeerDescriptions,
|
|
getBandwidthRates: callBandwidthRates,
|
|
pingPeer: callPingPeer,
|
|
getEndpoints: callGetEndpoints,
|
|
setEndpoint: callSetEndpoint,
|
|
setDefaultEndpoint: callSetDefaultEndpoint,
|
|
deleteEndpoint: callDeleteEndpoint,
|
|
buildEndpointSelector: buildEndpointSelector,
|
|
getEndpointValue: getEndpointValue,
|
|
formatBytes: formatBytes,
|
|
formatLastHandshake: formatLastHandshake,
|
|
getPeerStatusClass: getPeerStatusClass,
|
|
shortenKey: shortenKey,
|
|
formatHandshake: formatHandshake,
|
|
formatRate: formatRate,
|
|
|
|
// Aggregate function for overview page
|
|
getAllData: function() {
|
|
return Promise.all([
|
|
callStatus(),
|
|
callGetPeers(),
|
|
callGetInterfaces(),
|
|
callGetTraffic(),
|
|
callPeerDescriptions()
|
|
]).then(function(results) {
|
|
// Handle RPC expect unwrapping - results may be array or object
|
|
var peersData = results[1] || [];
|
|
var interfacesData = results[2] || [];
|
|
|
|
return {
|
|
status: results[0] || {},
|
|
peers: Array.isArray(peersData) ? peersData : (peersData.peers || []),
|
|
interfaces: Array.isArray(interfacesData) ? interfacesData : (interfacesData.interfaces || []),
|
|
traffic: results[3] || {},
|
|
descriptions: (results[4] || {}).descriptions || {}
|
|
};
|
|
});
|
|
},
|
|
|
|
// Get data with bandwidth rates for real-time monitoring
|
|
getMonitoringData: function() {
|
|
return Promise.all([
|
|
callStatus(),
|
|
callGetPeers(),
|
|
callBandwidthRates(),
|
|
callPeerDescriptions()
|
|
]).then(function(results) {
|
|
// Handle RPC expect unwrapping - results may be array or object
|
|
var peersData = results[1] || [];
|
|
var ratesData = results[2] || [];
|
|
|
|
return {
|
|
status: results[0] || {},
|
|
peers: Array.isArray(peersData) ? peersData : (peersData.peers || []),
|
|
rates: Array.isArray(ratesData) ? ratesData : (ratesData.rates || []),
|
|
descriptions: (results[3] || {}).descriptions || {}
|
|
};
|
|
});
|
|
}
|
|
});
|