add ethernet information looks like openwrt 23.05 (#340)

* add ethernet information in luci overview

* add ethernet information looks like openwrt 23.05
This commit is contained in:
wsk170 2024-11-18 09:25:43 +08:00 committed by GitHub
parent fa0b7600f5
commit 0e2fe86ae5
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 136 additions and 45 deletions

View File

@ -48,7 +48,7 @@ define Package/autocore/install/Default
$(INSTALL_DIR) $(1)/usr/share/rpcd/acl.d $(INSTALL_DIR) $(1)/usr/share/rpcd/acl.d
$(CP) ./files/generic/luci-mod-status-autocore.json $(1)/usr/share/rpcd/acl.d/ $(CP) ./files/generic/luci-mod-status-autocore.json $(1)/usr/share/rpcd/acl.d/
ifeq ($(filter ipq% mediatek%, $(TARGETID)),) ifeq ($(filter ipq%, $(TARGETID)),)
$(INSTALL_BIN) ./files/generic/ethinfo $(1)/sbin/ $(INSTALL_BIN) ./files/generic/ethinfo $(1)/sbin/
$(INSTALL_DIR) $(1)/www/luci-static/resources/view/status/include $(INSTALL_DIR) $(1)/www/luci-static/resources/view/status/include

View File

@ -1,60 +1,151 @@
'use strict'; 'use strict';
'require baseclass'; 'require baseclass';
'require rpc'; 'require rpc';
'require uci';
'require network';
var callLuciETHInfo = rpc.declare({ var callSwconfigFeatures = rpc.declare({
object: 'luci', object: 'luci',
method: 'getETHInfo', method: 'getSwconfigFeatures',
params: ['switch'],
expect: { '': {} } expect: { '': {} }
}); });
var callSwconfigPortState = rpc.declare({
object: 'luci',
method: 'getSwconfigPortState',
params: ['switch'],
expect: { result: [] }
});
var callLuciBoardJSON = rpc.declare({
object: 'luci-rpc',
method: 'getBoardJSON',
expect: { '': {} }
});
var callLuciNetworkDevices = rpc.declare({
object: 'luci-rpc',
method: 'getNetworkDevices',
expect: { '': {} }
});
function formatSpeed(speed) {
if (!speed) return '-';
return speed < 1000 ? `${speed} M` : `${speed / 1000} GbE`;
}
function getPortColor(link, duplex) {
if (!link) return 'background-color: whitesmoke;';
const color = duplex == 'full' || duplex ? 'greenyellow' : 'darkorange';
return 'background-color: ' + color;
}
function getPortIcon(link) {
return L.resource(`icons/port_${link ? 'up' : 'down'}.png`);
}
return L.Class.extend({ return L.Class.extend({
title: _('Ethernet Information'), title: _('Ethernet Information'),
load: function() { load: function () {
return network.getSwitchTopologies().then(function (topologies) {
let tasks = [];
for (let switchName in topologies) {
tasks.push(
callSwconfigFeatures(switchName).then(
L.bind(function (features) {
this.features = features;
}, topologies[switchName])
)
);
tasks.push(
callSwconfigPortState(switchName).then(
L.bind(function (ports) {
this.portstate = ports;
}, topologies[switchName])
)
);
}
return Promise.all([ return Promise.all([
L.resolveDefault(callLuciETHInfo(), {}) topologies,
L.resolveDefault(callLuciBoardJSON(), {}),
L.resolveDefault(callLuciNetworkDevices(), {})
]); ]);
});
}, },
render: function(data) { render: function (data) {
var ethinfo = Array.isArray(data[0].ethinfo) ? data[0].ethinfo : []; const topologies = data[0];
const board = data[1];
const netdevs = data[2];
var table = E('table', { 'class': 'table' }, [ const boxStyle = 'max-width: 100px;';
E('tr', { 'class': 'tr table-titles' }, [ const boxHeadStyle =
E('th', { 'class': 'th' }, _('Ethernet Name')), 'border-radius: 7px 7px 0 0;' +
E('th', { 'class': 'th' }, _('Link Status')), 'text-align: center;' +
E('th', { 'class': 'th' }, _('Speed')), 'font-weight:bold;';
E('th', { 'class': 'th' }, _('Duplex')) const boxbodyStyle =
'border: 1px solid lightgrey;' +
'border-radius: 0 0 7px 7px;' +
'display:flex; flex-direction: column;' +
'align-items: center; justify-content:center;';
const iconStyle = 'margin: 5px; width: 40px;';
const speedStyle = 'font-size:0.8rem; font-weight:bold;';
const trafficStyle =
'border-top: 1px solid lightgrey;' + 'font-size:0.8rem;';
const ethPorts = [];
const wan = netdevs[board.network.wan.device];
const { speed, duplex, carrier } = wan.link;
let portIcon = getPortIcon(carrier);
let portColor = getPortColor(carrier, duplex);
ethPorts.push(
E('div', { style: boxStyle }, [
E('div', { style: boxHeadStyle + portColor }, 'WAN'),
E('div', { style: boxbodyStyle }, [
E('img', { style: iconStyle, src: portIcon }),
E('div', { style: speedStyle }, formatSpeed(speed)),
E('div', { style: trafficStyle }, [
'\u25b2\u202f%1024.1mB'.format(wan.stats.tx_bytes),
E('br'),
'\u25bc\u202f%1024.1mB'.format(wan.stats.rx_bytes)
]) ])
]); ])
])
);
cbi_update_table(table, ethinfo.map(function(info) { const switch0 = topologies.switch0;
var exp1; for (const port of switch0.ports) {
var exp2; if (!port.label.startsWith('LAN')) continue;
const { link, duplex, speed } = switch0.portstate[port.num];
portIcon = getPortIcon(link);
portColor = getPortColor(link, duplex);
const txrx = { tx_bytes: 0, rx_bytes: 0 };
const lanStats = netdevs['br-lan'].stats;
const { tx_bytes, rx_bytes } = link ? lanStats : txrx;
ethPorts.push(
E('div', { style: boxStyle }, [
E('div', { style: boxHeadStyle + portColor }, port.label),
E('div', { style: boxbodyStyle }, [
E('img', { style: iconStyle, src: portIcon }),
E('div', { style: speedStyle }, formatSpeed(speed)),
E('div', { style: trafficStyle }, [
'\u25b2\u202f%1024.1mB'.format(tx_bytes),
E('br'),
'\u25bc\u202f%1024.1mB'.format(rx_bytes)
])
])
])
);
}
if (info.status == "yes") const gridStyle =
exp1 = _('Link Up'); 'display:grid; grid-gap: 5px 5px;' +
else if (info.status == "no") 'grid-template-columns:repeat(auto-fit, minmax(70px, 1fr));' +
exp1 = _('Link Down'); 'margin-bottom:1em';
return E('div', { style: gridStyle }, ethPorts);
if (info.duplex == "Full")
exp2 = _('Full Duplex');
else if (info.duplex == "Half")
exp2 = _('Half Duplex');
else
exp2 = _('-');
return [
info.name,
exp1,
info.speed,
exp2
];
}));
return E([
table
]);
} }
}); });