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
use futures::stream::StreamExt;
use std::net::{IpAddr, Ipv4Addr};
use netlink_packet_route::{
nlas::address::Nla,
AddressMessage,
NetlinkMessage,
RtnlMessage,
AF_INET,
AF_INET6,
NLM_F_ACK,
NLM_F_CREATE,
NLM_F_EXCL,
NLM_F_REPLACE,
NLM_F_REQUEST,
};
use crate::{try_nl, Error, Handle};
pub struct AddressAddRequest {
handle: Handle,
message: AddressMessage,
replace: bool,
}
impl AddressAddRequest {
pub(crate) fn new(handle: Handle, index: u32, address: IpAddr, prefix_len: u8) -> Self {
let mut message = AddressMessage::default();
message.header.prefix_len = prefix_len;
message.header.index = index;
let address_vec = match address {
IpAddr::V4(ipv4) => {
message.header.family = AF_INET as u8;
ipv4.octets().to_vec()
}
IpAddr::V6(ipv6) => {
message.header.family = AF_INET6 as u8;
ipv6.octets().to_vec()
}
};
if address.is_multicast() {
message.nlas.push(Nla::Multicast(address_vec));
} else if address.is_unspecified() {
message.nlas.push(Nla::Unspec(address_vec));
} else if address.is_ipv6() {
message.nlas.push(Nla::Address(address_vec));
} else {
message.nlas.push(Nla::Address(address_vec.clone()));
message.nlas.push(Nla::Local(address_vec.clone()));
if prefix_len == 32 {
message.nlas.push(Nla::Broadcast(address_vec));
} else {
let ip_addr: u32 = u32::from(Ipv4Addr::new(
address_vec[0],
address_vec[1],
address_vec[2],
address_vec[3],
));
let brd = Ipv4Addr::from((0xffff_ffff_u32) >> u32::from(prefix_len) | ip_addr);
message.nlas.push(Nla::Broadcast(brd.octets().to_vec()));
};
}
AddressAddRequest {
handle,
message,
replace: false,
}
}
pub fn replace(self) -> Self {
Self {
replace: true,
..self
}
}
pub async fn execute(self) -> Result<(), Error> {
let AddressAddRequest {
mut handle,
message,
replace,
} = self;
let mut req = NetlinkMessage::from(RtnlMessage::NewAddress(message));
let replace = if replace { NLM_F_REPLACE } else { NLM_F_EXCL };
req.header.flags = NLM_F_REQUEST | NLM_F_ACK | replace | NLM_F_CREATE;
let mut response = handle.request(req)?;
while let Some(message) = response.next().await {
try_nl!(message);
}
Ok(())
}
pub fn message_mut(&mut self) -> &mut AddressMessage {
&mut self.message
}
}