mirror of
https://git.proxmox.com/git/proxmox
synced 2025-05-02 08:27:14 +00:00
233 lines
6.4 KiB
Rust
233 lines
6.4 KiB
Rust
use std::collections::HashMap;
|
|
use std::os::unix::io::{AsRawFd, FromRawFd, OwnedFd};
|
|
use std::path::Path;
|
|
use std::process::Command;
|
|
|
|
use anyhow::{bail, format_err, Error};
|
|
use const_format::concatcp;
|
|
use lazy_static::lazy_static;
|
|
use nix::ioctl_read_bad;
|
|
use nix::sys::socket::{socket, AddressFamily, SockFlag, SockType};
|
|
use regex::Regex;
|
|
|
|
use proxmox_schema::api_types::IPV4RE_STR;
|
|
use proxmox_schema::api_types::IPV6RE_STR;
|
|
|
|
pub static IPV4_REVERSE_MASK: &[&str] = &[
|
|
"0.0.0.0",
|
|
"128.0.0.0",
|
|
"192.0.0.0",
|
|
"224.0.0.0",
|
|
"240.0.0.0",
|
|
"248.0.0.0",
|
|
"252.0.0.0",
|
|
"254.0.0.0",
|
|
"255.0.0.0",
|
|
"255.128.0.0",
|
|
"255.192.0.0",
|
|
"255.224.0.0",
|
|
"255.240.0.0",
|
|
"255.248.0.0",
|
|
"255.252.0.0",
|
|
"255.254.0.0",
|
|
"255.255.0.0",
|
|
"255.255.128.0",
|
|
"255.255.192.0",
|
|
"255.255.224.0",
|
|
"255.255.240.0",
|
|
"255.255.248.0",
|
|
"255.255.252.0",
|
|
"255.255.254.0",
|
|
"255.255.255.0",
|
|
"255.255.255.128",
|
|
"255.255.255.192",
|
|
"255.255.255.224",
|
|
"255.255.255.240",
|
|
"255.255.255.248",
|
|
"255.255.255.252",
|
|
"255.255.255.254",
|
|
"255.255.255.255",
|
|
];
|
|
|
|
lazy_static! {
|
|
pub static ref IPV4_MASK_HASH_LOCALNET: HashMap<&'static str, u8> = {
|
|
let mut map = HashMap::new();
|
|
#[allow(clippy::needless_range_loop)]
|
|
for i in 0..IPV4_REVERSE_MASK.len() {
|
|
map.insert(IPV4_REVERSE_MASK[i], i as u8);
|
|
}
|
|
map
|
|
};
|
|
}
|
|
|
|
pub fn parse_cidr(cidr: &str) -> Result<(String, u8, bool), Error> {
|
|
let (address, mask, is_v6) = parse_address_or_cidr(cidr)?;
|
|
if let Some(mask) = mask {
|
|
Ok((address, mask, is_v6))
|
|
} else {
|
|
bail!("missing netmask in '{}'", cidr);
|
|
}
|
|
}
|
|
|
|
pub(crate) fn check_netmask(mask: u8, is_v6: bool) -> Result<(), Error> {
|
|
let (ver, min, max) = if is_v6 {
|
|
("IPv6", 1, 128)
|
|
} else {
|
|
("IPv4", 1, 32)
|
|
};
|
|
|
|
if !(mask >= min && mask <= max) {
|
|
bail!(
|
|
"{} mask '{}' is out of range ({}..{}).",
|
|
ver,
|
|
mask,
|
|
min,
|
|
max
|
|
);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// parse ip address with optional cidr mask
|
|
pub(crate) fn parse_address_or_cidr(cidr: &str) -> Result<(String, Option<u8>, bool), Error> {
|
|
// NOTE: This is NOT the same regex as in proxmox-schema as this one has capture groups for
|
|
// the addresses vs cidr portions!
|
|
lazy_static! {
|
|
pub static ref CIDR_V4_REGEX: Regex =
|
|
Regex::new(concatcp!(r"^(", IPV4RE_STR, r")(?:/(\d{1,2}))?$")).unwrap();
|
|
pub static ref CIDR_V6_REGEX: Regex =
|
|
Regex::new(concatcp!(r"^(", IPV6RE_STR, r")(?:/(\d{1,3}))?$")).unwrap();
|
|
}
|
|
|
|
if let Some(caps) = CIDR_V4_REGEX.captures(cidr) {
|
|
let address = &caps[1];
|
|
if let Some(mask) = caps.get(2) {
|
|
let mask: u8 = mask.as_str().parse()?;
|
|
check_netmask(mask, false)?;
|
|
Ok((address.to_string(), Some(mask), false))
|
|
} else {
|
|
Ok((address.to_string(), None, false))
|
|
}
|
|
} else if let Some(caps) = CIDR_V6_REGEX.captures(cidr) {
|
|
let address = &caps[1];
|
|
if let Some(mask) = caps.get(2) {
|
|
let mask: u8 = mask.as_str().parse()?;
|
|
check_netmask(mask, true)?;
|
|
Ok((address.to_string(), Some(mask), true))
|
|
} else {
|
|
Ok((address.to_string(), None, true))
|
|
}
|
|
} else {
|
|
bail!("invalid address/mask '{}'", cidr);
|
|
}
|
|
}
|
|
|
|
pub(crate) fn get_network_interfaces() -> Result<HashMap<String, bool>, Error> {
|
|
const PROC_NET_DEV: &str = "/proc/net/dev";
|
|
|
|
#[repr(C)]
|
|
pub struct ifreq {
|
|
ifr_name: [libc::c_uchar; libc::IFNAMSIZ],
|
|
ifru_flags: libc::c_short,
|
|
}
|
|
|
|
ioctl_read_bad!(get_interface_flags, libc::SIOCGIFFLAGS, ifreq);
|
|
|
|
lazy_static! {
|
|
static ref IFACE_LINE_REGEX: Regex = Regex::new(r"^\s*([^:\s]+):").unwrap();
|
|
}
|
|
let raw = std::fs::read_to_string(PROC_NET_DEV)
|
|
.map_err(|err| format_err!("unable to read {} - {}", PROC_NET_DEV, err))?;
|
|
|
|
let lines = raw.lines();
|
|
|
|
let sock = unsafe {
|
|
OwnedFd::from_raw_fd(
|
|
socket(
|
|
AddressFamily::Inet,
|
|
SockType::Datagram,
|
|
SockFlag::empty(),
|
|
None,
|
|
)
|
|
.or_else(|_| {
|
|
socket(
|
|
AddressFamily::Inet6,
|
|
SockType::Datagram,
|
|
SockFlag::empty(),
|
|
None,
|
|
)
|
|
})?,
|
|
)
|
|
};
|
|
|
|
let mut interface_list = HashMap::new();
|
|
|
|
for line in lines {
|
|
if let Some(cap) = IFACE_LINE_REGEX.captures(line) {
|
|
let ifname = &cap[1];
|
|
|
|
let mut req = ifreq {
|
|
ifr_name: *b"0000000000000000",
|
|
ifru_flags: 0,
|
|
};
|
|
for (i, b) in std::ffi::CString::new(ifname)?
|
|
.as_bytes_with_nul()
|
|
.iter()
|
|
.enumerate()
|
|
{
|
|
if i < (libc::IFNAMSIZ - 1) {
|
|
req.ifr_name[i] = *b as libc::c_uchar;
|
|
}
|
|
}
|
|
let res = unsafe { get_interface_flags(sock.as_raw_fd(), &mut req)? };
|
|
if res != 0 {
|
|
bail!(
|
|
"ioctl get_interface_flags for '{}' failed ({})",
|
|
ifname,
|
|
res
|
|
);
|
|
}
|
|
let is_up = (req.ifru_flags & (libc::IFF_UP as libc::c_short)) != 0;
|
|
interface_list.insert(ifname.to_string(), is_up);
|
|
}
|
|
}
|
|
|
|
Ok(interface_list)
|
|
}
|
|
|
|
pub(crate) fn compute_file_diff(filename: &str, shadow: &str) -> Result<String, Error> {
|
|
let output = Command::new("diff")
|
|
.arg("-b")
|
|
.arg("-u")
|
|
.arg(filename)
|
|
.arg(shadow)
|
|
.output()
|
|
.map_err(|err| format_err!("failed to execute diff - {}", err))?;
|
|
|
|
let diff = proxmox_sys::command::command_output_as_string(output, Some(|c| c == 0 || c == 1))
|
|
.map_err(|err| format_err!("diff failed: {}", err))?;
|
|
|
|
Ok(diff)
|
|
}
|
|
|
|
pub fn assert_ifupdown2_installed() -> Result<(), Error> {
|
|
if !Path::new("/usr/share/ifupdown2").exists() {
|
|
bail!("ifupdown2 is not installed.");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn network_reload() -> Result<(), Error> {
|
|
let output = Command::new("ifreload")
|
|
.arg("-a")
|
|
.output()
|
|
.map_err(|err| format_err!("failed to execute 'ifreload' - {}", err))?;
|
|
|
|
proxmox_sys::command::command_output(output, None)
|
|
.map_err(|err| format_err!("ifreload failed: {}", err))?;
|
|
|
|
Ok(())
|
|
}
|