318 lines
10 KiB
Rust
318 lines
10 KiB
Rust
|
use std::io::{BufReader};
|
||
|
use std::fs::File;
|
||
|
use std::iter::{Peekable, Iterator};
|
||
|
|
||
|
use anyhow::{Error, bail, format_err};
|
||
|
use lazy_static::lazy_static;
|
||
|
use regex::Regex;
|
||
|
|
||
|
use proxmox::*; // for IP macros
|
||
|
|
||
|
use super::helper::*;
|
||
|
use super::lexer::*;
|
||
|
|
||
|
use super::{NetworkConfig, NetworkOrderEntry, Interface, AddressFamily};
|
||
|
|
||
|
pub struct NetworkParser {
|
||
|
input: Peekable<Lexer<BufReader<File>>>,
|
||
|
line_nr: usize,
|
||
|
}
|
||
|
|
||
|
impl NetworkParser {
|
||
|
|
||
|
pub fn new(file: File) -> Self {
|
||
|
let reader = BufReader::new(file);
|
||
|
let input = Lexer::new(reader).peekable();
|
||
|
Self { input, line_nr: 1 }
|
||
|
}
|
||
|
|
||
|
fn peek(&mut self) -> Result<Token, Error> {
|
||
|
match self.input.peek() {
|
||
|
Some(Err(err)) => {
|
||
|
bail!("input error - {}", err);
|
||
|
}
|
||
|
Some(Ok((token, _))) => {
|
||
|
return Ok(*token);
|
||
|
}
|
||
|
None => {
|
||
|
bail!("got unexpected end of stream (inside peek)");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn next(&mut self) -> Result<(Token, String), Error> {
|
||
|
match self.input.next() {
|
||
|
Some(Err(err)) => {
|
||
|
bail!("input error - {}", err);
|
||
|
}
|
||
|
Some(Ok((token, text))) => {
|
||
|
if token == Token::Newline { self.line_nr += 1; }
|
||
|
return Ok((token, text));
|
||
|
}
|
||
|
None => {
|
||
|
bail!("got unexpected end of stream (inside peek)");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn next_text(&mut self) -> Result<String, Error> {
|
||
|
match self.next()? {
|
||
|
(Token::Text, text) => Ok(text),
|
||
|
(unexpected, _) => bail!("got unexpected token {:?} (expecting Text)", unexpected),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn eat(&mut self, expected: Token) -> Result<(), Error> {
|
||
|
let (next, _) = self.next()?;
|
||
|
if next != expected {
|
||
|
bail!("expected {:?}, got {:?}", expected, next);
|
||
|
}
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_auto(&mut self, config: &mut NetworkConfig) -> Result<(), Error> {
|
||
|
self.eat(Token::Auto)?;
|
||
|
|
||
|
let iface = self.next_text()?;
|
||
|
println!("AUTO {}", iface);
|
||
|
|
||
|
self.eat(Token::Newline)?;
|
||
|
|
||
|
config.auto_flag.insert(iface.to_string());
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_iface_address(&mut self, interface: &mut Interface) -> Result<(), Error> {
|
||
|
self.eat(Token::Address)?;
|
||
|
let address = self.next_text()?;
|
||
|
|
||
|
lazy_static! {
|
||
|
pub static ref ADDRESS_V4_REGEX: Regex = Regex::new(
|
||
|
concat!(r"^(", IPV4RE!(), r")(?:/(\d{1,2}))?$")
|
||
|
).unwrap();
|
||
|
pub static ref ADDRESS_V6_REGEX: Regex = Regex::new(
|
||
|
concat!(r"^(", IPV6RE!(), r")(?:/(\d{1,2}))?$")
|
||
|
).unwrap();
|
||
|
}
|
||
|
|
||
|
if let Some(caps) = ADDRESS_V4_REGEX.captures(&address) {
|
||
|
let address = caps.get(1).unwrap().as_str();
|
||
|
interface.set_address_v4(address.to_string())?;
|
||
|
if let Some(mask) = caps.get(2) {
|
||
|
let mask = u8::from_str_radix(mask.as_str(), 10)?;
|
||
|
interface.set_netmask_v4(mask)?;
|
||
|
}
|
||
|
} else if let Some(caps) = ADDRESS_V6_REGEX.captures(&address) {
|
||
|
let address = caps.get(1).unwrap().as_str();
|
||
|
interface.set_address_v6(address.to_string())?;
|
||
|
if let Some(mask) = caps.get(2) {
|
||
|
let mask = u8::from_str_radix(mask.as_str(), 10)?;
|
||
|
interface.set_netmask_v6(mask)?;
|
||
|
}
|
||
|
} else {
|
||
|
bail!("unable to parse IP address");
|
||
|
}
|
||
|
|
||
|
self.eat(Token::Newline)?;
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_iface_gateway(&mut self, interface: &mut Interface) -> Result<(), Error> {
|
||
|
self.eat(Token::Gateway)?;
|
||
|
let gateway = self.next_text()?;
|
||
|
|
||
|
if proxmox::tools::common_regex::IP_REGEX.is_match(&gateway) {
|
||
|
if gateway.contains(':') {
|
||
|
interface.set_gateway_v6(gateway)?;
|
||
|
} else {
|
||
|
interface.set_gateway_v4(gateway)?;
|
||
|
}
|
||
|
} else {
|
||
|
bail!("unable to parse gateway address");
|
||
|
}
|
||
|
|
||
|
self.eat(Token::Newline)?;
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_to_eol(&mut self) -> Result<String, Error> {
|
||
|
let mut line = String::new();
|
||
|
loop {
|
||
|
match self.next()? {
|
||
|
(Token::Newline, _) => return Ok(line),
|
||
|
(_, text) => {
|
||
|
if !line.is_empty() { line.push(' '); }
|
||
|
line.push_str(&text);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn parse_iface_addon_attribute(&mut self, interface: &mut Interface) -> Result<(), Error> {
|
||
|
let option = self.parse_to_eol()?;
|
||
|
if !option.is_empty() { interface.push_addon_option(option) };
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_iface_netmask(&mut self, interface: &mut Interface) -> Result<(), Error> {
|
||
|
self.eat(Token::Netmask)?;
|
||
|
let netmask = self.next_text()?;
|
||
|
|
||
|
if let Some(mask) = IPV4_MASK_HASH_LOCALNET.get(netmask.as_str()) {
|
||
|
interface.set_netmask_v4(*mask)?;
|
||
|
} else {
|
||
|
match u8::from_str_radix(netmask.as_str(), 10) {
|
||
|
Ok(mask) => {
|
||
|
if mask <= 32 { interface.set_netmask_v4(mask)?; }
|
||
|
interface.set_netmask_v6(mask)?;
|
||
|
}
|
||
|
Err(err) => {
|
||
|
bail!("unable to parse netmask '{}' - {}", netmask, err);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
self.eat(Token::Newline)?;
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_iface_attributes(&mut self, interface: &mut Interface) -> Result<(), Error> {
|
||
|
|
||
|
loop {
|
||
|
match self.peek()? {
|
||
|
Token::Attribute => self.eat(Token::Attribute)?,
|
||
|
Token::Newline => break,
|
||
|
unexpected => bail!("unknown token {:?}", unexpected),
|
||
|
}
|
||
|
|
||
|
match self.peek()? {
|
||
|
Token::Address => self.parse_iface_address(interface)?,
|
||
|
Token::Gateway => self.parse_iface_gateway(interface)?,
|
||
|
Token::Netmask => self.parse_iface_netmask(interface)?,
|
||
|
_ => {
|
||
|
self.parse_iface_addon_attribute(interface)?;
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
fn parse_iface(&mut self, config: &mut NetworkConfig) -> Result<(), Error> {
|
||
|
self.eat(Token::Iface)?;
|
||
|
let iface = self.next_text()?;
|
||
|
|
||
|
let mut address_family = None;
|
||
|
let mut config_method = None;
|
||
|
|
||
|
loop {
|
||
|
let (token, text) = self.next()?;
|
||
|
match token {
|
||
|
Token::Newline => break,
|
||
|
Token::Inet => {
|
||
|
address_family = Some(match address_family {
|
||
|
None => AddressFamily::Inet,
|
||
|
Some(AddressFamily::Inet) => AddressFamily::Inet,
|
||
|
Some(AddressFamily::Inet6) => AddressFamily::Inet4and6,
|
||
|
Some(AddressFamily::Inet4and6) => AddressFamily::Inet4and6,
|
||
|
});
|
||
|
}
|
||
|
Token::Inet6 => {
|
||
|
address_family = Some(match address_family {
|
||
|
None => AddressFamily::Inet6,
|
||
|
Some(AddressFamily::Inet) => AddressFamily::Inet4and6,
|
||
|
Some(AddressFamily::Inet6) => AddressFamily::Inet6,
|
||
|
Some(AddressFamily::Inet4and6) => AddressFamily::Inet4and6,
|
||
|
});
|
||
|
}
|
||
|
Token::Loopback | Token::Static | Token::Manual | Token::DHCP => {
|
||
|
if config_method.is_none() {
|
||
|
config_method = Some(token);
|
||
|
} else {
|
||
|
bail!("multiple configuration method definitions");
|
||
|
}
|
||
|
}
|
||
|
_ => bail!("unknown iface option {}", text),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
let address_family = address_family.unwrap_or(AddressFamily::Inet4and6);
|
||
|
|
||
|
let has_attributes = self.peek()? == Token::Attribute;
|
||
|
|
||
|
if let Some(mut interface) = config.interfaces.get_mut(&iface) {
|
||
|
let compatible = match interface.address_family {
|
||
|
AddressFamily::Inet => {
|
||
|
interface.method_v6 = config_method;
|
||
|
address_family == AddressFamily::Inet6
|
||
|
}
|
||
|
AddressFamily::Inet6 => {
|
||
|
interface.method_v4 = config_method;
|
||
|
address_family == AddressFamily::Inet
|
||
|
}
|
||
|
_ => false,
|
||
|
};
|
||
|
if !compatible {
|
||
|
bail!("duplicate config for iface '{}'", iface);
|
||
|
}
|
||
|
interface.address_family = AddressFamily::Inet4and6;
|
||
|
if has_attributes { self.parse_iface_attributes(&mut interface)?; }
|
||
|
} else {
|
||
|
let mut interface = Interface::new(iface.clone(), address_family, config_method);
|
||
|
|
||
|
if has_attributes { self.parse_iface_attributes(&mut interface)?; }
|
||
|
|
||
|
config.interfaces.insert(interface.name.clone(), interface);
|
||
|
config.order.push(NetworkOrderEntry::Iface(iface));
|
||
|
}
|
||
|
|
||
|
Ok(())
|
||
|
}
|
||
|
|
||
|
pub fn parse_interfaces(&mut self) -> Result<NetworkConfig, Error> {
|
||
|
self._parse_interfaces()
|
||
|
.map_err(|err| format_err!("line {}: {}", self.line_nr, err))
|
||
|
}
|
||
|
|
||
|
pub fn _parse_interfaces(&mut self) -> Result<NetworkConfig, Error> {
|
||
|
let mut config = NetworkConfig::new();
|
||
|
|
||
|
loop {
|
||
|
let peek = self.peek()?;
|
||
|
println!("TOKEN: {:?}", peek);
|
||
|
match peek {
|
||
|
Token::EOF => {
|
||
|
// fixme: trailing comments
|
||
|
return Ok(config);
|
||
|
}
|
||
|
Token::Newline => {
|
||
|
self.eat(Token::Newline)?;
|
||
|
// fixme end of entry
|
||
|
}
|
||
|
Token::Comment => {
|
||
|
let (_, text) = self.next()?;
|
||
|
println!("COMMENT: {}", text);
|
||
|
config.order.push(NetworkOrderEntry::Comment(text));
|
||
|
self.eat(Token::Newline)?;
|
||
|
}
|
||
|
Token::Auto => {
|
||
|
self.parse_auto(&mut config)?;
|
||
|
}
|
||
|
Token::Iface => {
|
||
|
self.parse_iface(&mut config)?;
|
||
|
}
|
||
|
_ => {
|
||
|
let option = self.parse_to_eol()?;
|
||
|
if !option.is_empty() {
|
||
|
config.order.push(NetworkOrderEntry::Option(option));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|