Generate net-shaper configuration from stdin, or randomly (#7021)

This commit is contained in:
Pankaj Garg 2019-11-18 14:47:07 -08:00 committed by GitHub
parent c09469fa3a
commit a6196901de
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 167 additions and 5 deletions

1
Cargo.lock generated
View File

@ -3752,6 +3752,7 @@ version = "0.1.0"
dependencies = [ dependencies = [
"clap 2.33.0 (registry+https://github.com/rust-lang/crates.io-index)", "clap 2.33.0 (registry+https://github.com/rust-lang/crates.io-index)",
"log 0.4.8 (registry+https://github.com/rust-lang/crates.io-index)", "log 0.4.8 (registry+https://github.com/rust-lang/crates.io-index)",
"rand 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)",
"semver 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)", "semver 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)",
"serde 1.0.102 (registry+https://github.com/rust-lang/crates.io-index)", "serde 1.0.102 (registry+https://github.com/rust-lang/crates.io-index)",
"serde_derive 1.0.102 (registry+https://github.com/rust-lang/crates.io-index)", "serde_derive 1.0.102 (registry+https://github.com/rust-lang/crates.io-index)",

View File

@ -18,6 +18,7 @@ serde_derive = "1.0.102"
serde_json = "1.0.41" serde_json = "1.0.41"
solana-clap-utils = { path = "../clap-utils", version = "0.21.0" } solana-clap-utils = { path = "../clap-utils", version = "0.21.0" }
solana-logger = { path = "../logger", version = "0.21.0" } solana-logger = { path = "../logger", version = "0.21.0" }
rand = "0.6.5"
[[bin]] [[bin]]
name = "solana-net-shaper" name = "solana-net-shaper"

View File

@ -1,10 +1,12 @@
use clap::{ use clap::{
crate_description, crate_name, crate_version, value_t_or_exit, App, Arg, ArgMatches, SubCommand, crate_description, crate_name, crate_version, value_t, value_t_or_exit, App, Arg, ArgMatches,
SubCommand,
}; };
use rand::{thread_rng, Rng};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::fs;
use std::path::PathBuf; use std::path::PathBuf;
use std::{fs, io};
#[derive(Deserialize, Serialize, Debug)] #[derive(Deserialize, Serialize, Debug)]
struct NetworkInterconnect { struct NetworkInterconnect {
@ -43,6 +45,115 @@ impl NetworkTopology {
true true
} }
pub fn new_from_stdin() -> Self {
let mut input = String::new();
println!("Configure partition map (must add up to 100, e.g. [70, 20, 10]):");
let partitions_str = match io::stdin().read_line(&mut input) {
Ok(_) => input,
Err(error) => panic!("error: {}", error),
};
let partitions: Vec<u8> = serde_json::from_str(&partitions_str)
.expect("Failed to parse input. It must be a JSON string");
let mut interconnects: Vec<NetworkInterconnect> = vec![];
for i in 0..partitions.len() - 1 {
for j in i + 1..partitions.len() {
println!("Configure interconnect ({} <-> {}):", i, j);
let mut input = String::new();
let mut interconnect_config = match io::stdin().read_line(&mut input) {
Ok(_) => input,
Err(error) => panic!("error: {}", error),
};
if interconnect_config.ends_with('\n') {
interconnect_config.pop();
if interconnect_config.ends_with('\r') {
interconnect_config.pop();
}
}
if !interconnect_config.is_empty() {
let interconnect = NetworkInterconnect {
a: i as u8,
b: j as u8,
config: interconnect_config.clone(),
};
interconnects.push(interconnect);
let interconnect = NetworkInterconnect {
a: j as u8,
b: i as u8,
config: interconnect_config,
};
interconnects.push(interconnect);
}
}
}
Self {
partitions,
interconnects,
}
}
fn new_random(max_partitions: usize, max_packet_drop: u8, max_packet_delay: u32) -> Self {
let mut rng = thread_rng();
let num_partitions = rng.gen_range(0, max_partitions + 1);
if num_partitions == 0 {
return NetworkTopology::default();
}
let mut partitions = vec![];
let mut used_partition = 0;
for i in 0..num_partitions {
let partition = if i == num_partitions - 1 {
100 - used_partition
} else {
rng.gen_range(0, 100 - used_partition - num_partitions + i)
};
used_partition += partition;
partitions.push(partition as u8);
}
let mut interconnects: Vec<NetworkInterconnect> = vec![];
for i in 0..partitions.len() - 1 {
for j in i + 1..partitions.len() {
let drop_config = if max_packet_drop > 0 {
let packet_drop = rng.gen_range(0, max_packet_drop + 1);
format!("loss {}% 25% ", packet_drop)
} else {
String::default()
};
let config = if max_packet_delay > 0 {
let packet_delay = rng.gen_range(0, max_packet_delay + 1);
format!("{}delay {}ms 10ms", drop_config, packet_delay)
} else {
drop_config
};
let interconnect = NetworkInterconnect {
a: i as u8,
b: j as u8,
config: config.clone(),
};
interconnects.push(interconnect);
let interconnect = NetworkInterconnect {
a: j as u8,
b: i as u8,
config,
};
interconnects.push(interconnect);
}
}
Self {
partitions,
interconnects,
}
}
} }
fn run( fn run(
@ -377,8 +488,20 @@ fn force_cleanup_network(matches: &ArgMatches) {
flush_iptables_rule(); flush_iptables_rule();
} }
fn configure(_matches: &ArgMatches) { fn configure(matches: &ArgMatches) {
let config = NetworkTopology::default(); let config = if !matches.is_present("random") {
NetworkTopology::new_from_stdin()
} else {
let max_partitions = value_t!(matches, "max-partitions", usize).unwrap_or(4);
let max_drop = value_t!(matches, "max-drop", u8).unwrap_or(100);
let max_delay = value_t!(matches, "max-delay", u32).unwrap_or(50);
NetworkTopology::new_random(max_partitions, max_drop, max_delay)
};
if !config.verify() {
panic!("Failed to verify the configuration");
}
let topology = serde_json::to_string(&config).expect("Failed to write as JSON"); let topology = serde_json::to_string(&config).expect("Failed to write as JSON");
println!("{}", topology); println!("{}", topology);
@ -483,7 +606,44 @@ fn main() {
.help("Name of network interface"), .help("Name of network interface"),
), ),
) )
.subcommand(SubCommand::with_name("configure").about("Generate a config file")) .subcommand(
SubCommand::with_name("configure")
.about("Generate a config file")
.arg(
Arg::with_name("random")
.short("r")
.long("random")
.required(false)
.help("Generate a random config file"),
)
.arg(
Arg::with_name("max-partitions")
.short("p")
.long("max-partitions")
.value_name("count")
.takes_value(true)
.required(false)
.help("Maximum number of partitions. Used only with random configuration generation"),
)
.arg(
Arg::with_name("max-drop")
.short("d")
.long("max-drop")
.value_name("percentage")
.takes_value(true)
.required(false)
.help("Maximum amount of packet drop. Used only with random configuration generation"),
)
.arg(
Arg::with_name("max-delay")
.short("y")
.long("max-delay")
.value_name("ms")
.takes_value(true)
.required(false)
.help("Maximum amount of packet delay. Used only with random configuration generation"),
),
)
.get_matches(); .get_matches();
match matches.subcommand() { match matches.subcommand() {