use std::collections::HashMap;
use std::fmt;
use std::net::IpAddr;
use crate::{NetworkDataInner, NetworksInner};
pub struct Networks {
pub(crate) inner: NetworksInner,
}
impl<'a> IntoIterator for &'a Networks {
type Item = (&'a String, &'a NetworkData);
type IntoIter = std::collections::hash_map::Iter<'a, String, NetworkData>;
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
impl Default for Networks {
fn default() -> Self {
Networks::new()
}
}
impl Networks {
pub fn new() -> Self {
Self {
inner: NetworksInner::new(),
}
}
pub fn new_with_refreshed_list() -> Self {
let mut networks = Self::new();
networks.refresh_list();
networks
}
pub fn list(&self) -> &HashMap<String, NetworkData> {
self.inner.list()
}
pub fn refresh_list(&mut self) {
self.inner.refresh_list()
}
pub fn refresh(&mut self) {
self.inner.refresh()
}
}
impl std::ops::Deref for Networks {
type Target = HashMap<String, NetworkData>;
fn deref(&self) -> &Self::Target {
self.list()
}
}
pub struct NetworkData {
pub(crate) inner: NetworkDataInner,
}
impl NetworkData {
pub fn received(&self) -> u64 {
self.inner.received()
}
pub fn total_received(&self) -> u64 {
self.inner.total_received()
}
pub fn transmitted(&self) -> u64 {
self.inner.transmitted()
}
pub fn total_transmitted(&self) -> u64 {
self.inner.total_transmitted()
}
pub fn packets_received(&self) -> u64 {
self.inner.packets_received()
}
pub fn total_packets_received(&self) -> u64 {
self.inner.total_packets_received()
}
pub fn packets_transmitted(&self) -> u64 {
self.inner.packets_transmitted()
}
pub fn total_packets_transmitted(&self) -> u64 {
self.inner.total_packets_transmitted()
}
pub fn errors_on_received(&self) -> u64 {
self.inner.errors_on_received()
}
pub fn total_errors_on_received(&self) -> u64 {
self.inner.total_errors_on_received()
}
pub fn errors_on_transmitted(&self) -> u64 {
self.inner.errors_on_transmitted()
}
pub fn total_errors_on_transmitted(&self) -> u64 {
self.inner.total_errors_on_transmitted()
}
pub fn mac_address(&self) -> MacAddr {
self.inner.mac_address()
}
pub fn ip_networks(&self) -> &[IpNetwork] {
self.inner.ip_networks()
}
}
#[derive(PartialEq, Eq, Copy, Clone, Debug)]
pub struct MacAddr(pub [u8; 6]);
impl MacAddr {
pub const UNSPECIFIED: Self = MacAddr([0; 6]);
pub fn is_unspecified(&self) -> bool {
self == &MacAddr::UNSPECIFIED
}
}
impl fmt::Display for MacAddr {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let data = &self.0;
write!(
f,
"{:02x}:{:02x}:{:02x}:{:02x}:{:02x}:{:02x}",
data[0], data[1], data[2], data[3], data[4], data[5],
)
}
}
#[derive(Debug, Clone, Copy, Hash, PartialEq, Eq, PartialOrd, Ord)]
pub struct IpNetwork {
pub addr: IpAddr,
pub prefix: u8,
}
impl fmt::Display for IpNetwork {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}/{}", self.addr, self.prefix)
}
}
#[cfg(test)]
mod tests {
use crate::*;
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
#[test]
fn check_display_impl_mac_address() {
println!(
"{} {:?}",
MacAddr([0x1, 0x2, 0x3, 0x4, 0x5, 0x6]),
MacAddr([0xa, 0xb, 0xc, 0xd, 0xe, 0xf])
);
}
#[test]
fn check_mac_address_is_unspecified_true() {
assert!(MacAddr::UNSPECIFIED.is_unspecified());
assert!(MacAddr([0; 6]).is_unspecified());
}
#[test]
fn check_mac_address_is_unspecified_false() {
assert!(!MacAddr([1, 2, 3, 4, 5, 6]).is_unspecified());
}
#[test]
fn check_display_impl_ip_network_ipv4() {
println!(
"{} {:?}",
IpNetwork {
addr: IpAddr::from(Ipv4Addr::new(1, 2, 3, 4)),
prefix: 3
},
IpNetwork {
addr: IpAddr::from(Ipv4Addr::new(255, 255, 255, 0)),
prefix: 21
}
);
}
#[test]
fn check_display_impl_ip_network_ipv6() {
println!(
"{} {:?}",
IpNetwork {
addr: IpAddr::from(Ipv6Addr::new(0xffff, 0xaabb, 00, 0, 0, 0x000c, 11, 21)),
prefix: 127
},
IpNetwork {
addr: IpAddr::from(Ipv6Addr::new(0xffcc, 0, 0, 0xffcc, 0, 0xffff, 0, 0xccaa)),
prefix: 120
}
)
}
#[test]
fn check_ip_networks() {
if !IS_SUPPORTED_SYSTEM {
return;
}
let networks = Networks::new_with_refreshed_list();
if networks.iter().any(|(_, n)| !n.ip_networks().is_empty()) {
return;
}
panic!("Networks should have at least one IP network ");
}
}