refactor brocade to support different shell versions (e.g. FastIron vs NOS)
Some checks failed
Run Check Script / check (pull_request) Failing after 1m17s
Some checks failed
Run Check Script / check (pull_request) Failing after 1m17s
This commit is contained in:
parent
45e0de2097
commit
ad61be277b
34
Cargo.lock
generated
34
Cargo.lock
generated
@ -679,8 +679,10 @@ name = "brocade"
|
|||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"async-trait",
|
"async-trait",
|
||||||
|
"env_logger",
|
||||||
"harmony_types",
|
"harmony_types",
|
||||||
"log",
|
"log",
|
||||||
|
"regex",
|
||||||
"russh",
|
"russh",
|
||||||
"russh-keys",
|
"russh-keys",
|
||||||
"tokio",
|
"tokio",
|
||||||
@ -2427,6 +2429,17 @@ dependencies = [
|
|||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "harmony_derive"
|
||||||
|
version = "0.1.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2d138bbb32bb346299c5f95fbb53532313f39927cb47c411c99c634ef8665ef7"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn 1.0.109",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "harmony_inventory_agent"
|
name = "harmony_inventory_agent"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
@ -3873,6 +3886,19 @@ dependencies = [
|
|||||||
"web-time",
|
"web-time",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "okd_host_network"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"harmony",
|
||||||
|
"harmony_cli",
|
||||||
|
"harmony_derive",
|
||||||
|
"harmony_inventory_agent",
|
||||||
|
"harmony_macros",
|
||||||
|
"harmony_types",
|
||||||
|
"tokio",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "once_cell"
|
name = "once_cell"
|
||||||
version = "1.21.3"
|
version = "1.21.3"
|
||||||
@ -4560,9 +4586,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "regex"
|
name = "regex"
|
||||||
version = "1.11.2"
|
version = "1.11.3"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "23d7fd106d8c02486a8d64e778353d1cffe08ce79ac2e82f540c86d0facf6912"
|
checksum = "8b5288124840bee7b386bc413c487869b360b2b4ec421ea56425128692f2a82c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"aho-corasick 1.1.3",
|
"aho-corasick 1.1.3",
|
||||||
"memchr",
|
"memchr",
|
||||||
@ -4572,9 +4598,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "regex-automata"
|
name = "regex-automata"
|
||||||
version = "0.4.10"
|
version = "0.4.11"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "6b9458fa0bfeeac22b5ca447c63aaf45f28439a709ccd244698632f9aa6394d6"
|
checksum = "833eb9ce86d40ef33cb1306d8accf7bc8ec2bfea4355cbdebb3df68b40925cad"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"aho-corasick 1.1.3",
|
"aho-corasick 1.1.3",
|
||||||
"memchr",
|
"memchr",
|
||||||
|
@ -1,24 +1,29 @@
|
|||||||
use std::net::{IpAddr, Ipv4Addr};
|
use std::net::{IpAddr, Ipv4Addr};
|
||||||
|
|
||||||
use brocade::BrocadeClient;
|
use harmony_types::switch::PortLocation;
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() {
|
async fn main() {
|
||||||
env_logger::init();
|
env_logger::init();
|
||||||
|
|
||||||
let ip = IpAddr::V4(Ipv4Addr::new(10, 0, 0, 250));
|
let ip = IpAddr::V4(Ipv4Addr::new(10, 0, 0, 250)); // old brocade @ ianlet
|
||||||
|
// let ip = IpAddr::V4(Ipv4Addr::new(192, 168, 55, 101)); // brocade @ sto1
|
||||||
let switch_addresses = vec![ip];
|
let switch_addresses = vec![ip];
|
||||||
|
|
||||||
let brocade = BrocadeClient::init(&switch_addresses, "admin", "password", None)
|
let brocade = brocade::init(&switch_addresses, 22, "admin", "password", None)
|
||||||
.await
|
.await
|
||||||
.expect("Brocade client failed to connect");
|
.expect("Brocade client failed to connect");
|
||||||
|
|
||||||
|
let version = brocade.version().await.unwrap();
|
||||||
|
println!("Version: {version:?}");
|
||||||
|
|
||||||
|
println!("--------------");
|
||||||
println!("Showing MAC Address table...");
|
println!("Showing MAC Address table...");
|
||||||
|
|
||||||
let mac_adddresses = brocade.show_mac_address_table().await.unwrap();
|
let mac_adddresses = brocade.show_mac_address_table().await.unwrap();
|
||||||
println!("VLAN\tMAC\t\t\tPORT");
|
println!("VLAN\tMAC\t\t\tPORT");
|
||||||
for mac in mac_adddresses {
|
for mac in mac_adddresses {
|
||||||
println!("{}\t{}\t{}", mac.vlan, mac.mac_address, mac.port_name);
|
println!("{}\t{}\t{}", mac.vlan, mac.mac_address, mac.port);
|
||||||
}
|
}
|
||||||
|
|
||||||
println!("--------------");
|
println!("--------------");
|
||||||
@ -37,11 +42,11 @@ async fn main() {
|
|||||||
|
|
||||||
println!("--------------");
|
println!("--------------");
|
||||||
let channel_name = "HARMONY_LAG";
|
let channel_name = "HARMONY_LAG";
|
||||||
let ports = vec!["1/1/3".to_string()];
|
let ports = [PortLocation(1, 1, 3), PortLocation(1, 1, 4)];
|
||||||
println!("Creating port channel '{channel_name}' with ports {ports:?}'...");
|
println!("Creating port channel '{channel_name}' with ports {ports:?}'...");
|
||||||
|
|
||||||
brocade
|
brocade
|
||||||
.create_port_channel(channel_name, channel_id, &ports)
|
.create_port_channel(channel_id, channel_name, &ports)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
161
brocade/src/fast_iron.rs
Normal file
161
brocade/src/fast_iron.rs
Normal file
@ -0,0 +1,161 @@
|
|||||||
|
use super::BrocadeClient;
|
||||||
|
use crate::{
|
||||||
|
BrocadeInfo, Error, ExecutionMode, MacAddressEntry, PortChannelId, parse_brocade_mac_address,
|
||||||
|
shell::BrocadeShell,
|
||||||
|
};
|
||||||
|
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use harmony_types::switch::{PortDeclaration, PortLocation};
|
||||||
|
use log::{debug, info};
|
||||||
|
use regex::Regex;
|
||||||
|
use std::{collections::HashSet, str::FromStr};
|
||||||
|
|
||||||
|
pub struct FastIronClient {
|
||||||
|
pub shell: BrocadeShell,
|
||||||
|
pub version: BrocadeInfo,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FastIronClient {
|
||||||
|
pub fn parse_mac_entry(&self, line: &str) -> Option<Result<MacAddressEntry, Error>> {
|
||||||
|
debug!("[Brocade] Parsing mac address entry: {line}");
|
||||||
|
let parts: Vec<&str> = line.split_whitespace().collect();
|
||||||
|
if parts.len() < 3 {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let (vlan, mac_address, port) = match parts.len() {
|
||||||
|
3 => (
|
||||||
|
u16::from_str(parts[0]).ok()?,
|
||||||
|
parse_brocade_mac_address(parts[1]).ok()?,
|
||||||
|
parts[2].to_string(),
|
||||||
|
),
|
||||||
|
_ => (
|
||||||
|
1,
|
||||||
|
parse_brocade_mac_address(parts[0]).ok()?,
|
||||||
|
parts[1].to_string(),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
let port =
|
||||||
|
PortDeclaration::parse(&port).map_err(|e| Error::UnexpectedError(format!("{e}")));
|
||||||
|
|
||||||
|
match port {
|
||||||
|
Ok(p) => Some(Ok(MacAddressEntry {
|
||||||
|
vlan,
|
||||||
|
mac_address,
|
||||||
|
port: p,
|
||||||
|
})),
|
||||||
|
Err(e) => Some(Err(e)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn build_port_channel_commands(
|
||||||
|
&self,
|
||||||
|
channel_name: &str,
|
||||||
|
channel_id: u8,
|
||||||
|
ports: &[PortLocation],
|
||||||
|
) -> Vec<String> {
|
||||||
|
let mut commands = vec![
|
||||||
|
"configure terminal".to_string(),
|
||||||
|
format!("lag {channel_name} static id {channel_id}"),
|
||||||
|
];
|
||||||
|
|
||||||
|
for port in ports {
|
||||||
|
commands.push(format!("ports ethernet {port}"));
|
||||||
|
}
|
||||||
|
|
||||||
|
commands.push(format!("primary-port {}", ports[0]));
|
||||||
|
commands.push("deploy".into());
|
||||||
|
commands.push("exit".into());
|
||||||
|
commands.push("write memory".into());
|
||||||
|
commands.push("exit".into());
|
||||||
|
|
||||||
|
commands
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl BrocadeClient for FastIronClient {
|
||||||
|
async fn version(&self) -> Result<BrocadeInfo, Error> {
|
||||||
|
Ok(self.version.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn show_mac_address_table(&self) -> Result<Vec<MacAddressEntry>, Error> {
|
||||||
|
info!("[Brocade] Showing MAC address table...");
|
||||||
|
|
||||||
|
let output = self
|
||||||
|
.shell
|
||||||
|
.run_command("show mac-address", ExecutionMode::Regular)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
output
|
||||||
|
.lines()
|
||||||
|
.skip(2)
|
||||||
|
.filter_map(|line| self.parse_mac_entry(line))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn find_available_channel_id(&self) -> Result<PortChannelId, Error> {
|
||||||
|
info!("[Brocade] Finding next available channel id...");
|
||||||
|
|
||||||
|
let output = self
|
||||||
|
.shell
|
||||||
|
.run_command("show lag", ExecutionMode::Regular)
|
||||||
|
.await?;
|
||||||
|
let re = Regex::new(r"=== LAG .* ID\s+(\d+)").expect("Invalid regex");
|
||||||
|
|
||||||
|
let used_ids: HashSet<u8> = output
|
||||||
|
.lines()
|
||||||
|
.filter_map(|line| {
|
||||||
|
re.captures(line)
|
||||||
|
.and_then(|c| c.get(1))
|
||||||
|
.and_then(|id_match| id_match.as_str().parse().ok())
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let mut next_id: u8 = 1;
|
||||||
|
loop {
|
||||||
|
if !used_ids.contains(&next_id) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
next_id += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
info!("[Brocade] Found channel id: {next_id}");
|
||||||
|
Ok(next_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn create_port_channel(
|
||||||
|
&self,
|
||||||
|
channel_id: PortChannelId,
|
||||||
|
channel_name: &str,
|
||||||
|
ports: &[PortLocation],
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
info!(
|
||||||
|
"[Brocade] Configuring port-channel '{channel_name} {channel_id}' with ports: {ports:?}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let commands = self.build_port_channel_commands(channel_name, channel_id, ports);
|
||||||
|
self.shell
|
||||||
|
.run_commands(commands, ExecutionMode::Privileged)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
info!("[Brocade] Port-channel '{channel_name}' configured.");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn clear_port_channel(&self, channel_name: &str) -> Result<(), Error> {
|
||||||
|
debug!("[Brocade] Clearing port-channel: {channel_name}");
|
||||||
|
|
||||||
|
let commands = vec![
|
||||||
|
"configure terminal".to_string(),
|
||||||
|
format!("no lag {channel_name}"),
|
||||||
|
"write memory".to_string(),
|
||||||
|
];
|
||||||
|
self.shell
|
||||||
|
.run_commands(commands, ExecutionMode::Privileged)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
@ -1,57 +1,27 @@
|
|||||||
|
use std::net::IpAddr;
|
||||||
use std::{
|
use std::{
|
||||||
borrow::Cow,
|
|
||||||
collections::HashSet,
|
|
||||||
fmt::{self, Display},
|
fmt::{self, Display},
|
||||||
sync::Arc,
|
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
fast_iron::FastIronClient,
|
||||||
|
shell::{BrocadeSession, BrocadeShell},
|
||||||
|
};
|
||||||
|
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use harmony_types::net::{IpAddress, MacAddress};
|
use harmony_types::net::MacAddress;
|
||||||
use log::{debug, info};
|
use harmony_types::switch::{PortDeclaration, PortLocation};
|
||||||
use regex::Regex;
|
use regex::Regex;
|
||||||
use russh::{ChannelMsg, client::Handler, kex::DH_G1_SHA1};
|
|
||||||
use russh_keys::key::{self, SSH_RSA};
|
|
||||||
use std::str::FromStr;
|
|
||||||
use tokio::time::{Instant, timeout};
|
|
||||||
|
|
||||||
static PORT_CHANNEL_NAME: &str = "HARMONY_LAG";
|
mod fast_iron;
|
||||||
|
mod shell;
|
||||||
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone)]
|
mod ssh;
|
||||||
pub struct MacAddressEntry {
|
|
||||||
pub vlan: u16,
|
|
||||||
pub mac_address: MacAddress,
|
|
||||||
pub port_name: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Display for MacAddressEntry {
|
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
f.write_str(
|
|
||||||
format!(
|
|
||||||
"VLAN\tMAC-Address\t\tPort\n{}\t{}\t{}",
|
|
||||||
self.vlan, self.mac_address, self.port_name
|
|
||||||
)
|
|
||||||
.as_str(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct BrocadeClient {
|
|
||||||
ip: IpAddress,
|
|
||||||
user: UserConfig,
|
|
||||||
options: BrocadeOptions,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Default, Clone, Debug)]
|
|
||||||
struct UserConfig {
|
|
||||||
username: String,
|
|
||||||
password: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Default, Clone, Debug)]
|
#[derive(Default, Clone, Debug)]
|
||||||
pub struct BrocadeOptions {
|
pub struct BrocadeOptions {
|
||||||
pub dry_run: bool,
|
pub dry_run: bool,
|
||||||
pub ssh: SshOptions,
|
pub ssh: ssh::SshOptions,
|
||||||
pub timeouts: TimeoutConfig,
|
pub timeouts: TimeoutConfig,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -74,590 +44,127 @@ impl Default for TimeoutConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
|
||||||
pub struct SshOptions {
|
|
||||||
pub preferred_algorithms: russh::Preferred,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for SshOptions {
|
|
||||||
fn default() -> Self {
|
|
||||||
Self {
|
|
||||||
preferred_algorithms: russh::Preferred {
|
|
||||||
kex: Cow::Borrowed(&[DH_G1_SHA1]),
|
|
||||||
key: Cow::Borrowed(&[SSH_RSA]),
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
enum ExecutionMode {
|
enum ExecutionMode {
|
||||||
Regular,
|
Regular,
|
||||||
Privileged,
|
Privileged,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BrocadeClient {
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct BrocadeInfo {
|
||||||
|
os: BrocadeOs,
|
||||||
|
version: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub enum BrocadeOs {
|
||||||
|
NetworkOperatingSystem,
|
||||||
|
FastIron,
|
||||||
|
Unknown,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone)]
|
||||||
|
pub struct MacAddressEntry {
|
||||||
|
pub vlan: u16,
|
||||||
|
pub mac_address: MacAddress,
|
||||||
|
pub port: PortDeclaration,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub type PortChannelId = u8;
|
||||||
|
|
||||||
pub async fn init(
|
pub async fn init(
|
||||||
ip_addresses: &[IpAddress],
|
ip_addresses: &[IpAddr],
|
||||||
|
port: u16,
|
||||||
username: &str,
|
username: &str,
|
||||||
password: &str,
|
password: &str,
|
||||||
options: Option<BrocadeOptions>,
|
options: Option<BrocadeOptions>,
|
||||||
) -> Result<Self, Error> {
|
) -> Result<Box<dyn BrocadeClient + Send + Sync>, Error> {
|
||||||
let ip = ip_addresses
|
let shell = BrocadeShell::init(ip_addresses, port, username, password, options).await?;
|
||||||
.first()
|
|
||||||
.ok_or_else(|| Error::ConfigurationError("No IP addresses provided".to_string()))?;
|
|
||||||
|
|
||||||
let options = options.unwrap_or_default();
|
let version_info = shell
|
||||||
|
.with_session(ExecutionMode::Regular, |session| {
|
||||||
|
Box::pin(get_brocade_info(session))
|
||||||
|
})
|
||||||
|
.await?;
|
||||||
|
|
||||||
Ok(Self {
|
Ok(match version_info.os {
|
||||||
ip: *ip,
|
BrocadeOs::FastIron => Box::new(FastIronClient {
|
||||||
user: UserConfig {
|
shell,
|
||||||
username: username.to_string(),
|
version: version_info,
|
||||||
password: password.to_string(),
|
}),
|
||||||
},
|
BrocadeOs::NetworkOperatingSystem => todo!(),
|
||||||
options,
|
BrocadeOs::Unknown => todo!(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn show_mac_address_table(&self) -> Result<Vec<MacAddressEntry>, Error> {
|
#[async_trait]
|
||||||
info!("[Brocade] Showing MAC address table...");
|
pub trait BrocadeClient {
|
||||||
|
async fn version(&self) -> Result<BrocadeInfo, Error>;
|
||||||
|
|
||||||
let output = self
|
async fn show_mac_address_table(&self) -> Result<Vec<MacAddressEntry>, Error>;
|
||||||
.run_command("show mac-address", ExecutionMode::Regular)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
output
|
async fn find_available_channel_id(&self) -> Result<PortChannelId, Error>;
|
||||||
.lines()
|
|
||||||
.skip(2)
|
|
||||||
.filter_map(|line| self.parse_mac_entry(line))
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn find_available_channel_id(&self) -> Result<u8, Error> {
|
async fn create_port_channel(
|
||||||
info!("[Brocade] Finding next available channel id...");
|
|
||||||
|
|
||||||
let output = self.run_command("show lag", ExecutionMode::Regular).await?;
|
|
||||||
let re = Regex::new(r"=== LAG .* ID\s+(\d+)").expect("Invalid regex");
|
|
||||||
|
|
||||||
let used_ids: HashSet<u8> = output
|
|
||||||
.lines()
|
|
||||||
.filter_map(|line| {
|
|
||||||
re.captures(line)
|
|
||||||
.and_then(|c| c.get(1))
|
|
||||||
.and_then(|id_match| id_match.as_str().parse().ok())
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let mut next_id: u8 = 1;
|
|
||||||
loop {
|
|
||||||
if !used_ids.contains(&next_id) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
next_id += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
info!("[Brocade] Found channel id: {next_id}");
|
|
||||||
Ok(next_id)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn create_port_channel(
|
|
||||||
&self,
|
&self,
|
||||||
|
channel_id: PortChannelId,
|
||||||
channel_name: &str,
|
channel_name: &str,
|
||||||
channel_id: u8,
|
ports: &[PortLocation],
|
||||||
ports: &[String],
|
) -> Result<(), Error>;
|
||||||
) -> Result<(), Error> {
|
|
||||||
info!(
|
|
||||||
"[Brocade] Configuring port-channel '{channel_name} {channel_id}' with ports: {ports:?}"
|
|
||||||
);
|
|
||||||
|
|
||||||
let commands = self.build_port_channel_commands(channel_name, channel_id, ports);
|
async fn clear_port_channel(&self, channel_name: &str) -> Result<(), Error>;
|
||||||
self.run_commands(commands, ExecutionMode::Privileged)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
info!("[Brocade] Port-channel '{PORT_CHANNEL_NAME}' configured.");
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn clear_port_channel(&self, channel_name: &str) -> Result<(), Error> {
|
async fn get_brocade_info(session: &mut BrocadeSession) -> Result<BrocadeInfo, Error> {
|
||||||
debug!("[Brocade] Clearing port-channel: {channel_name}");
|
let output = session.run_command("show version").await?;
|
||||||
|
|
||||||
let commands = vec![
|
if output.contains("Network Operating System") {
|
||||||
"configure terminal".to_string(),
|
let re = Regex::new(r"Network Operating System Version:\s*(?P<version>[a-zA-Z0-9.\-]+)")
|
||||||
format!("no lag {channel_name}"),
|
.expect("Invalid regex");
|
||||||
"write memory".to_string(),
|
let version = re
|
||||||
];
|
.captures(&output)
|
||||||
self.run_commands(commands, ExecutionMode::Privileged)
|
.and_then(|cap| cap.name("version"))
|
||||||
.await?;
|
.map(|m| m.as_str().to_string())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
Ok(())
|
return Ok(BrocadeInfo {
|
||||||
|
os: BrocadeOs::NetworkOperatingSystem,
|
||||||
|
version,
|
||||||
|
});
|
||||||
|
} else if output.contains("ICX") {
|
||||||
|
let re = Regex::new(r"(?m)^\s*SW: Version\s*(?P<version>[a-zA-Z0-9.\-]+)")
|
||||||
|
.expect("Invalid regex");
|
||||||
|
let version = re
|
||||||
|
.captures(&output)
|
||||||
|
.and_then(|cap| cap.name("version"))
|
||||||
|
.map(|m| m.as_str().to_string())
|
||||||
|
.unwrap_or_default();
|
||||||
|
|
||||||
|
return Ok(BrocadeInfo {
|
||||||
|
os: BrocadeOs::FastIron,
|
||||||
|
version,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_port_channel_commands(
|
Err(Error::UnexpectedError("Unknown Brocade OS version".into()))
|
||||||
&self,
|
|
||||||
channel_name: &str,
|
|
||||||
channel_id: u8,
|
|
||||||
ports: &[String],
|
|
||||||
) -> Vec<String> {
|
|
||||||
let mut commands = vec![
|
|
||||||
"configure terminal".to_string(),
|
|
||||||
format!("lag {channel_name} static id {channel_id}"),
|
|
||||||
];
|
|
||||||
|
|
||||||
for port in ports {
|
|
||||||
commands.push(format!("ports ethernet {port}"));
|
|
||||||
}
|
|
||||||
|
|
||||||
commands.push(format!("primary-port {}", ports.first().unwrap()));
|
|
||||||
commands.push("deploy".into());
|
|
||||||
commands.push("exit".into());
|
|
||||||
commands.push("write memory".into());
|
|
||||||
commands.push("exit".into());
|
|
||||||
|
|
||||||
commands
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn run_command(&self, command: &str, mode: ExecutionMode) -> Result<String, Error> {
|
|
||||||
if self.should_skip_command(command) {
|
|
||||||
return Ok(String::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut channel = self.open_session(&mode).await?;
|
|
||||||
|
|
||||||
let output = self
|
|
||||||
.execute_command_in_session(&mut channel, command)
|
|
||||||
.await?;
|
|
||||||
let cleaned = self.clean_brocade_output(&output, command);
|
|
||||||
|
|
||||||
self.close_session(channel, &mode).await?;
|
|
||||||
|
|
||||||
Ok(cleaned)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn run_commands(&self, commands: Vec<String>, mode: ExecutionMode) -> Result<(), Error> {
|
|
||||||
if commands.is_empty() {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut channel = self.open_session(&mode).await?;
|
|
||||||
|
|
||||||
for command in commands {
|
|
||||||
if self.should_skip_command(&command) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.execute_command_in_session(&mut channel, &command)
|
|
||||||
.await?;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.close_session(channel, &mode).await?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn open_session(
|
|
||||||
&self,
|
|
||||||
mode: &ExecutionMode,
|
|
||||||
) -> Result<russh::Channel<russh::client::Msg>, Error> {
|
|
||||||
let config = russh::client::Config {
|
|
||||||
preferred: self.options.ssh.preferred_algorithms.clone(),
|
|
||||||
..Default::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut client = russh::client::connect(Arc::new(config), (self.ip, 22), Client {}).await?;
|
|
||||||
if !client
|
|
||||||
.authenticate_password(&self.user.username, &self.user.password)
|
|
||||||
.await?
|
|
||||||
{
|
|
||||||
return Err(Error::AuthenticationError(
|
|
||||||
"ssh authentication failed".to_string(),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut channel = client.channel_open_session().await?;
|
|
||||||
self.setup_channel(&mut channel, mode).await?;
|
|
||||||
|
|
||||||
Ok(channel)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn setup_channel(
|
|
||||||
&self,
|
|
||||||
channel: &mut russh::Channel<russh::client::Msg>,
|
|
||||||
mode: &ExecutionMode,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
// Setup PTY and shell
|
|
||||||
channel
|
|
||||||
.request_pty(false, "vt100", 80, 24, 0, 0, &[])
|
|
||||||
.await?;
|
|
||||||
channel.request_shell(false).await?;
|
|
||||||
|
|
||||||
self.wait_for_shell_ready(channel).await?;
|
|
||||||
|
|
||||||
match mode {
|
|
||||||
ExecutionMode::Regular => Ok(()),
|
|
||||||
ExecutionMode::Privileged => {
|
|
||||||
debug!("[Brocade] Attempting privilege escalation (enable mode)...");
|
|
||||||
self.try_elevate_session(channel).await
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn execute_command_in_session(
|
|
||||||
&self,
|
|
||||||
channel: &mut russh::Channel<russh::client::Msg>,
|
|
||||||
command: &str,
|
|
||||||
) -> Result<String, Error> {
|
|
||||||
debug!("[Brocade] Running command: '{command}'...");
|
|
||||||
|
|
||||||
channel.data(format!("{}\n", command).as_bytes()).await?;
|
|
||||||
tokio::time::sleep(Duration::from_millis(100)).await;
|
|
||||||
|
|
||||||
let output = self.collect_command_output(channel).await?;
|
|
||||||
let output = String::from_utf8(output)
|
|
||||||
.map_err(|_| Error::UnexpectedError("Invalid UTF-8 in command output".to_string()))?;
|
|
||||||
|
|
||||||
self.check_for_command_errors(&output, command)?;
|
|
||||||
|
|
||||||
Ok(output)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn try_elevate_session(
|
|
||||||
&self,
|
|
||||||
channel: &mut russh::Channel<russh::client::Msg>,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
channel.data(&b"enable\n"[..]).await?;
|
|
||||||
let start = Instant::now();
|
|
||||||
let mut buffer = Vec::new();
|
|
||||||
|
|
||||||
while start.elapsed() < self.options.timeouts.shell_ready {
|
|
||||||
match timeout(self.options.timeouts.message_wait, channel.wait()).await {
|
|
||||||
Ok(Some(ChannelMsg::Data { data })) => {
|
|
||||||
buffer.extend_from_slice(&data);
|
|
||||||
let output = String::from_utf8_lossy(&buffer);
|
|
||||||
|
|
||||||
if output.ends_with('#') {
|
|
||||||
debug!("[Brocade] Privileged mode established");
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
if output.contains("User Name:") {
|
|
||||||
channel
|
|
||||||
.data(format!("{}\n", self.user.username).as_bytes())
|
|
||||||
.await?;
|
|
||||||
buffer.clear();
|
|
||||||
} else if output.contains("Password:") {
|
|
||||||
// Note: Brocade might not echo the password field
|
|
||||||
channel
|
|
||||||
.data(format!("{}\n", self.user.password).as_bytes())
|
|
||||||
.await?;
|
|
||||||
buffer.clear();
|
|
||||||
} else if output.contains('>') {
|
|
||||||
// Back to user mode, something failed (e.g., wrong password)
|
|
||||||
return Err(Error::AuthenticationError(
|
|
||||||
"Enable authentication failed or access denied.".to_string(),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(Some(_)) => continue,
|
|
||||||
Ok(None) => break,
|
|
||||||
Err(_) => continue,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check final state if timeout was reached
|
|
||||||
let output = String::from_utf8_lossy(&buffer);
|
|
||||||
let elevated = output.ends_with('#');
|
|
||||||
match elevated {
|
|
||||||
true => {
|
|
||||||
debug!("[Brocade] Privileged mode established");
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
false => Err(Error::AuthenticationError(format!(
|
|
||||||
"Enable authentication failed for an unknown reason. Output was:\n{output}",
|
|
||||||
))),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn wait_for_shell_ready(
|
|
||||||
&self,
|
|
||||||
channel: &mut russh::Channel<russh::client::Msg>,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
let mut buffer = Vec::new();
|
|
||||||
let start = Instant::now();
|
|
||||||
|
|
||||||
while start.elapsed() < self.options.timeouts.shell_ready {
|
|
||||||
match timeout(self.options.timeouts.message_wait, channel.wait()).await {
|
|
||||||
Ok(Some(ChannelMsg::Data { data })) => {
|
|
||||||
buffer.extend_from_slice(&data);
|
|
||||||
let output = String::from_utf8_lossy(&buffer);
|
|
||||||
if output.ends_with('>') || output.ends_with('#') {
|
|
||||||
debug!("[Brocade] Shell ready: {}", output.trim());
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(Some(_)) => continue,
|
|
||||||
Ok(None) => break,
|
|
||||||
Err(_) => continue,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn collect_command_output(
|
|
||||||
&self,
|
|
||||||
channel: &mut russh::Channel<russh::client::Msg>,
|
|
||||||
) -> Result<Vec<u8>, Error> {
|
|
||||||
let mut output = Vec::new();
|
|
||||||
let start = Instant::now();
|
|
||||||
|
|
||||||
let read_timeout = Duration::from_millis(500);
|
|
||||||
|
|
||||||
let log_interval = Duration::from_secs(3);
|
|
||||||
let mut last_log = Instant::now();
|
|
||||||
|
|
||||||
loop {
|
|
||||||
if start.elapsed() > self.options.timeouts.command_execution {
|
|
||||||
return Err(Error::TimeoutError(
|
|
||||||
"Timeout waiting for command completion.".to_string(),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if start.elapsed() > Duration::from_secs(5) && last_log.elapsed() > log_interval {
|
|
||||||
info!("[Brocade] Waiting for command output...");
|
|
||||||
last_log = Instant::now();
|
|
||||||
}
|
|
||||||
|
|
||||||
match timeout(read_timeout, channel.wait()).await {
|
|
||||||
Ok(Some(ChannelMsg::Data { data } | ChannelMsg::ExtendedData { data, .. })) => {
|
|
||||||
output.extend_from_slice(&data);
|
|
||||||
|
|
||||||
let current_output = String::from_utf8_lossy(&output);
|
|
||||||
if current_output.contains('>') || current_output.contains('#') {
|
|
||||||
return Ok(output);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Some(ChannelMsg::Eof | ChannelMsg::Close)) => {
|
|
||||||
return Ok(output);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Some(ChannelMsg::ExitStatus { exit_status })) => {
|
|
||||||
debug!("[Brocade] Command exit status: {exit_status}");
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(Some(_)) => continue, // Ignore other channel messages
|
|
||||||
Ok(None) | Err(_) => {
|
|
||||||
if output.is_empty() {
|
|
||||||
if let Ok(None) = timeout(read_timeout, channel.wait()).await {
|
|
||||||
// Check one last time if channel is closed
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we received a timeout (Err) and have output, wait a short time to check for a late prompt
|
|
||||||
tokio::time::sleep(Duration::from_millis(100)).await;
|
|
||||||
|
|
||||||
let current_output = String::from_utf8_lossy(&output);
|
|
||||||
if current_output.contains('>') || current_output.contains('#') {
|
|
||||||
return Ok(output);
|
|
||||||
}
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(output)
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn close_session(
|
|
||||||
&self,
|
|
||||||
mut channel: russh::Channel<russh::client::Msg>,
|
|
||||||
mode: &ExecutionMode,
|
|
||||||
) -> Result<(), Error> {
|
|
||||||
debug!("[Brocade] Closing session...");
|
|
||||||
|
|
||||||
channel.data(&b"exit\n"[..]).await?;
|
|
||||||
if let ExecutionMode::Privileged = mode {
|
|
||||||
channel.data(&b"exit\n"[..]).await?; // Previous exit closed "enable" mode
|
|
||||||
}
|
|
||||||
|
|
||||||
let start = Instant::now();
|
|
||||||
|
|
||||||
while start.elapsed() < self.options.timeouts.cleanup {
|
|
||||||
match timeout(self.options.timeouts.message_wait, channel.wait()).await {
|
|
||||||
Ok(Some(ChannelMsg::Close)) => break,
|
|
||||||
Ok(Some(_)) => continue,
|
|
||||||
Ok(None) | Err(_) => break,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
debug!("[Brocade] Session '{}' closed, bye bye.", channel.id());
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn should_skip_command(&self, command: &str) -> bool {
|
|
||||||
if (command.starts_with("write") || command.starts_with("deploy")) && self.options.dry_run {
|
|
||||||
info!("[Brocade] Dry-run mode enabled, skipping command: {command}");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
false
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_mac_entry(&self, line: &str) -> Option<Result<MacAddressEntry, Error>> {
|
|
||||||
debug!("[Brocade] Parsing mac address entry: {line}");
|
|
||||||
let parts: Vec<&str> = line.split_whitespace().collect();
|
|
||||||
if parts.len() < 3 {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
|
|
||||||
let (vlan, mac_address, port_name) = match parts.len() {
|
|
||||||
3 => (
|
|
||||||
// Format: VLAN/MAC/Port
|
|
||||||
u16::from_str(parts[0]).ok()?,
|
|
||||||
parse_brocade_mac_address(parts[1]).ok()?,
|
|
||||||
parts[2].to_string(),
|
|
||||||
),
|
|
||||||
_ => (
|
|
||||||
// Format: MAC/Port/Type/Index, default VLAN usually 1
|
|
||||||
1,
|
|
||||||
parse_brocade_mac_address(parts[0]).ok()?,
|
|
||||||
parts[1].to_string(),
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
Some(Ok(MacAddressEntry {
|
|
||||||
vlan,
|
|
||||||
mac_address,
|
|
||||||
port_name,
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn clean_brocade_output(&self, raw_output: &str, command: &str) -> String {
|
|
||||||
debug!("[Brocade] Received raw output:\n{raw_output}");
|
|
||||||
|
|
||||||
let lines: Vec<&str> = raw_output.lines().collect();
|
|
||||||
let mut cleaned_lines = Vec::new();
|
|
||||||
let mut output_started = false;
|
|
||||||
let mut command_echo_found = false;
|
|
||||||
|
|
||||||
for line in lines {
|
|
||||||
let trimmed = line.trim();
|
|
||||||
|
|
||||||
if !output_started && trimmed.is_empty() {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if !command_echo_found && trimmed.contains(command) {
|
|
||||||
command_echo_found = true;
|
|
||||||
output_started = true;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if self.is_prompt_line(trimmed) {
|
|
||||||
if output_started && !cleaned_lines.is_empty() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if trimmed == "exit" {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if output_started && !trimmed.is_empty() {
|
|
||||||
cleaned_lines.push(line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove trailing empty lines
|
|
||||||
while cleaned_lines.last() == Some(&"") {
|
|
||||||
cleaned_lines.pop();
|
|
||||||
}
|
|
||||||
|
|
||||||
let output = cleaned_lines.join("\n");
|
|
||||||
debug!("[Brocade] Command output:\n{output}");
|
|
||||||
|
|
||||||
output
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_prompt_line(&self, line: &str) -> bool {
|
|
||||||
line.ends_with('#') || line.ends_with('>') || line.starts_with("SSH@")
|
|
||||||
}
|
|
||||||
|
|
||||||
fn check_for_command_errors(&self, output: &str, command: &str) -> Result<(), Error> {
|
|
||||||
const ERROR_PATTERNS: &[&str] = &[
|
|
||||||
"invalid input",
|
|
||||||
"syntax error",
|
|
||||||
"command not found",
|
|
||||||
"unknown command",
|
|
||||||
"permission denied",
|
|
||||||
"access denied",
|
|
||||||
"authentication failed",
|
|
||||||
"configuration error",
|
|
||||||
"failed to",
|
|
||||||
"error:",
|
|
||||||
];
|
|
||||||
|
|
||||||
let output_lower = output.to_lowercase();
|
|
||||||
|
|
||||||
if ERROR_PATTERNS.iter().any(|&p| output_lower.contains(p)) {
|
|
||||||
return Err(Error::CommandError(format!(
|
|
||||||
"Command '{}' failed: {}",
|
|
||||||
command,
|
|
||||||
output.trim()
|
|
||||||
)));
|
|
||||||
}
|
|
||||||
|
|
||||||
if !command.starts_with("show") && output.trim().is_empty() {
|
|
||||||
return Err(Error::CommandError(format!(
|
|
||||||
"Command '{}' produced no output, which may indicate an error",
|
|
||||||
command
|
|
||||||
)));
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_brocade_mac_address(value: &str) -> Result<MacAddress, String> {
|
fn parse_brocade_mac_address(value: &str) -> Result<MacAddress, String> {
|
||||||
// Remove periods from the Brocade format
|
|
||||||
let cleaned_mac = value.replace('.', "");
|
let cleaned_mac = value.replace('.', "");
|
||||||
|
|
||||||
// Ensure the cleaned string has the correct length for a MAC address
|
|
||||||
if cleaned_mac.len() != 12 {
|
if cleaned_mac.len() != 12 {
|
||||||
return Err(format!("Invalid MAC address: {value}",));
|
return Err(format!("Invalid MAC address: {value}"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse the hexadecimal string into bytes
|
|
||||||
let mut bytes = [0u8; 6];
|
let mut bytes = [0u8; 6];
|
||||||
for (i, pair) in cleaned_mac.as_bytes().chunks(2).enumerate() {
|
for (i, pair) in cleaned_mac.as_bytes().chunks(2).enumerate() {
|
||||||
let byte_str =
|
let byte_str = std::str::from_utf8(pair).map_err(|_| "Invalid UTF-8")?;
|
||||||
std::str::from_utf8(pair).map_err(|_| "Invalid UTF-8 sequence".to_string())?;
|
bytes[i] =
|
||||||
|
u8::from_str_radix(byte_str, 16).map_err(|_| format!("Invalid hex in MAC: {value}"))?;
|
||||||
bytes[i] = u8::from_str_radix(byte_str, 16)
|
|
||||||
.map_err(|_| format!("Invalid hex byte in MAC address: {value}"))?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(MacAddress(bytes))
|
Ok(MacAddress(bytes))
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Client;
|
|
||||||
|
|
||||||
#[async_trait]
|
|
||||||
impl Handler for Client {
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
async fn check_server_key(
|
|
||||||
&mut self,
|
|
||||||
_server_public_key: &key::PublicKey,
|
|
||||||
) -> Result<bool, Self::Error> {
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
NetworkError(String),
|
NetworkError(String),
|
||||||
|
330
brocade/src/shell.rs
Normal file
330
brocade/src/shell.rs
Normal file
@ -0,0 +1,330 @@
|
|||||||
|
use std::net::IpAddr;
|
||||||
|
use std::time::Duration;
|
||||||
|
use std::time::Instant;
|
||||||
|
|
||||||
|
use crate::BrocadeOptions;
|
||||||
|
use crate::Error;
|
||||||
|
use crate::ExecutionMode;
|
||||||
|
use crate::TimeoutConfig;
|
||||||
|
use crate::ssh;
|
||||||
|
|
||||||
|
use log::debug;
|
||||||
|
use log::info;
|
||||||
|
use russh::ChannelMsg;
|
||||||
|
use tokio::time::timeout;
|
||||||
|
|
||||||
|
pub struct BrocadeShell {
|
||||||
|
pub ip: IpAddr,
|
||||||
|
pub port: u16,
|
||||||
|
pub username: String,
|
||||||
|
pub password: String,
|
||||||
|
pub options: BrocadeOptions,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BrocadeShell {
|
||||||
|
pub async fn init(
|
||||||
|
ip_addresses: &[IpAddr],
|
||||||
|
port: u16,
|
||||||
|
username: &str,
|
||||||
|
password: &str,
|
||||||
|
options: Option<BrocadeOptions>,
|
||||||
|
) -> Result<Self, Error> {
|
||||||
|
let ip = ip_addresses
|
||||||
|
.first()
|
||||||
|
.ok_or_else(|| Error::ConfigurationError("No IP addresses provided".to_string()))?;
|
||||||
|
|
||||||
|
let base_options = options.unwrap_or_default();
|
||||||
|
let options = ssh::try_init_client(username, password, ip, base_options).await?;
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
ip: *ip,
|
||||||
|
port,
|
||||||
|
username: username.to_string(),
|
||||||
|
password: password.to_string(),
|
||||||
|
options,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn open_session(&self, mode: ExecutionMode) -> Result<BrocadeSession, Error> {
|
||||||
|
BrocadeSession::open(
|
||||||
|
self.ip,
|
||||||
|
self.port,
|
||||||
|
&self.username,
|
||||||
|
&self.password,
|
||||||
|
self.options.clone(),
|
||||||
|
mode,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn with_session<F, R>(&self, mode: ExecutionMode, callback: F) -> Result<R, Error>
|
||||||
|
where
|
||||||
|
F: FnOnce(
|
||||||
|
&mut BrocadeSession,
|
||||||
|
) -> std::pin::Pin<
|
||||||
|
Box<dyn std::future::Future<Output = Result<R, Error>> + Send + '_>,
|
||||||
|
>,
|
||||||
|
{
|
||||||
|
let mut session = self.open_session(mode).await?;
|
||||||
|
let result = callback(&mut session).await;
|
||||||
|
session.close().await?;
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn run_command(&self, command: &str, mode: ExecutionMode) -> Result<String, Error> {
|
||||||
|
let mut session = self.open_session(mode).await?;
|
||||||
|
let result = session.run_command(command).await;
|
||||||
|
session.close().await?;
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn run_commands(
|
||||||
|
&self,
|
||||||
|
commands: Vec<String>,
|
||||||
|
mode: ExecutionMode,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
let mut session = self.open_session(mode).await?;
|
||||||
|
let result = session.run_commands(commands).await;
|
||||||
|
session.close().await?;
|
||||||
|
result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct BrocadeSession {
|
||||||
|
pub channel: russh::Channel<russh::client::Msg>,
|
||||||
|
pub mode: ExecutionMode,
|
||||||
|
pub options: BrocadeOptions,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BrocadeSession {
|
||||||
|
pub async fn open(
|
||||||
|
ip: IpAddr,
|
||||||
|
port: u16,
|
||||||
|
username: &str,
|
||||||
|
password: &str,
|
||||||
|
options: BrocadeOptions,
|
||||||
|
mode: ExecutionMode,
|
||||||
|
) -> Result<Self, Error> {
|
||||||
|
let client = ssh::create_client(ip, port, username, password, &options).await?;
|
||||||
|
let mut channel = client.channel_open_session().await?;
|
||||||
|
|
||||||
|
channel
|
||||||
|
.request_pty(false, "vt100", 80, 24, 0, 0, &[])
|
||||||
|
.await?;
|
||||||
|
channel.request_shell(false).await?;
|
||||||
|
|
||||||
|
wait_for_shell_ready(&mut channel, &options.timeouts).await?;
|
||||||
|
|
||||||
|
if let ExecutionMode::Privileged = mode {
|
||||||
|
try_elevate_session(&mut channel, username, password, &options.timeouts).await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
channel,
|
||||||
|
mode,
|
||||||
|
options,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn close(&mut self) -> Result<(), Error> {
|
||||||
|
debug!("[Brocade] Closing session...");
|
||||||
|
|
||||||
|
self.channel.data(&b"exit\n"[..]).await?;
|
||||||
|
if let ExecutionMode::Privileged = self.mode {
|
||||||
|
self.channel.data(&b"exit\n"[..]).await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let start = Instant::now();
|
||||||
|
while start.elapsed() < self.options.timeouts.cleanup {
|
||||||
|
match timeout(self.options.timeouts.message_wait, self.channel.wait()).await {
|
||||||
|
Ok(Some(ChannelMsg::Close)) => break,
|
||||||
|
Ok(Some(_)) => continue,
|
||||||
|
Ok(None) | Err(_) => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
debug!("[Brocade] Session closed.");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn run_command(&mut self, command: &str) -> Result<String, Error> {
|
||||||
|
debug!("[Brocade] Running command: '{command}'...");
|
||||||
|
|
||||||
|
self.channel
|
||||||
|
.data(format!("{}\n", command).as_bytes())
|
||||||
|
.await?;
|
||||||
|
tokio::time::sleep(Duration::from_millis(100)).await;
|
||||||
|
|
||||||
|
let output = self.collect_command_output().await?;
|
||||||
|
let output = String::from_utf8(output)
|
||||||
|
.map_err(|_| Error::UnexpectedError("Invalid UTF-8 in command output".to_string()))?;
|
||||||
|
|
||||||
|
self.check_for_command_errors(&output, command)?;
|
||||||
|
Ok(output)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn run_commands(&mut self, commands: Vec<String>) -> Result<(), Error> {
|
||||||
|
for command in commands {
|
||||||
|
self.run_command(&command).await?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn collect_command_output(&mut self) -> Result<Vec<u8>, Error> {
|
||||||
|
let mut output = Vec::new();
|
||||||
|
let start = Instant::now();
|
||||||
|
let read_timeout = Duration::from_millis(500);
|
||||||
|
let log_interval = Duration::from_secs(3);
|
||||||
|
let mut last_log = Instant::now();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
if start.elapsed() > self.options.timeouts.command_execution {
|
||||||
|
return Err(Error::TimeoutError(
|
||||||
|
"Timeout waiting for command completion.".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
if start.elapsed() > Duration::from_secs(5) && last_log.elapsed() > log_interval {
|
||||||
|
info!("[Brocade] Waiting for command output...");
|
||||||
|
last_log = Instant::now();
|
||||||
|
}
|
||||||
|
|
||||||
|
match timeout(read_timeout, self.channel.wait()).await {
|
||||||
|
Ok(Some(ChannelMsg::Data { data } | ChannelMsg::ExtendedData { data, .. })) => {
|
||||||
|
output.extend_from_slice(&data);
|
||||||
|
let current_output = String::from_utf8_lossy(&output);
|
||||||
|
if current_output.contains('>') || current_output.contains('#') {
|
||||||
|
return Ok(output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Some(ChannelMsg::Eof | ChannelMsg::Close)) => return Ok(output),
|
||||||
|
Ok(Some(ChannelMsg::ExitStatus { exit_status })) => {
|
||||||
|
debug!("[Brocade] Command exit status: {exit_status}");
|
||||||
|
}
|
||||||
|
Ok(Some(_)) => continue,
|
||||||
|
Ok(None) | Err(_) => {
|
||||||
|
if output.is_empty() {
|
||||||
|
if let Ok(None) = timeout(read_timeout, self.channel.wait()).await {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
tokio::time::sleep(Duration::from_millis(100)).await;
|
||||||
|
let current_output = String::from_utf8_lossy(&output);
|
||||||
|
if current_output.contains('>') || current_output.contains('#') {
|
||||||
|
return Ok(output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(output)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn check_for_command_errors(&self, output: &str, command: &str) -> Result<(), Error> {
|
||||||
|
const ERROR_PATTERNS: &[&str] = &[
|
||||||
|
"invalid input",
|
||||||
|
"syntax error",
|
||||||
|
"command not found",
|
||||||
|
"unknown command",
|
||||||
|
"permission denied",
|
||||||
|
"access denied",
|
||||||
|
"authentication failed",
|
||||||
|
"configuration error",
|
||||||
|
"failed to",
|
||||||
|
"error:",
|
||||||
|
];
|
||||||
|
|
||||||
|
let output_lower = output.to_lowercase();
|
||||||
|
if ERROR_PATTERNS.iter().any(|&p| output_lower.contains(p)) {
|
||||||
|
return Err(Error::CommandError(format!(
|
||||||
|
"Command '{command}' failed: {}",
|
||||||
|
output.trim()
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
|
||||||
|
if !command.starts_with("show") && output.trim().is_empty() {
|
||||||
|
return Err(Error::CommandError(format!(
|
||||||
|
"Command '{command}' produced no output"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn wait_for_shell_ready(
|
||||||
|
channel: &mut russh::Channel<russh::client::Msg>,
|
||||||
|
timeouts: &TimeoutConfig,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
let mut buffer = Vec::new();
|
||||||
|
let start = Instant::now();
|
||||||
|
|
||||||
|
while start.elapsed() < timeouts.shell_ready {
|
||||||
|
match timeout(timeouts.message_wait, channel.wait()).await {
|
||||||
|
Ok(Some(ChannelMsg::Data { data })) => {
|
||||||
|
buffer.extend_from_slice(&data);
|
||||||
|
let output = String::from_utf8_lossy(&buffer);
|
||||||
|
if output.ends_with('>') || output.ends_with('#') {
|
||||||
|
debug!("[Brocade] Shell ready");
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Some(_)) => continue,
|
||||||
|
Ok(None) => break,
|
||||||
|
Err(_) => continue,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn try_elevate_session(
|
||||||
|
channel: &mut russh::Channel<russh::client::Msg>,
|
||||||
|
username: &str,
|
||||||
|
password: &str,
|
||||||
|
timeouts: &TimeoutConfig,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
channel.data(&b"enable\n"[..]).await?;
|
||||||
|
let start = Instant::now();
|
||||||
|
let mut buffer = Vec::new();
|
||||||
|
|
||||||
|
while start.elapsed() < timeouts.shell_ready {
|
||||||
|
match timeout(timeouts.message_wait, channel.wait()).await {
|
||||||
|
Ok(Some(ChannelMsg::Data { data })) => {
|
||||||
|
buffer.extend_from_slice(&data);
|
||||||
|
let output = String::from_utf8_lossy(&buffer);
|
||||||
|
|
||||||
|
if output.ends_with('#') {
|
||||||
|
debug!("[Brocade] Privileged mode established");
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
if output.contains("User Name:") {
|
||||||
|
channel.data(format!("{}\n", username).as_bytes()).await?;
|
||||||
|
buffer.clear();
|
||||||
|
} else if output.contains("Password:") {
|
||||||
|
channel.data(format!("{}\n", password).as_bytes()).await?;
|
||||||
|
buffer.clear();
|
||||||
|
} else if output.contains('>') {
|
||||||
|
return Err(Error::AuthenticationError(
|
||||||
|
"Enable authentication failed".into(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Some(_)) => continue,
|
||||||
|
Ok(None) => break,
|
||||||
|
Err(_) => continue,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let output = String::from_utf8_lossy(&buffer);
|
||||||
|
if output.ends_with('#') {
|
||||||
|
debug!("[Brocade] Privileged mode established");
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(Error::AuthenticationError(format!(
|
||||||
|
"Enable failed. Output:\n{output}"
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
}
|
113
brocade/src/ssh.rs
Normal file
113
brocade/src/ssh.rs
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
use std::borrow::Cow;
|
||||||
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use russh::client::Handler;
|
||||||
|
use russh::kex::DH_G1_SHA1;
|
||||||
|
use russh::kex::ECDH_SHA2_NISTP256;
|
||||||
|
use russh_keys::key::SSH_RSA;
|
||||||
|
|
||||||
|
use super::BrocadeOptions;
|
||||||
|
use super::Error;
|
||||||
|
|
||||||
|
#[derive(Default, Clone, Debug)]
|
||||||
|
pub struct SshOptions {
|
||||||
|
pub preferred_algorithms: russh::Preferred,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SshOptions {
|
||||||
|
fn ecdhsa_sha2_nistp256() -> Self {
|
||||||
|
Self {
|
||||||
|
preferred_algorithms: russh::Preferred {
|
||||||
|
kex: Cow::Borrowed(&[ECDH_SHA2_NISTP256]),
|
||||||
|
key: Cow::Borrowed(&[SSH_RSA]),
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn legacy() -> Self {
|
||||||
|
Self {
|
||||||
|
preferred_algorithms: russh::Preferred {
|
||||||
|
kex: Cow::Borrowed(&[DH_G1_SHA1]),
|
||||||
|
key: Cow::Borrowed(&[SSH_RSA]),
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Client;
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl Handler for Client {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
async fn check_server_key(
|
||||||
|
&mut self,
|
||||||
|
_server_public_key: &russh_keys::key::PublicKey,
|
||||||
|
) -> Result<bool, Self::Error> {
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn try_init_client(
|
||||||
|
username: &str,
|
||||||
|
password: &str,
|
||||||
|
ip: &std::net::IpAddr,
|
||||||
|
base_options: BrocadeOptions,
|
||||||
|
) -> Result<BrocadeOptions, Error> {
|
||||||
|
let ssh_options = vec![
|
||||||
|
SshOptions::default(),
|
||||||
|
SshOptions::ecdhsa_sha2_nistp256(),
|
||||||
|
SshOptions::legacy(),
|
||||||
|
];
|
||||||
|
|
||||||
|
for ssh in ssh_options {
|
||||||
|
let opts = BrocadeOptions {
|
||||||
|
ssh,
|
||||||
|
..base_options.clone()
|
||||||
|
};
|
||||||
|
let client = create_client(*ip, 22, username, password, &opts).await;
|
||||||
|
|
||||||
|
match client {
|
||||||
|
Ok(_) => {
|
||||||
|
return Ok(opts);
|
||||||
|
}
|
||||||
|
Err(e) => match e {
|
||||||
|
Error::NetworkError(e) => {
|
||||||
|
if e.contains("No common key exchange algorithm") {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
return Err(Error::NetworkError(e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => return Err(e),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Err(Error::NetworkError(
|
||||||
|
"Could not establish ssh connection: wrong key exchange algorithm)".to_string(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn create_client(
|
||||||
|
ip: std::net::IpAddr,
|
||||||
|
port: u16,
|
||||||
|
username: &str,
|
||||||
|
password: &str,
|
||||||
|
options: &BrocadeOptions,
|
||||||
|
) -> Result<russh::client::Handle<Client>, Error> {
|
||||||
|
let config = russh::client::Config {
|
||||||
|
preferred: options.ssh.preferred_algorithms.clone(),
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
let mut client = russh::client::connect(Arc::new(config), (ip, port), Client {}).await?;
|
||||||
|
if !client.authenticate_password(username, password).await? {
|
||||||
|
return Err(Error::AuthenticationError(
|
||||||
|
"ssh authentication failed".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
Ok(client)
|
||||||
|
}
|
@ -4,6 +4,7 @@ use harmony_macros::ip;
|
|||||||
use harmony_secret::SecretManager;
|
use harmony_secret::SecretManager;
|
||||||
use harmony_types::net::MacAddress;
|
use harmony_types::net::MacAddress;
|
||||||
use harmony_types::net::Url;
|
use harmony_types::net::Url;
|
||||||
|
use harmony_types::switch::PortLocation;
|
||||||
use k8s_openapi::api::core::v1::Namespace;
|
use k8s_openapi::api::core::v1::Namespace;
|
||||||
use kube::api::ObjectMeta;
|
use kube::api::ObjectMeta;
|
||||||
use log::debug;
|
use log::debug;
|
||||||
@ -326,11 +327,7 @@ impl HAClusterTopology {
|
|||||||
debug!("Configuring port channel: {config:#?}");
|
debug!("Configuring port channel: {config:#?}");
|
||||||
let client = self.get_switch_client().await?;
|
let client = self.get_switch_client().await?;
|
||||||
|
|
||||||
let switch_ports: Vec<String> = config
|
let switch_ports = config.switch_ports.iter().map(|s| s.port.clone()).collect();
|
||||||
.switch_ports
|
|
||||||
.iter()
|
|
||||||
.map(|s| s.port_name.clone())
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
client
|
client
|
||||||
.configure_port_channel(&format!("Harmony_{}", host.id), switch_ports)
|
.configure_port_channel(&format!("Harmony_{}", host.id), switch_ports)
|
||||||
@ -519,7 +516,7 @@ impl Switch for HAClusterTopology {
|
|||||||
async fn get_port_for_mac_address(
|
async fn get_port_for_mac_address(
|
||||||
&self,
|
&self,
|
||||||
mac_address: &MacAddress,
|
mac_address: &MacAddress,
|
||||||
) -> Result<Option<String>, SwitchError> {
|
) -> Result<Option<PortLocation>, SwitchError> {
|
||||||
let client = self.get_switch_client().await?;
|
let client = self.get_switch_client().await?;
|
||||||
let port = client.find_port(mac_address).await?;
|
let port = client.find_port(mac_address).await?;
|
||||||
Ok(port)
|
Ok(port)
|
||||||
|
@ -2,7 +2,10 @@ use std::{error::Error, net::Ipv4Addr, str::FromStr, sync::Arc};
|
|||||||
|
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use derive_new::new;
|
use derive_new::new;
|
||||||
use harmony_types::net::{IpAddress, MacAddress};
|
use harmony_types::{
|
||||||
|
net::{IpAddress, MacAddress},
|
||||||
|
switch::PortLocation,
|
||||||
|
};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::{executors::ExecutorError, hardware::PhysicalHost};
|
use crate::{executors::ExecutorError, hardware::PhysicalHost};
|
||||||
@ -178,7 +181,7 @@ pub trait Switch: Send + Sync {
|
|||||||
async fn get_port_for_mac_address(
|
async fn get_port_for_mac_address(
|
||||||
&self,
|
&self,
|
||||||
mac_address: &MacAddress,
|
mac_address: &MacAddress,
|
||||||
) -> Result<Option<String>, SwitchError>;
|
) -> Result<Option<PortLocation>, SwitchError>;
|
||||||
|
|
||||||
async fn configure_host_network(
|
async fn configure_host_network(
|
||||||
&self,
|
&self,
|
||||||
@ -195,7 +198,7 @@ pub struct HostNetworkConfig {
|
|||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
pub struct SwitchPort {
|
pub struct SwitchPort {
|
||||||
pub interface: NetworkInterface,
|
pub interface: NetworkInterface,
|
||||||
pub port_name: String,
|
pub port: PortLocation,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq)]
|
#[derive(Clone, Debug, PartialEq)]
|
||||||
@ -221,12 +224,15 @@ impl Error for SwitchError {}
|
|||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
pub trait SwitchClient: Send + Sync {
|
pub trait SwitchClient: Send + Sync {
|
||||||
async fn find_port(&self, mac_address: &MacAddress) -> Result<Option<String>, SwitchError>;
|
async fn find_port(
|
||||||
|
&self,
|
||||||
|
mac_address: &MacAddress,
|
||||||
|
) -> Result<Option<PortLocation>, SwitchError>;
|
||||||
|
|
||||||
async fn configure_port_channel(
|
async fn configure_port_channel(
|
||||||
&self,
|
&self,
|
||||||
channel_name: &str,
|
channel_name: &str,
|
||||||
switch_ports: Vec<String>,
|
switch_ports: Vec<PortLocation>,
|
||||||
) -> Result<u8, SwitchError>;
|
) -> Result<u8, SwitchError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use brocade::{BrocadeClient, BrocadeOptions};
|
use brocade::{BrocadeClient, BrocadeOptions};
|
||||||
use harmony_secret::Secret;
|
use harmony_secret::Secret;
|
||||||
use harmony_types::net::{IpAddress, MacAddress};
|
use harmony_types::{
|
||||||
|
net::{IpAddress, MacAddress},
|
||||||
|
switch::{PortDeclaration, PortLocation},
|
||||||
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::topology::{SwitchClient, SwitchError};
|
use crate::topology::{SwitchClient, SwitchError};
|
||||||
|
|
||||||
pub struct BrocadeSwitchClient {
|
pub struct BrocadeSwitchClient {
|
||||||
brocade: BrocadeClient,
|
brocade: Box<dyn BrocadeClient + Send + Sync>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BrocadeSwitchClient {
|
impl BrocadeSwitchClient {
|
||||||
@ -17,30 +20,44 @@ impl BrocadeSwitchClient {
|
|||||||
password: &str,
|
password: &str,
|
||||||
options: Option<BrocadeOptions>,
|
options: Option<BrocadeOptions>,
|
||||||
) -> Result<Self, brocade::Error> {
|
) -> Result<Self, brocade::Error> {
|
||||||
let brocade = BrocadeClient::init(ip_addresses, username, password, options).await?;
|
let brocade = brocade::init(ip_addresses, 22, username, password, options).await?;
|
||||||
Ok(Self { brocade })
|
Ok(Self { brocade })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl SwitchClient for BrocadeSwitchClient {
|
impl SwitchClient for BrocadeSwitchClient {
|
||||||
async fn find_port(&self, mac_address: &MacAddress) -> Result<Option<String>, SwitchError> {
|
async fn find_port(
|
||||||
|
&self,
|
||||||
|
mac_address: &MacAddress,
|
||||||
|
) -> Result<Option<PortLocation>, SwitchError> {
|
||||||
let table = self
|
let table = self
|
||||||
.brocade
|
.brocade
|
||||||
.show_mac_address_table()
|
.show_mac_address_table()
|
||||||
.await
|
.await
|
||||||
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
||||||
|
|
||||||
Ok(table
|
let port = table
|
||||||
.iter()
|
.iter()
|
||||||
.find(|entry| entry.mac_address == *mac_address)
|
.find(|entry| entry.mac_address == *mac_address)
|
||||||
.map(|entry| entry.port_name.clone()))
|
.map(|entry| match &entry.port {
|
||||||
|
PortDeclaration::Single(port_location) => Ok(port_location.clone()),
|
||||||
|
_ => Err(SwitchError::new(
|
||||||
|
"Multiple ports found for MAC address".into(),
|
||||||
|
)),
|
||||||
|
});
|
||||||
|
|
||||||
|
match port {
|
||||||
|
Some(Ok(p)) => Ok(Some(p)),
|
||||||
|
Some(Err(e)) => Err(e),
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn configure_port_channel(
|
async fn configure_port_channel(
|
||||||
&self,
|
&self,
|
||||||
channel_name: &str,
|
channel_name: &str,
|
||||||
switch_ports: Vec<String>,
|
switch_ports: Vec<PortLocation>,
|
||||||
) -> Result<u8, SwitchError> {
|
) -> Result<u8, SwitchError> {
|
||||||
let channel_id = self
|
let channel_id = self
|
||||||
.brocade
|
.brocade
|
||||||
@ -49,7 +66,7 @@ impl SwitchClient for BrocadeSwitchClient {
|
|||||||
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
||||||
|
|
||||||
self.brocade
|
self.brocade
|
||||||
.create_port_channel(channel_name, channel_id, &switch_ports)
|
.create_port_channel(channel_id, channel_name, &switch_ports)
|
||||||
.await
|
.await
|
||||||
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
.map_err(|e| SwitchError::new(format!("{e}")))?;
|
||||||
|
|
||||||
|
@ -71,7 +71,7 @@ impl<T: Topology + Switch> Interpret<T> for HostNetworkConfigurationInterpret {
|
|||||||
let mac_address = network_interface.mac_address;
|
let mac_address = network_interface.mac_address;
|
||||||
|
|
||||||
match topology.get_port_for_mac_address(&mac_address).await {
|
match topology.get_port_for_mac_address(&mac_address).await {
|
||||||
Ok(Some(port_name)) => {
|
Ok(Some(port)) => {
|
||||||
switch_ports.push(SwitchPort {
|
switch_ports.push(SwitchPort {
|
||||||
interface: NetworkInterface {
|
interface: NetworkInterface {
|
||||||
name: network_interface.name.clone(),
|
name: network_interface.name.clone(),
|
||||||
@ -79,7 +79,7 @@ impl<T: Topology + Switch> Interpret<T> for HostNetworkConfigurationInterpret {
|
|||||||
speed_mbps: network_interface.speed_mbps,
|
speed_mbps: network_interface.speed_mbps,
|
||||||
mtu: network_interface.mtu,
|
mtu: network_interface.mtu,
|
||||||
},
|
},
|
||||||
port_name,
|
port,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
Ok(None) => debug!("No port found for host '{}', skipping", host.id),
|
Ok(None) => debug!("No port found for host '{}', skipping", host.id),
|
||||||
@ -110,7 +110,7 @@ impl<T: Topology + Switch> Interpret<T> for HostNetworkConfigurationInterpret {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use assertor::*;
|
use assertor::*;
|
||||||
use harmony_types::net::MacAddress;
|
use harmony_types::{net::MacAddress, switch::PortLocation};
|
||||||
use lazy_static::lazy_static;
|
use lazy_static::lazy_static;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
@ -147,8 +147,8 @@ mod tests {
|
|||||||
speed_mbps: None,
|
speed_mbps: None,
|
||||||
mtu: 1,
|
mtu: 1,
|
||||||
};
|
};
|
||||||
pub static ref PORT: String = "1/0/42".into();
|
pub static ref PORT: PortLocation = PortLocation(1, 0, 42);
|
||||||
pub static ref ANOTHER_PORT: String = "2/0/42".into();
|
pub static ref ANOTHER_PORT: PortLocation = PortLocation(2, 0, 42);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
@ -165,7 +165,7 @@ mod tests {
|
|||||||
HostNetworkConfig {
|
HostNetworkConfig {
|
||||||
switch_ports: vec![SwitchPort {
|
switch_ports: vec![SwitchPort {
|
||||||
interface: EXISTING_INTERFACE.clone(),
|
interface: EXISTING_INTERFACE.clone(),
|
||||||
port_name: PORT.clone(),
|
port: PORT.clone(),
|
||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
)]);
|
)]);
|
||||||
@ -191,11 +191,11 @@ mod tests {
|
|||||||
switch_ports: vec![
|
switch_ports: vec![
|
||||||
SwitchPort {
|
SwitchPort {
|
||||||
interface: EXISTING_INTERFACE.clone(),
|
interface: EXISTING_INTERFACE.clone(),
|
||||||
port_name: PORT.clone(),
|
port: PORT.clone(),
|
||||||
},
|
},
|
||||||
SwitchPort {
|
SwitchPort {
|
||||||
interface: ANOTHER_EXISTING_INTERFACE.clone(),
|
interface: ANOTHER_EXISTING_INTERFACE.clone(),
|
||||||
port_name: ANOTHER_PORT.clone(),
|
port: ANOTHER_PORT.clone(),
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@ -219,7 +219,7 @@ mod tests {
|
|||||||
HostNetworkConfig {
|
HostNetworkConfig {
|
||||||
switch_ports: vec![SwitchPort {
|
switch_ports: vec![SwitchPort {
|
||||||
interface: EXISTING_INTERFACE.clone(),
|
interface: EXISTING_INTERFACE.clone(),
|
||||||
port_name: PORT.clone(),
|
port: PORT.clone(),
|
||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@ -228,7 +228,7 @@ mod tests {
|
|||||||
HostNetworkConfig {
|
HostNetworkConfig {
|
||||||
switch_ports: vec![SwitchPort {
|
switch_ports: vec![SwitchPort {
|
||||||
interface: ANOTHER_EXISTING_INTERFACE.clone(),
|
interface: ANOTHER_EXISTING_INTERFACE.clone(),
|
||||||
port_name: ANOTHER_PORT.clone(),
|
port: ANOTHER_PORT.clone(),
|
||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
@ -282,7 +282,7 @@ mod tests {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct TopologyWithSwitch {
|
struct TopologyWithSwitch {
|
||||||
available_ports: Arc<Mutex<Vec<String>>>,
|
available_ports: Arc<Mutex<Vec<PortLocation>>>,
|
||||||
configured_host_networks: Arc<Mutex<Vec<(Id, HostNetworkConfig)>>>,
|
configured_host_networks: Arc<Mutex<Vec<(Id, HostNetworkConfig)>>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -318,7 +318,7 @@ mod tests {
|
|||||||
async fn get_port_for_mac_address(
|
async fn get_port_for_mac_address(
|
||||||
&self,
|
&self,
|
||||||
_mac_address: &MacAddress,
|
_mac_address: &MacAddress,
|
||||||
) -> Result<Option<String>, SwitchError> {
|
) -> Result<Option<PortLocation>, SwitchError> {
|
||||||
let mut ports = self.available_ports.lock().unwrap();
|
let mut ports = self.available_ports.lock().unwrap();
|
||||||
if ports.is_empty() {
|
if ports.is_empty() {
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
|
@ -19,7 +19,7 @@ use serde::{Deserialize, Serialize};
|
|||||||
///
|
///
|
||||||
/// **It is not meant to be very secure or unique**, it is suitable to generate up to 10 000 items per
|
/// **It is not meant to be very secure or unique**, it is suitable to generate up to 10 000 items per
|
||||||
/// second with a reasonable collision rate of 0,000014 % as calculated by this calculator : https://kevingal.com/apps/collision.html
|
/// second with a reasonable collision rate of 0,000014 % as calculated by this calculator : https://kevingal.com/apps/collision.html
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
|
||||||
pub struct Id {
|
pub struct Id {
|
||||||
value: String,
|
value: String,
|
||||||
}
|
}
|
||||||
|
@ -1,2 +1,3 @@
|
|||||||
pub mod id;
|
pub mod id;
|
||||||
pub mod net;
|
pub mod net;
|
||||||
|
pub mod switch;
|
||||||
|
176
harmony_types/src/switch.rs
Normal file
176
harmony_types/src/switch.rs
Normal file
@ -0,0 +1,176 @@
|
|||||||
|
use std::{fmt, str::FromStr};
|
||||||
|
|
||||||
|
/// Simple error type for port parsing failures.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum PortParseError {
|
||||||
|
/// The port string did not conform to the expected S/M/P or range format.
|
||||||
|
InvalidFormat,
|
||||||
|
/// A stack, module, or port segment could not be parsed as a number.
|
||||||
|
InvalidSegment(String),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for PortParseError {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
PortParseError::InvalidFormat => write!(f, "Port string is in an unexpected format."),
|
||||||
|
PortParseError::InvalidSegment(s) => write!(f, "Invalid segment in port string: {}", s),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Represents the atomic, physical location of a switch port: `<Stack>/<Module>/<Port>`.
|
||||||
|
///
|
||||||
|
/// Example: `1/1/1`
|
||||||
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone)]
|
||||||
|
pub struct PortLocation(pub u8, pub u8, pub u8);
|
||||||
|
|
||||||
|
impl fmt::Display for PortLocation {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
write!(f, "{}/{}/{}", self.0, self.1, self.2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromStr for PortLocation {
|
||||||
|
type Err = PortParseError;
|
||||||
|
|
||||||
|
/// Parses a string slice into a `PortLocation`.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// use std::str::FromStr;
|
||||||
|
/// use brocade::port::PortLocation;
|
||||||
|
///
|
||||||
|
/// assert_eq!(PortLocation::from_str("1/1/1").unwrap(), PortLocation(1, 1, 1));
|
||||||
|
/// assert_eq!(PortLocation::from_str("12/5/48").unwrap(), PortLocation(12, 5, 48));
|
||||||
|
/// assert!(PortLocation::from_str("1/A/1").is_err());
|
||||||
|
/// ```
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
let parts: Vec<&str> = s.split('/').collect();
|
||||||
|
|
||||||
|
if parts.len() != 3 {
|
||||||
|
return Err(PortParseError::InvalidFormat);
|
||||||
|
}
|
||||||
|
|
||||||
|
let parse_segment = |part: &str| -> Result<u8, Self::Err> {
|
||||||
|
u8::from_str(part).map_err(|_| PortParseError::InvalidSegment(part.to_string()))
|
||||||
|
};
|
||||||
|
|
||||||
|
let stack = parse_segment(parts[0])?;
|
||||||
|
let module = parse_segment(parts[1])?;
|
||||||
|
let port = parse_segment(parts[2])?;
|
||||||
|
|
||||||
|
Ok(PortLocation(stack, module, port))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Represents a Brocade Port configuration input, which can be a single port, a sequential
|
||||||
|
/// range, or an explicit set defined by endpoints.
|
||||||
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone)]
|
||||||
|
pub enum PortDeclaration {
|
||||||
|
/// A single switch port defined by its location. Example: `PortDeclaration::Single(1/1/1)`
|
||||||
|
Single(PortLocation),
|
||||||
|
/// A strictly sequential range defined by two endpoints using the hyphen separator (`-`).
|
||||||
|
/// All ports between the endpoints (inclusive) are implicitly included.
|
||||||
|
/// Example: `PortDeclaration::Range(1/1/1, 1/1/4)`
|
||||||
|
Range(PortLocation, PortLocation),
|
||||||
|
/// A set of ports defined by two endpoints using the asterisk separator (`*`).
|
||||||
|
/// The actual member ports must be determined contextually (e.g., from MAC tables or
|
||||||
|
/// explicit configuration lists).
|
||||||
|
/// Example: `PortDeclaration::Set(1/1/1, 1/1/3)` where only ports 1 and 3 might be active.
|
||||||
|
Set(PortLocation, PortLocation),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PortDeclaration {
|
||||||
|
/// Parses a Brocade port configuration string into a structured `PortDeclaration` enum.
|
||||||
|
///
|
||||||
|
/// This function performs only basic format and numerical parsing, assuming the input
|
||||||
|
/// strings (e.g., from `show` commands) are semantically valid and logically ordered.
|
||||||
|
///
|
||||||
|
/// # Supported Formats
|
||||||
|
///
|
||||||
|
/// * **Single Port:** `"1/1/1"`
|
||||||
|
/// * **Range (Hyphen, `-`):** `"1/1/1-1/1/4"`
|
||||||
|
/// * **Set (Asterisk, `*`):** `"1/1/1*1/1/4"`
|
||||||
|
///
|
||||||
|
/// # Errors
|
||||||
|
///
|
||||||
|
/// Returns `PortParseError` if the string format is incorrect or numerical segments
|
||||||
|
/// cannot be parsed.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// use brocade::port::{PortDeclaration, PortLocation};
|
||||||
|
///
|
||||||
|
/// // Single Port
|
||||||
|
/// assert_eq!(PortDeclaration::parse("3/2/15").unwrap(), PortDeclaration::Single(PortLocation(3, 2, 15)));
|
||||||
|
///
|
||||||
|
/// // Range (Hyphen) - implies sequential ports
|
||||||
|
/// let result_range = PortDeclaration::parse("1/1/1-1/1/4").unwrap();
|
||||||
|
/// assert_eq!(result_range, PortDeclaration::Range(PortLocation(1, 1, 1), PortLocation(1, 1, 4)));
|
||||||
|
///
|
||||||
|
/// // Set (Asterisk) - implies non-sequential set defined by endpoints
|
||||||
|
/// let result_set = PortDeclaration::parse("1/1/48*2/1/48").unwrap();
|
||||||
|
/// assert_eq!(result_set, PortDeclaration::Set(PortLocation(1, 1, 48), PortLocation(2, 1, 48)));
|
||||||
|
///
|
||||||
|
/// // Invalid Format (will still fail basic parsing)
|
||||||
|
/// assert!(PortDeclaration::parse("1/1/1/1").is_err());
|
||||||
|
/// ```
|
||||||
|
pub fn parse(port_str: &str) -> Result<Self, PortParseError> {
|
||||||
|
if let Some((start_str, end_str)) = port_str.split_once('-') {
|
||||||
|
let start_port = PortLocation::from_str(start_str.trim())?;
|
||||||
|
let end_port = PortLocation::from_str(end_str.trim())?;
|
||||||
|
return Ok(PortDeclaration::Range(start_port, end_port));
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some((start_str, end_str)) = port_str.split_once('*') {
|
||||||
|
let start_port = PortLocation::from_str(start_str.trim())?;
|
||||||
|
let end_port = PortLocation::from_str(end_str.trim())?;
|
||||||
|
return Ok(PortDeclaration::Set(start_port, end_port));
|
||||||
|
}
|
||||||
|
|
||||||
|
let location = PortLocation::from_str(port_str)?;
|
||||||
|
Ok(PortDeclaration::Single(location))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for PortDeclaration {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
match self {
|
||||||
|
PortDeclaration::Single(port) => write!(f, "{port}"),
|
||||||
|
PortDeclaration::Range(start, end) => write!(f, "{start}-{end}"),
|
||||||
|
PortDeclaration::Set(start, end) => write!(f, "{start}*{end}"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_single_port_location_invalid() {
|
||||||
|
assert!(PortLocation::from_str("1/1").is_err());
|
||||||
|
assert!(PortLocation::from_str("1/A/1").is_err());
|
||||||
|
assert!(PortLocation::from_str("1/1/256").is_err());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_declaration_single() {
|
||||||
|
let single_result = PortDeclaration::parse("1/1/4").unwrap();
|
||||||
|
assert!(matches!(single_result, PortDeclaration::Single(_)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_declaration_range() {
|
||||||
|
let range_result = PortDeclaration::parse("1/1/1-1/1/4").unwrap();
|
||||||
|
assert!(matches!(range_result, PortDeclaration::Range(_, _)));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_declaration_set() {
|
||||||
|
let set_result = PortDeclaration::parse("1/1/48*2/1/48").unwrap();
|
||||||
|
assert!(matches!(set_result, PortDeclaration::Set(_, _)));
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user