split host & switch config
This commit is contained in:
parent
61b02e7a28
commit
fe0501b784
@ -28,6 +28,7 @@ use super::PreparationOutcome;
|
|||||||
use super::Router;
|
use super::Router;
|
||||||
use super::Switch;
|
use super::Switch;
|
||||||
use super::SwitchError;
|
use super::SwitchError;
|
||||||
|
use super::SwitchNetworkConfig;
|
||||||
use super::TftpServer;
|
use super::TftpServer;
|
||||||
|
|
||||||
use super::Topology;
|
use super::Topology;
|
||||||
@ -280,6 +281,14 @@ impl Switch for HAClusterTopology {
|
|||||||
) -> Result<(), SwitchError> {
|
) -> Result<(), SwitchError> {
|
||||||
todo!()
|
todo!()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn configure_switch_network(
|
||||||
|
&self,
|
||||||
|
_host: &PhysicalHost,
|
||||||
|
_config: SwitchNetworkConfig,
|
||||||
|
) -> Result<(), SwitchError> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
@ -179,8 +179,14 @@ pub trait Switch: Send + Sync {
|
|||||||
|
|
||||||
async fn configure_host_network(
|
async fn configure_host_network(
|
||||||
&self,
|
&self,
|
||||||
_host: &PhysicalHost,
|
host: &PhysicalHost,
|
||||||
_config: HostNetworkConfig,
|
config: HostNetworkConfig,
|
||||||
|
) -> Result<(), SwitchError>;
|
||||||
|
|
||||||
|
async fn configure_switch_network(
|
||||||
|
&self,
|
||||||
|
host: &PhysicalHost,
|
||||||
|
config: SwitchNetworkConfig,
|
||||||
) -> Result<(), SwitchError>;
|
) -> Result<(), SwitchError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -200,6 +206,16 @@ pub struct SlaveInterface {
|
|||||||
// FIXME: Should we add speed as well? And other params
|
// FIXME: Should we add speed as well? And other params
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
|
pub struct SwitchNetworkConfig {
|
||||||
|
pub port_channel: PortChannel,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
|
pub struct PortChannel {
|
||||||
|
pub ports: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, new)]
|
#[derive(Debug, Clone, new)]
|
||||||
pub struct SwitchError {
|
pub struct SwitchError {
|
||||||
msg: String,
|
msg: String,
|
||||||
|
@ -8,7 +8,10 @@ use crate::{
|
|||||||
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
|
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
|
||||||
inventory::Inventory,
|
inventory::Inventory,
|
||||||
score::Score,
|
score::Score,
|
||||||
topology::{self, Bond, HostNetworkConfig, SlaveInterface, Switch, Topology},
|
topology::{
|
||||||
|
self, Bond, HostNetworkConfig, PortChannel, SlaveInterface, Switch, SwitchNetworkConfig,
|
||||||
|
Topology,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize)]
|
#[derive(Debug, Clone, Serialize)]
|
||||||
@ -56,20 +59,34 @@ impl<T: Topology + Switch> Interpret<T> for HostNetworkConfigurationInterpret {
|
|||||||
_inventory: &Inventory,
|
_inventory: &Inventory,
|
||||||
topology: &T,
|
topology: &T,
|
||||||
) -> Result<Outcome, InterpretError> {
|
) -> Result<Outcome, InterpretError> {
|
||||||
let host = self.score.hosts.first().unwrap();
|
for host in &self.score.hosts {
|
||||||
let mac_addresses = host.get_mac_address();
|
let mut interfaces = vec![];
|
||||||
let mac_address = mac_addresses.first().unwrap();
|
let mut ports = vec![];
|
||||||
let host_network_config = HostNetworkConfig {
|
|
||||||
bond: Bond {
|
|
||||||
interfaces: vec![SlaveInterface {
|
|
||||||
mac_address: *mac_address,
|
|
||||||
}],
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
let _ = topology
|
for mac_address in host.get_mac_address() {
|
||||||
.configure_host_network(host, host_network_config)
|
if let Some(port) = topology.get_port_for_mac_address(&mac_address).await {
|
||||||
.await;
|
interfaces.push(SlaveInterface { mac_address });
|
||||||
|
ports.push(port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let _ = topology
|
||||||
|
.configure_host_network(
|
||||||
|
host,
|
||||||
|
HostNetworkConfig {
|
||||||
|
bond: Bond { interfaces },
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
let _ = topology
|
||||||
|
.configure_switch_network(
|
||||||
|
host,
|
||||||
|
SwitchNetworkConfig {
|
||||||
|
port_channel: PortChannel { ports },
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
}
|
||||||
|
|
||||||
// foreach hosts
|
// foreach hosts
|
||||||
// foreach mac addresses
|
// foreach mac addresses
|
||||||
@ -78,17 +95,12 @@ impl<T: Topology + Switch> Interpret<T> for HostNetworkConfigurationInterpret {
|
|||||||
// create bond for all valid addresses (port found)
|
// create bond for all valid addresses (port found)
|
||||||
// apply network to host first, then switch (to avoid losing hosts that are already connected)
|
// apply network to host first, then switch (to avoid losing hosts that are already connected)
|
||||||
// topology.configure_host_network(host, config) <--- will create bonds
|
// topology.configure_host_network(host, config) <--- will create bonds
|
||||||
// topology.configure_switch_network(port, config) <--- will create port channels
|
// topology.configure_switch_network(host, config) <--- will create port channels
|
||||||
|
|
||||||
Ok(Outcome::success("".into()))
|
Ok(Outcome::success("".into()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct PortMapping {
|
|
||||||
port: String,
|
|
||||||
mac_address: MacAddress,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use assertor::*;
|
use assertor::*;
|
||||||
@ -98,8 +110,8 @@ mod tests {
|
|||||||
use crate::{
|
use crate::{
|
||||||
hardware::HostCategory,
|
hardware::HostCategory,
|
||||||
topology::{
|
topology::{
|
||||||
Bond, HostNetworkConfig, PreparationError, PreparationOutcome, SlaveInterface,
|
Bond, HostNetworkConfig, PortChannel, PreparationError, PreparationOutcome,
|
||||||
SwitchError,
|
SlaveInterface, SwitchError, SwitchNetworkConfig,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
use std::{
|
use std::{
|
||||||
@ -111,15 +123,22 @@ mod tests {
|
|||||||
|
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
pub static ref HOST_ID: Id = Id::from_str("host-1").unwrap();
|
pub static ref HOST_ID: Id = Id::from_str("host-1").unwrap();
|
||||||
pub static ref INTERFACE: MacAddress =
|
pub static ref ANOTHER_HOST_ID: Id = Id::from_str("host-2").unwrap();
|
||||||
MacAddress::try_from("00:11:22:33:44:55".to_string()).unwrap();
|
pub static ref EXISTING_INTERFACE: MacAddress =
|
||||||
|
MacAddress::try_from("00:00:00:00:00:00".to_string()).unwrap();
|
||||||
|
pub static ref ANOTHER_EXISTING_INTERFACE: MacAddress =
|
||||||
|
MacAddress::try_from("42:42:42:42:42:42".to_string()).unwrap();
|
||||||
|
pub static ref UNKNOWN_INTERFACE: MacAddress =
|
||||||
|
MacAddress::try_from("99:99:99:99:99:99".to_string()).unwrap();
|
||||||
|
pub static ref PORT: String = "1/0/42".into();
|
||||||
|
pub static ref ANOTHER_PORT: String = "2/0/42".into();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
async fn one_host_one_mac_address_should_create_bond_with_one_interface() {
|
async fn host_with_one_mac_address_should_create_bond_with_one_interface() {
|
||||||
let host = given_host(&HOST_ID, *INTERFACE);
|
let host = given_host(&HOST_ID, vec![*EXISTING_INTERFACE]);
|
||||||
let score = given_score(vec![host]);
|
let score = given_score(vec![host]);
|
||||||
let topology = SwitchWithPortTopology::new();
|
let topology = TopologyWithSwitch::new();
|
||||||
|
|
||||||
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
@ -129,28 +148,161 @@ mod tests {
|
|||||||
HostNetworkConfig {
|
HostNetworkConfig {
|
||||||
bond: Bond {
|
bond: Bond {
|
||||||
interfaces: vec![SlaveInterface {
|
interfaces: vec![SlaveInterface {
|
||||||
mac_address: *INTERFACE,
|
mac_address: *EXISTING_INTERFACE,
|
||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)]);
|
)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn given_host(id: &Id, mac_address: MacAddress) -> PhysicalHost {
|
#[tokio::test]
|
||||||
|
async fn host_with_one_mac_address_should_create_port_channel_with_one_port() {
|
||||||
|
let host = given_host(&HOST_ID, vec![*EXISTING_INTERFACE]);
|
||||||
|
let score = given_score(vec![host]);
|
||||||
|
let topology = TopologyWithSwitch::new();
|
||||||
|
|
||||||
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
|
let configured_switch_networks = topology.configured_switch_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_switch_networks).contains_exactly(vec![(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
SwitchNetworkConfig {
|
||||||
|
port_channel: PortChannel {
|
||||||
|
ports: vec![PORT.clone()],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn host_with_multiple_mac_addresses_should_create_one_bond_with_all_interfaces() {
|
||||||
|
let score = given_score(vec![given_host(
|
||||||
|
&HOST_ID,
|
||||||
|
vec![*EXISTING_INTERFACE, *ANOTHER_EXISTING_INTERFACE],
|
||||||
|
)]);
|
||||||
|
let topology = TopologyWithSwitch::new();
|
||||||
|
|
||||||
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
|
let configured_host_networks = topology.configured_host_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_host_networks).contains_exactly(vec![(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
HostNetworkConfig {
|
||||||
|
bond: Bond {
|
||||||
|
interfaces: vec![
|
||||||
|
SlaveInterface {
|
||||||
|
mac_address: *EXISTING_INTERFACE,
|
||||||
|
},
|
||||||
|
SlaveInterface {
|
||||||
|
mac_address: *ANOTHER_EXISTING_INTERFACE,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn multiple_hosts_should_create_one_bond_per_host() {
|
||||||
|
let score = given_score(vec![
|
||||||
|
given_host(&HOST_ID, vec![*EXISTING_INTERFACE]),
|
||||||
|
given_host(&ANOTHER_HOST_ID, vec![*ANOTHER_EXISTING_INTERFACE]),
|
||||||
|
]);
|
||||||
|
let topology = TopologyWithSwitch::new();
|
||||||
|
|
||||||
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
|
let configured_host_networks = topology.configured_host_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_host_networks).contains_exactly(vec![
|
||||||
|
(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
HostNetworkConfig {
|
||||||
|
bond: Bond {
|
||||||
|
interfaces: vec![SlaveInterface {
|
||||||
|
mac_address: *EXISTING_INTERFACE,
|
||||||
|
}],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
ANOTHER_HOST_ID.clone(),
|
||||||
|
HostNetworkConfig {
|
||||||
|
bond: Bond {
|
||||||
|
interfaces: vec![SlaveInterface {
|
||||||
|
mac_address: *ANOTHER_EXISTING_INTERFACE,
|
||||||
|
}],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn multiple_hosts_should_create_one_port_channel_per_host() {
|
||||||
|
let score = given_score(vec![
|
||||||
|
given_host(&HOST_ID, vec![*EXISTING_INTERFACE]),
|
||||||
|
given_host(&ANOTHER_HOST_ID, vec![*ANOTHER_EXISTING_INTERFACE]),
|
||||||
|
]);
|
||||||
|
let topology = TopologyWithSwitch::new();
|
||||||
|
|
||||||
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
|
let configured_switch_networks = topology.configured_switch_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_switch_networks).contains_exactly(vec![
|
||||||
|
(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
SwitchNetworkConfig {
|
||||||
|
port_channel: PortChannel {
|
||||||
|
ports: vec![PORT.clone()],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
ANOTHER_HOST_ID.clone(),
|
||||||
|
SwitchNetworkConfig {
|
||||||
|
port_channel: PortChannel {
|
||||||
|
ports: vec![ANOTHER_PORT.clone()],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn port_not_found_for_mac_address_should_not_configure_interface() {
|
||||||
|
// FIXME: Should it still configure an empty bond/port channel?
|
||||||
|
let score = given_score(vec![given_host(&HOST_ID, vec![*UNKNOWN_INTERFACE])]);
|
||||||
|
let topology = TopologyWithSwitch::new_port_not_found();
|
||||||
|
|
||||||
|
let _ = score.interpret(&Inventory::empty(), &topology).await;
|
||||||
|
|
||||||
|
let configured_host_networks = topology.configured_host_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_host_networks).contains_exactly(vec![(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
HostNetworkConfig {
|
||||||
|
bond: Bond { interfaces: vec![] },
|
||||||
|
},
|
||||||
|
)]);
|
||||||
|
let configured_switch_networks = topology.configured_switch_networks.lock().unwrap();
|
||||||
|
assert_that!(*configured_switch_networks).contains_exactly(vec![(
|
||||||
|
HOST_ID.clone(),
|
||||||
|
SwitchNetworkConfig {
|
||||||
|
port_channel: PortChannel { ports: vec![] },
|
||||||
|
},
|
||||||
|
)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn given_score(hosts: Vec<PhysicalHost>) -> HostNetworkConfigurationScore {
|
||||||
|
HostNetworkConfigurationScore { hosts }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn given_host(id: &Id, mac_addresses: Vec<MacAddress>) -> PhysicalHost {
|
||||||
|
let network = mac_addresses.iter().map(|m| given_interface(*m)).collect();
|
||||||
|
|
||||||
PhysicalHost {
|
PhysicalHost {
|
||||||
id: id.clone(),
|
id: id.clone(),
|
||||||
category: HostCategory::Server,
|
category: HostCategory::Server,
|
||||||
network: vec![NetworkInterface {
|
network,
|
||||||
name: "interface-1".into(),
|
|
||||||
mac_address,
|
|
||||||
speed_mbps: None,
|
|
||||||
is_up: true,
|
|
||||||
mtu: 1,
|
|
||||||
ipv4_addresses: vec![],
|
|
||||||
ipv6_addresses: vec![],
|
|
||||||
driver: "driver".into(),
|
|
||||||
firmware_version: None,
|
|
||||||
}],
|
|
||||||
storage: vec![],
|
storage: vec![],
|
||||||
labels: vec![],
|
labels: vec![],
|
||||||
memory_modules: vec![],
|
memory_modules: vec![],
|
||||||
@ -158,24 +310,46 @@ mod tests {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn given_score(hosts: Vec<PhysicalHost>) -> HostNetworkConfigurationScore {
|
fn given_interface(mac_address: MacAddress) -> NetworkInterface {
|
||||||
HostNetworkConfigurationScore { hosts }
|
NetworkInterface {
|
||||||
|
name: format!("{mac_address}"),
|
||||||
|
mac_address,
|
||||||
|
speed_mbps: None,
|
||||||
|
is_up: true,
|
||||||
|
mtu: 1,
|
||||||
|
ipv4_addresses: vec![],
|
||||||
|
ipv6_addresses: vec![],
|
||||||
|
driver: "driver".into(),
|
||||||
|
firmware_version: None,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct SwitchWithPortTopology {
|
struct TopologyWithSwitch {
|
||||||
|
available_ports: Vec<String>,
|
||||||
configured_host_networks: Arc<Mutex<Vec<(Id, HostNetworkConfig)>>>,
|
configured_host_networks: Arc<Mutex<Vec<(Id, HostNetworkConfig)>>>,
|
||||||
|
configured_switch_networks: Arc<Mutex<Vec<(Id, SwitchNetworkConfig)>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SwitchWithPortTopology {
|
impl TopologyWithSwitch {
|
||||||
fn new() -> Self {
|
fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
available_ports: vec![PORT.clone(), ANOTHER_PORT.clone()],
|
||||||
configured_host_networks: Arc::new(Mutex::new(vec![])),
|
configured_host_networks: Arc::new(Mutex::new(vec![])),
|
||||||
|
configured_switch_networks: Arc::new(Mutex::new(vec![])),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new_port_not_found() -> Self {
|
||||||
|
Self {
|
||||||
|
available_ports: vec![],
|
||||||
|
configured_host_networks: Arc::new(Mutex::new(vec![])),
|
||||||
|
configured_switch_networks: Arc::new(Mutex::new(vec![])),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl Topology for SwitchWithPortTopology {
|
impl Topology for TopologyWithSwitch {
|
||||||
fn name(&self) -> &str {
|
fn name(&self) -> &str {
|
||||||
"SwitchWithPortTopology"
|
"SwitchWithPortTopology"
|
||||||
}
|
}
|
||||||
@ -186,9 +360,18 @@ mod tests {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl Switch for SwitchWithPortTopology {
|
impl Switch for TopologyWithSwitch {
|
||||||
async fn get_port_for_mac_address(&self, mac_address: &MacAddress) -> Option<String> {
|
async fn get_port_for_mac_address(&self, _mac_address: &MacAddress) -> Option<String> {
|
||||||
Some("1/0/42".into())
|
if self.available_ports.is_empty() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
Some(
|
||||||
|
self.available_ports
|
||||||
|
.get(self.configured_host_networks.lock().unwrap().len() % 2)
|
||||||
|
.unwrap()
|
||||||
|
.clone(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn configure_host_network(
|
async fn configure_host_network(
|
||||||
@ -201,5 +384,16 @@ mod tests {
|
|||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn configure_switch_network(
|
||||||
|
&self,
|
||||||
|
host: &PhysicalHost,
|
||||||
|
config: SwitchNetworkConfig,
|
||||||
|
) -> Result<(), SwitchError> {
|
||||||
|
let mut configured_switch_networks = self.configured_switch_networks.lock().unwrap();
|
||||||
|
configured_switch_networks.push((host.id.clone(), config.clone()));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user