use anyhow::Context;
use netlink_packet_utils::{
nla::{DefaultNla, Nla, NlaBuffer},
traits::{Emitable, Parseable},
DecodeError,
};
use super::super::{LinkMessage, LinkMessageBuffer};
const VETH_INFO_PEER: u16 = 1;
#[derive(Debug, PartialEq, Eq, Clone)]
#[non_exhaustive]
pub enum InfoVeth {
Peer(LinkMessage),
Other(DefaultNla),
}
impl Nla for InfoVeth {
fn value_len(&self) -> usize {
use self::InfoVeth::*;
match *self {
Peer(ref message) => message.buffer_len(),
Other(ref attr) => attr.value_len(),
}
}
fn emit_value(&self, buffer: &mut [u8]) {
use self::InfoVeth::*;
match *self {
Peer(ref message) => message.emit(buffer),
Other(ref attr) => attr.emit_value(buffer),
}
}
fn kind(&self) -> u16 {
use self::InfoVeth::*;
match *self {
Peer(_) => VETH_INFO_PEER,
Other(ref attr) => attr.kind(),
}
}
}
impl<'a, T: AsRef<[u8]> + ?Sized> Parseable<NlaBuffer<&'a T>> for InfoVeth {
fn parse(buf: &NlaBuffer<&'a T>) -> Result<Self, DecodeError> {
use self::InfoVeth::*;
let payload = buf.value();
Ok(match buf.kind() {
VETH_INFO_PEER => {
let err = "failed to parse veth link info";
let buffer =
LinkMessageBuffer::new_checked(&payload).context(err)?;
Peer(LinkMessage::parse(&buffer).context(err)?)
}
kind => Other(
DefaultNla::parse(buf)
.context(format!("unknown NLA type {kind}"))?,
),
})
}
}