1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
'use strict';
'require baseclass';
'require fs';
'require network';
'require rpc';
'require ui';
/* returns per odhcp6c active interface JSON like:
{"result":{"eth1":{"dhcp_solicit":3,"dhcp_advertise":3,"dhcp_request":3,...}}} */
const callOdhcp6cStats = rpc.declare({
object: 'luci',
method: 'getOdhcp6cStats',
expect: { '': {} },
});
function progressbar(value, max, byte) {
const vn = parseInt(value) || 0;
const mn = parseInt(max) || 100;
const fv = byte ? String.format('%1024.2mB', value) : value;
const fm = byte ? String.format('%1024.2mB', max) : max;
const pc = Math.floor((100 / mn) * vn);
return E('div', {
'class': 'cbi-progressbar',
'title': '%s / %s (%d%%)'.format(fv, fm, pc)
}, E('div', { 'style': 'width:%.2f%%'.format(pc) }));
}
function renderbox(ifc, ipv6, dhcpv6_stats) {
const dev = ifc.getL3Device();
const active = (dev && ifc.getProtocol() != 'none');
const addrs = (ipv6 ? ifc.getIP6Addrs() : ifc.getIPAddrs()) || [];
const dnssrv = (ipv6 ? ifc.getDNS6Addrs() : ifc.getDNSAddrs()) || [];
const expires = ifc.getExpiry();
const uptime = ifc.getUptime();
function addEntries(label, array) {
return Array.isArray(array) ? array.flatMap((item) => [label, item]) : [label, null];
}
function addDhcpv6Stats() {
if (ipv6 && ifc.getProtocol() === 'dhcpv6' && dhcpv6_stats && dhcpv6_stats[dev.device]) {
const arr = [];
for (const [pkt_type, count] of Object.entries(dhcpv6_stats[dev.device]))
arr.push(pkt_type.replace('dhcp_', _('DHCPv6') + ' '), `${count} ${_('pkts', 'packets, abbreviated')}`);
return [_('DHCPv6 Statistics'), E('span', { 'class': 'cbi-tooltip-container'}, [
'📊',
E('span', { 'class': 'cbi-tooltip' }, ui.itemlist(E('span'), arr))
])];
}
return ['', null];
}
return E('div', { class: 'ifacebox' }, [
E('div', { class: 'ifacebox-head center ' + (active ? 'active' : '') },
E('strong', ipv6 ? _('IPv6 Upstream') : _('IPv4 Upstream'))),
E('div', { class: 'ifacebox-body left' }, [
L.itemlist(E('span'), [
_('Protocol'), ifc.getI18n() || E('em', _('Not connected')),
...addEntries(_('Prefix Delegated'), ipv6 ? ifc.getIP6Prefixes?.() : null),
...addEntries(_('Address'), addrs),
_('Gateway'), ipv6 ? (ifc.getGateway6Addr() || '::') : (ifc.getGatewayAddr() || '0.0.0.0'),
...addEntries(_('DNS'), dnssrv),
_('Expires'), (expires != null && expires > -1) ? '%t'.format(expires) : null,
_('Connected'), (uptime > 0) ? '%t'.format(uptime) : null,
...addDhcpv6Stats(),
]),
E('div', {}, renderBadge(
L.resource('icons/%s.svg').format(dev ? dev.getType() : 'ethernet_disabled'), null,
_('Device'), dev ? dev.getI18n() : '-',
_('MAC address'), dev.getMAC())
)
])
]);
}
return baseclass.extend({
title: _('Network'),
load() {
return Promise.all([
fs.trimmed('/proc/sys/net/netfilter/nf_conntrack_count'),
fs.trimmed('/proc/sys/net/netfilter/nf_conntrack_max'),
network.getWANNetworks(),
network.getWAN6Networks(),
callOdhcp6cStats(),
]);
},
render([ct_count, ct_max, wan_nets, wan6_nets, dhcpv6_stats]) {
const fields = [
{ label: _('Active Connections'), value: ct_max ? ct_count : null }
];
const ctstatus = E('table', { 'class': 'table' });
for (const { label, value } of fields) {
ctstatus.appendChild(E('tr', { 'class': 'tr' }, [
E('td', { 'class': 'td left', 'width': '33%' }, [ label ]),
E('td', { 'class': 'td left' }, [
(value != null) ? progressbar(value, ct_max) : '?'
])
]));
}
const netstatus = E('div', { 'class': 'network-status-table' });
for (const wan_net of wan_nets)
netstatus.appendChild(renderbox(wan_net, false));
for (const wan6_net of wan6_nets)
netstatus.appendChild(renderbox(wan6_net, true, dhcpv6_stats?.result));
return E([
netstatus,
ctstatus
]);
}
});
|