secubox-openwrt/package/secubox/luci-app-secubox-netifyd/htdocs/luci-static/resources/view/secubox-netifyd/applications.js
CyberMind-FR 8fcd34abd0 feat: Netifyd Integration & Build System Improvements (v0.9.1)
Major updates:
- Replace luci-app-netifyd-dashboard with enhanced luci-app-secubox-netifyd
- Add netifyd 5.2.1 package with GCC 13.3/C++17 build fixes
- Fix nd-risks.cpp compilation errors via inline static maps patch
- Enhance local-build.sh with improved package building workflow
- Update secubox-core scripts version to v0.9.1

New Features:
- Complete netifyd dashboard with flows, devices, applications, and settings
- Local data collection with netifyd-collector
- Automated cron-based data aggregation
- RPCd integration for real-time statistics

Build Fixes:
- Patch 001: Fix C++17 inline static maps in nd-risks.hpp and nd-protos.hpp
- Patch 003: Skip ndpi tests to resolve roaring_v2 dependency issues
- Add libatomic dependency
- Include libnetifyd shared libraries in package

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-05 17:35:11 +01:00

314 lines
9.3 KiB
JavaScript

'use strict';
'require view';
'require poll';
'require ui';
'require dom';
'require secubox-netifyd/api as netifydAPI';
return view.extend({
refreshInterval: 10,
appsData: [],
sortColumn: 'bytes',
sortDirection: 'desc',
load: function() {
return netifydAPI.getTopApplications();
},
sortApplications: function(apps, column, direction) {
return apps.slice().sort(function(a, b) {
var valA = a[column] || 0;
var valB = b[column] || 0;
if (direction === 'asc') {
return valA > valB ? 1 : valA < valB ? -1 : 0;
} else {
return valA < valB ? 1 : valA > valB ? -1 : 0;
}
});
},
handleSort: function(column, ev) {
if (this.sortColumn === column) {
this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';
} else {
this.sortColumn = column;
this.sortDirection = 'desc';
}
this.renderApplicationsTable();
},
handleExport: function(ev) {
var csvContent = 'Application,Flows,Packets,Bytes,Percentage\n';
var totalBytes = this.appsData.reduce(function(sum, app) {
return sum + (app.bytes || 0);
}, 0);
this.appsData.forEach(function(app) {
var percentage = totalBytes > 0 ? ((app.bytes / totalBytes) * 100).toFixed(2) : 0;
csvContent += [
'"' + (app.name || 'Unknown') + '"',
app.flows || 0,
app.packets || 0,
app.bytes || 0,
percentage
].join(',') + '\n';
});
var blob = new Blob([csvContent], { type: 'text/csv' });
var url = window.URL.createObjectURL(blob);
var a = document.createElement('a');
a.href = url;
a.download = 'netifyd-applications-' + new Date().toISOString().slice(0,10) + '.csv';
a.click();
window.URL.revokeObjectURL(url);
ui.addNotification(null, E('p', _('Applications exported to CSV')), 'info');
},
renderSummaryCards: function(apps) {
var totalBytes = apps.reduce(function(sum, app) {
return sum + (app.bytes || 0);
}, 0);
var totalFlows = apps.reduce(function(sum, app) {
return sum + (app.flows || 0);
}, 0);
var totalPackets = apps.reduce(function(sum, app) {
return sum + (app.packets || 0);
}, 0);
var cards = [
{
title: _('Unique Applications'),
value: apps.length.toString(),
icon: 'cubes',
gradient: 'linear-gradient(135deg, #667eea 0%, #764ba2 100%)'
},
{
title: _('Total Flows'),
value: totalFlows.toString(),
icon: 'stream',
gradient: 'linear-gradient(135deg, #84fab0 0%, #8fd3f4 100%)'
},
{
title: _('Total Packets'),
value: totalPackets.toLocaleString(),
icon: 'boxes',
gradient: 'linear-gradient(135deg, #a18cd1 0%, #fbc2eb 100%)'
},
{
title: _('Total Traffic'),
value: netifydAPI.formatBytes(totalBytes),
icon: 'chart-line',
gradient: 'linear-gradient(135deg, #fa709a 0%, #fee140 100%)'
}
];
return E('div', {
'style': 'display: grid; grid-template-columns: repeat(auto-fit, minmax(220px, 1fr)); gap: 1rem; margin-bottom: 1.5rem'
}, cards.map(function(card) {
return E('div', {
'style': 'background: ' + card.gradient + '; color: white; padding: 1.5rem; border-radius: 8px; box-shadow: 0 4px 6px rgba(0,0,0,0.1)'
}, [
E('div', { 'style': 'display: flex; justify-content: space-between; align-items: flex-start; margin-bottom: 1rem' }, [
E('div', { 'style': 'font-size: 0.9em; opacity: 0.9' }, card.title),
E('i', {
'class': 'fa fa-' + card.icon,
'style': 'font-size: 2em; opacity: 0.3'
})
]),
E('div', { 'style': 'font-size: 2em; font-weight: bold' }, card.value)
]);
}.bind(this)));
},
renderApplicationsTable: function() {
var container = document.getElementById('apps-table-container');
if (!container) return;
var apps = this.appsData;
var sortedApps = this.sortApplications(apps, this.sortColumn, this.sortDirection);
var totalBytes = apps.reduce(function(sum, app) {
return sum + (app.bytes || 0);
}, 0);
var appColors = [
'#3b82f6', '#10b981', '#f59e0b', '#ef4444', '#8b5cf6',
'#ec4899', '#14b8a6', '#f97316', '#06b6d4', '#84cc16',
'#6366f1', '#f43f5e', '#0ea5e9', '#8b5cf6', '#22c55e'
];
var getSortIcon = function(column) {
if (this.sortColumn !== column) {
return E('i', { 'class': 'fa fa-sort', 'style': 'opacity: 0.3' });
}
return E('i', {
'class': 'fa fa-sort-' + (this.sortDirection === 'asc' ? 'up' : 'down'),
'style': 'color: #3b82f6'
});
}.bind(this);
dom.content(container, [
apps.length > 0 ? E('div', { 'class': 'table', 'style': 'font-size: 0.95em' }, [
// Header
E('div', { 'class': 'tr table-titles' }, [
E('div', {
'class': 'th left',
'style': 'width: 30%; cursor: pointer',
'click': ui.createHandlerFn(this, 'handleSort', 'name')
}, [
_('Application'),
' ',
getSortIcon('name')
]),
E('div', {
'class': 'th center',
'style': 'width: 15%; cursor: pointer',
'click': ui.createHandlerFn(this, 'handleSort', 'flows')
}, [
_('Flows'),
' ',
getSortIcon('flows')
]),
E('div', {
'class': 'th center',
'style': 'width: 15%; cursor: pointer',
'click': ui.createHandlerFn(this, 'handleSort', 'packets')
}, [
_('Packets'),
' ',
getSortIcon('packets')
]),
E('div', {
'class': 'th right',
'style': 'width: 20%; cursor: pointer',
'click': ui.createHandlerFn(this, 'handleSort', 'bytes')
}, [
_('Total Traffic'),
' ',
getSortIcon('bytes')
]),
E('div', { 'class': 'th', 'style': 'width: 20%' }, _('% of Total'))
]),
// Rows
sortedApps.map(function(app, idx) {
var percentage = totalBytes > 0 ? (app.bytes / totalBytes * 100) : 0;
var color = appColors[idx % appColors.length];
return E('div', {
'class': 'tr',
'style': idx % 2 === 0 ? 'background: #f9fafb' : ''
}, [
E('div', { 'class': 'td left', 'style': 'width: 30%' }, [
E('div', { 'style': 'display: flex; align-items: center; gap: 0.5rem' }, [
E('div', {
'style': 'width: 10px; height: 10px; border-radius: 50%; background: ' + color + '; flex-shrink: 0'
}),
E('strong', app.name || 'Unknown')
])
]),
E('div', { 'class': 'td center', 'style': 'width: 15%' },
(app.flows || 0).toLocaleString()),
E('div', { 'class': 'td center', 'style': 'width: 15%' },
(app.packets || 0).toLocaleString()),
E('div', { 'class': 'td right', 'style': 'width: 20%' }, [
E('strong', { 'style': 'color: ' + color },
netifydAPI.formatBytes(app.bytes || 0))
]),
E('div', { 'class': 'td', 'style': 'width: 20%' }, [
E('div', {
'style': 'background: #e5e7eb; border-radius: 10px; height: 24px; position: relative; overflow: hidden'
}, [
E('div', {
'style': 'background: ' + color + '; height: 100%; width: ' + percentage + '%; transition: width 0.3s ease; border-radius: 10px'
}),
E('span', {
'style': 'position: absolute; left: 50%; top: 50%; transform: translate(-50%, -50%); font-size: 0.8em; font-weight: bold; color: ' + (percentage > 40 ? '#fff' : '#374151')
}, percentage.toFixed(1) + '%')
])
])
]);
}.bind(this))
]) : E('div', {
'class': 'alert-message info',
'style': 'text-align: center; padding: 3rem'
}, [
E('i', { 'class': 'fa fa-cubes', 'style': 'font-size: 3em; opacity: 0.3; display: block; margin-bottom: 1rem' }),
E('h4', _('No Application Data')),
E('p', { 'class': 'text-muted' }, _('No applications have been detected yet')),
E('small', _('Data will appear once network traffic is analyzed'))
])
]);
},
render: function(data) {
this.appsData = data.applications || [];
var self = this;
// Set up polling
poll.add(L.bind(function() {
return netifydAPI.getTopApplications().then(L.bind(function(result) {
this.appsData = result.applications || [];
this.renderApplicationsTable();
}, this));
}, this), this.refreshInterval);
return E('div', { 'class': 'cbi-map' }, [
E('h2', [
E('i', { 'class': 'fa fa-cubes', 'style': 'margin-right: 0.5rem' }),
_('Detected Applications')
]),
E('div', { 'class': 'cbi-map-descr' },
_('Applications detected and identified by Netifyd deep packet inspection engine')),
E('div', { 'class': 'cbi-section' }, [
E('div', { 'class': 'cbi-section-node' }, [
E('div', { 'style': 'display: flex; justify-content: space-between; align-items: center; margin-bottom: 1rem' }, [
E('h3', { 'style': 'margin: 0' }, [
E('i', { 'class': 'fa fa-chart-bar', 'style': 'margin-right: 0.5rem' }),
_('Summary')
]),
E('button', {
'class': 'btn btn-primary',
'click': ui.createHandlerFn(this, 'handleExport')
}, [
E('i', { 'class': 'fa fa-download' }),
' ',
_('Export CSV')
])
]),
this.renderSummaryCards(this.appsData)
])
]),
E('div', { 'class': 'cbi-section' }, [
E('h3', [
E('i', { 'class': 'fa fa-list', 'style': 'margin-right: 0.5rem' }),
_('Application List'),
' ',
E('span', {
'class': 'badge',
'style': 'background: #3b82f6; color: white; margin-left: 0.5rem'
}, this.appsData.length)
]),
E('div', { 'class': 'cbi-section-node' }, [
E('div', { 'id': 'apps-table-container' })
])
])
]);
},
addFooter: function() {
this.renderApplicationsTable();
},
handleSaveApply: null,
handleSave: null,
handleReset: null
});