secubox-openwrt/luci-app-system-hub/htdocs/luci-static/resources/view/system-hub/overview.js
2025-12-29 09:03:49 +01:00

316 lines
11 KiB
JavaScript

'use strict';
'require view';
'require ui';
'require dom';
'require poll';
'require system-hub/api as API';
'require secubox-theme/theme as Theme';
'require system-hub/nav as HubNav';
var shLang = (typeof L !== 'undefined' && L.env && L.env.lang) ||
(document.documentElement && document.documentElement.getAttribute('lang')) ||
(navigator.language ? navigator.language.split('-')[0] : 'en');
Theme.init({ language: shLang });
return view.extend({
sysInfo: null,
healthData: null,
load: function() {
return Promise.all([
API.getSystemInfo(),
API.getHealth()
]);
},
render: function(data) {
this.sysInfo = data[0] || {};
this.healthData = data[1] || {};
var container = E('div', { 'class': 'sh-overview' }, [
E('link', { 'rel': 'stylesheet', 'href': L.resource('secubox-theme/secubox-theme.css') }),
E('link', { 'rel': 'stylesheet', 'href': L.resource('system-hub/common.css') }),
E('link', { 'rel': 'stylesheet', 'href': L.resource('system-hub/dashboard.css') }),
E('link', { 'rel': 'stylesheet', 'href': L.resource('system-hub/overview.css') }),
HubNav.renderTabs('overview'),
this.renderPageHeader(),
this.renderInfoGrid(),
this.renderResourceMonitors(),
this.renderQuickStatus()
]);
var self = this;
poll.add(function() {
return Promise.all([
API.getSystemInfo(),
API.getHealth()
]).then(function(refresh) {
self.sysInfo = refresh[0] || {};
self.healthData = refresh[1] || {};
self.updateDynamicSections();
});
}, 30);
return container;
},
renderPageHeader: function() {
var uptime = this.sysInfo.uptime_formatted || '0d 0h 0m';
var hostname = this.sysInfo.hostname || 'OpenWrt';
var kernel = this.sysInfo.kernel || '-';
var score = (this.healthData.score || 0);
var version = this.sysInfo.version || _('Unknown');
var stats = [
{ label: _('Version'), value: version, icon: '🏷️' },
{ label: _('Uptime'), value: uptime, icon: '⏱' },
{ label: _('Hostname'), value: hostname, icon: '🖥' },
{ label: _('Kernel'), value: kernel, copy: kernel, icon: '🧬' },
{ label: _('Health'), value: score + '/100', icon: '❤️' }
];
return E('div', { 'class': 'sh-page-header sh-page-header-lite' }, [
E('div', {}, [
E('h2', { 'class': 'sh-page-title' }, [
E('span', { 'class': 'sh-page-title-icon' }, '⚙️'),
_('System Control Center')
]),
E('p', { 'class': 'sh-page-subtitle' }, _('Unified telemetry & orchestration'))
]),
E('div', { 'class': 'sh-header-meta' }, stats.map(this.renderHeaderChip, this))
]);
},
renderHeaderChip: function(stat) {
var chip = E('div', { 'class': 'sh-header-chip' }, [
E('span', { 'class': 'sh-chip-icon' }, stat.icon || '•'),
E('div', { 'class': 'sh-chip-text' }, [
E('span', { 'class': 'sh-chip-label' }, stat.label),
E('strong', {}, stat.value || '-')
])
]);
if (stat.copy && navigator.clipboard) {
chip.style.cursor = 'pointer';
chip.addEventListener('click', function() {
navigator.clipboard.writeText(stat.copy).then(function() {
ui.addNotification(null, E('p', {}, _('Copied to clipboard')), 'info');
});
});
}
return chip;
},
renderInfoGrid: function() {
var cards = [
{ id: 'hostname', label: _('Hostname'), value: this.sysInfo.hostname || 'OpenWrt', action: _('Edit'), handler: this.openSystemSettings },
{ id: 'uptime', label: _('Uptime'), value: this.sysInfo.uptime_formatted || '0d 0h 0m' },
{ id: 'load', label: _('Load Avg (1/5/15)'), value: (this.sysInfo.load || []).join(' · ') || '0.00 · 0.00 · 0.00', monospace: true },
{ id: 'kernel', label: _('Kernel Version'), value: this.sysInfo.kernel || '-', action: _('Copy'), handler: this.copyKernel.bind(this) }
];
return E('section', { 'class': 'sh-info-grid', 'id': 'sh-info-grid' },
cards.map(function(card) {
return E('div', { 'class': 'sh-info-card' }, [
E('div', { 'class': 'sh-info-label' }, card.label),
E('div', {
'class': 'sh-info-value' + (card.monospace ? ' mono' : ''),
'id': 'sh-info-' + card.id
}, card.value),
card.action ? E('button', {
'class': 'sh-info-action',
'type': 'button',
'click': card.handler
}, card.action) : ''
]);
}, this)
);
},
renderResourceMonitors: function() {
var monitors = [
this.createMonitor('cpu', _('CPU Usage'), this.healthData.cpu || {}, '🔥'),
this.createMonitor('memory', _('Memory'), this.healthData.memory || {}, '💾'),
this.createMonitor('storage', _('Storage'), this.healthData.disk || {}, '💿'),
this.createMonitor('network', _('Network'), this.healthData.network || {}, '🌐')
];
return E('section', { 'class': 'sh-monitor-panel' }, [
E('div', { 'class': 'sh-section-header' }, [
E('h2', {}, _('Resource Monitors')),
E('p', {}, _('Live usage for CPU, RAM, Storage, Network'))
]),
E('div', { 'class': 'sh-monitor-grid', 'id': 'sh-monitor-grid' }, monitors)
]);
},
createMonitor: function(id, label, data, icon) {
var percent = Math.round(data.percent || data.usage || 0);
var detail = '';
if (id === 'memory' || id === 'storage') {
var used = API.formatBytes((data.used_kb || 0) * 1024);
var total = API.formatBytes((data.total_kb || 0) * 1024);
detail = used + ' / ' + total;
} else if (id === 'network') {
detail = data.wan_up ? _('Online') : _('Offline');
percent = data.wan_up ? 100 : 0;
} else {
detail = _('Load: ') + (data.load_1m || data.load || '0.00');
}
return E('div', { 'class': 'sh-monitor-card' }, [
E('div', { 'class': 'sh-monitor-icon' }, icon),
E('div', { 'class': 'sh-monitor-info' }, [
E('div', { 'class': 'sh-monitor-label' }, label),
E('div', { 'class': 'sh-monitor-detail', 'id': 'sh-monitor-detail-' + id }, detail)
]),
E('div', { 'class': 'sh-monitor-progress' }, [
E('div', {
'class': 'sh-monitor-bar',
'id': 'sh-monitor-bar-' + id,
'style': 'width:' + percent + '%'
})
]),
E('div', { 'class': 'sh-monitor-percent', 'id': 'sh-monitor-percent-' + id }, percent + '%')
]);
},
renderQuickStatus: function() {
var status = this.sysInfo.status || {};
var indicators = [
{ id: 'internet', label: _('Internet Connectivity'), state: status.internet, icon: '🌍' },
{ id: 'dns', label: _('DNS Resolution'), state: status.dns, icon: '🧭' },
{ id: 'ntp', label: _('NTP Sync'), state: status.ntp, icon: '⏱' },
{ id: 'firewall', label: _('Firewall Rules'), state: status.firewall, icon: '🛡', extra: status.firewall_rules ? status.firewall_rules + _(' rules') : '' }
];
return E('section', { 'class': 'sh-status-panel' }, [
E('div', { 'class': 'sh-section-header' }, [
E('h2', {}, _('Quick Status Indicators')),
E('p', {}, _('Checks for connectivity, DNS, NTP, firewall'))
]),
E('div', { 'class': 'sh-status-grid', 'id': 'sh-status-grid' },
indicators.map(function(item) {
return E('div', {
'class': 'sh-status-card ' + this.getStatusClass(item.state),
'id': 'sh-status-' + item.id
}, [
E('div', { 'class': 'sh-status-icon' }, item.icon),
E('div', { 'class': 'sh-status-body' }, [
E('strong', {}, item.label),
E('span', { 'class': 'sh-status-value', 'id': 'sh-status-label-' + item.id },
this.getStatusLabel(item.state)),
item.extra ? E('span', { 'class': 'sh-status-extra', 'id': 'sh-status-extra-' + item.id }, item.extra) : ''
])
]);
}, this))
]);
},
updateDynamicSections: function() {
this.updateInfoGrid();
this.updateMonitors();
this.updateStatuses();
},
updateInfoGrid: function() {
var mappings = {
hostname: this.sysInfo.hostname || 'OpenWrt',
uptime: this.sysInfo.uptime_formatted || '0d 0h 0m',
load: (this.sysInfo.load || []).join(' · ') || '0.00 · 0.00 · 0.00',
kernel: this.sysInfo.kernel || '-'
};
Object.keys(mappings).forEach(function(key) {
var node = document.getElementById('sh-info-' + key);
if (node) node.textContent = mappings[key];
});
var score = this.healthData.score || 0;
var scoreNode = document.getElementById('sh-score-value');
var labelNode = document.getElementById('sh-score-label');
if (scoreNode) scoreNode.textContent = score;
if (labelNode) labelNode.textContent = this.getScoreLabel(score);
},
updateMonitors: function() {
var health = this.healthData || {};
var map = {
cpu: { percent: Math.round((health.cpu && (health.cpu.percent || health.cpu.usage)) || 0), detail: _('Load: ') + ((health.cpu && (health.cpu.load_1m || health.cpu.load)) || '0.00') },
memory: {
percent: Math.round((health.memory && (health.memory.percent || health.memory.usage)) || 0),
detail: API.formatBytes(((health.memory && health.memory.used_kb) || 0) * 1024) + ' / ' +
API.formatBytes(((health.memory && health.memory.total_kb) || 0) * 1024)
},
storage: {
percent: Math.round((health.disk && (health.disk.percent || health.disk.usage)) || 0),
detail: API.formatBytes(((health.disk && health.disk.used_kb) || 0) * 1024) + ' / ' +
API.formatBytes(((health.disk && health.disk.total_kb) || 0) * 1024)
},
network: {
percent: (health.network && health.network.wan_up) ? 100 : 0,
detail: (health.network && health.network.wan_up) ? _('Online') : _('Offline')
}
};
Object.keys(map).forEach(function(key) {
var percent = Math.min(100, Math.max(0, map[key].percent));
var bar = document.getElementById('sh-monitor-bar-' + key);
var val = document.getElementById('sh-monitor-percent-' + key);
var detail = document.getElementById('sh-monitor-detail-' + key);
if (bar) bar.style.width = percent + '%';
if (val) val.textContent = percent + '%';
if (detail) detail.textContent = map[key].detail;
});
},
updateStatuses: function() {
var status = this.sysInfo.status || {};
var ids = ['internet', 'dns', 'ntp', 'firewall'];
ids.forEach(function(id) {
var node = document.getElementById('sh-status-' + id);
var label = document.getElementById('sh-status-label-' + id);
var extra = document.getElementById('sh-status-extra-' + id);
var state = status[id];
if (node) {
node.className = 'sh-status-card ' + this.getStatusClass(state);
}
if (label) label.textContent = this.getStatusLabel(state);
if (extra && id === 'firewall') {
var rules = status.firewall_rules ? status.firewall_rules + _(' rules') : '';
extra.textContent = rules;
}
}, this);
},
getStatusClass: function(state) {
if (state === undefined || state === null) return 'unknown';
return state ? 'ok' : 'warn';
},
getStatusLabel: function(state) {
if (state === undefined || state === null) return _('Unknown');
return state ? _('Healthy') : _('Attention');
},
getScoreLabel: function(score) {
if (score >= 80) return _('Excellent');
if (score >= 60) return _('Good');
if (score >= 40) return _('Warning');
return _('Critical');
},
openSystemSettings: function() {
window.location.href = L.url('admin/secubox/system/system-hub/settings');
},
copyKernel: function() {
if (navigator.clipboard && this.sysInfo.kernel) {
navigator.clipboard.writeText(this.sysInfo.kernel);
ui.addNotification(null, E('p', {}, _('Kernel version copied')), 'info');
}
}
});