Major structural reorganization and feature additions: ## Folder Reorganization - Move 17 luci-app-* packages to package/secubox/ (except luci-app-secubox core hub) - Update all tooling to support new structure: - secubox-tools/quick-deploy.sh: search both locations - secubox-tools/validate-modules.sh: validate both directories - secubox-tools/fix-permissions.sh: fix permissions in both locations - .github/workflows/test-validate.yml: build from both paths - Update README.md links to new package/secubox/ paths ## AppStore Migration (Complete) - Add catalog entries for all remaining luci-app packages: - network-tweaks.json: Network optimization tools - secubox-bonus.json: Documentation & demos hub - Total: 24 apps in AppStore catalog (22 existing + 2 new) - New category: 'documentation' for docs/demos/tutorials ## VHost Manager v2.0 Enhancements - Add profile activation system for Internal Services and Redirects - Implement createVHost() API wrapper for template-based deployment - Fix Virtual Hosts view rendering with proper LuCI patterns - Fix RPCD backend shell script errors (remove invalid local declarations) - Extend backend validation for nginx return directives (redirect support) - Add section_id parameter for named VHost profiles - Add Remove button to Redirects page for feature parity - Update README to v2.0 with comprehensive feature documentation ## Network Tweaks Dashboard - Close button added to component details modal Files changed: 340+ (336 renames with preserved git history) Packages affected: 19 luci-app, 2 secubox-app, 1 theme, 4 tools 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
126 lines
3.0 KiB
JavaScript
126 lines
3.0 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: 'get_peers',
|
|
expect: { peers: [] }
|
|
});
|
|
|
|
var callGetInterfaces = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'get_interfaces',
|
|
expect: { interfaces: [] }
|
|
});
|
|
|
|
var callGenerateKeys = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'generate_keys',
|
|
expect: { }
|
|
});
|
|
|
|
var callAddPeer = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'add_peer',
|
|
params: ['interface', 'name', 'allowed_ips', 'public_key', 'preshared_key', 'endpoint', 'persistent_keepalive'],
|
|
expect: { success: false }
|
|
});
|
|
|
|
var callRemovePeer = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'remove_peer',
|
|
params: ['interface', 'public_key'],
|
|
expect: { success: false }
|
|
});
|
|
|
|
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: { config: '' }
|
|
});
|
|
|
|
var callGenerateQR = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'generate_qr',
|
|
params: ['interface', 'peer', 'private_key', 'endpoint'],
|
|
expect: { qrcode: '' }
|
|
});
|
|
|
|
var callGetTraffic = rpc.declare({
|
|
object: 'luci.wireguard-dashboard',
|
|
method: 'traffic',
|
|
expect: { }
|
|
});
|
|
|
|
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';
|
|
}
|
|
|
|
return baseclass.extend({
|
|
getStatus: callStatus,
|
|
getPeers: callGetPeers,
|
|
getInterfaces: callGetInterfaces,
|
|
getConfig: callGetConfig,
|
|
getTraffic: callGetTraffic,
|
|
generateKeys: callGenerateKeys,
|
|
addPeer: callAddPeer,
|
|
removePeer: callRemovePeer,
|
|
generateConfig: callGenerateConfig,
|
|
generateQR: callGenerateQR,
|
|
formatBytes: formatBytes,
|
|
formatLastHandshake: formatLastHandshake,
|
|
|
|
// Aggregate function for overview page
|
|
getAllData: function() {
|
|
return Promise.all([
|
|
callStatus(),
|
|
callGetPeers(),
|
|
callGetInterfaces(),
|
|
callGetTraffic()
|
|
]).then(function(results) {
|
|
return {
|
|
status: results[0] || {},
|
|
peers: results[1] || { peers: [] },
|
|
interfaces: results[2] || { interfaces: [] },
|
|
traffic: results[3] || {}
|
|
};
|
|
});
|
|
}
|
|
});
|