| 
									
										
										
										
											2021-02-16 14:48:20 -07:00
										 |  |  | #![allow(clippy::integer_arithmetic)]
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  | use clap::{crate_description, crate_name, value_t, value_t_or_exit, App, Arg};
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | use log::*;
 | 
					
						
							|  |  |  | use rand::{thread_rng, Rng};
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  | use solana_client::rpc_client::RpcClient;
 | 
					
						
							| 
									
										
										
										
											2021-05-26 09:15:46 -06:00
										 |  |  | use solana_core::serve_repair::RepairProtocol;
 | 
					
						
							|  |  |  | use solana_gossip::{contact_info::ContactInfo, gossip_service::discover};
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  | use solana_sdk::pubkey::Pubkey;
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | use std::net::{SocketAddr, UdpSocket};
 | 
					
						
							|  |  |  | use std::process::exit;
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  | use std::str::FromStr;
 | 
					
						
							| 
									
										
										
										
											2021-05-22 12:02:13 -04:00
										 |  |  | use std::time::{Duration, Instant};
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  | fn run_dos(
 | 
					
						
							|  |  |  |     nodes: &[ContactInfo],
 | 
					
						
							|  |  |  |     iterations: usize,
 | 
					
						
							|  |  |  |     entrypoint_addr: SocketAddr,
 | 
					
						
							|  |  |  |     data_type: String,
 | 
					
						
							|  |  |  |     data_size: usize,
 | 
					
						
							|  |  |  |     mode: String,
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     data_input: Option<String>,
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  | ) {
 | 
					
						
							|  |  |  |     let mut target = None;
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     let mut rpc_client = None;
 | 
					
						
							|  |  |  |     if nodes.is_empty() {
 | 
					
						
							|  |  |  |         if mode == "rpc" {
 | 
					
						
							|  |  |  |             rpc_client = Some(RpcClient::new_socket(entrypoint_addr));
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         target = Some(entrypoint_addr);
 | 
					
						
							|  |  |  |     } else {
 | 
					
						
							|  |  |  |         for node in nodes {
 | 
					
						
							|  |  |  |             if node.gossip == entrypoint_addr {
 | 
					
						
							|  |  |  |                 target = match mode.as_str() {
 | 
					
						
							|  |  |  |                     "gossip" => Some(node.gossip),
 | 
					
						
							|  |  |  |                     "tvu" => Some(node.tvu),
 | 
					
						
							|  |  |  |                     "tvu_forwards" => Some(node.tvu_forwards),
 | 
					
						
							|  |  |  |                     "tpu" => Some(node.tpu),
 | 
					
						
							|  |  |  |                     "tpu_forwards" => Some(node.tpu_forwards),
 | 
					
						
							|  |  |  |                     "repair" => Some(node.repair),
 | 
					
						
							|  |  |  |                     "serve_repair" => Some(node.serve_repair),
 | 
					
						
							|  |  |  |                     "rpc" => {
 | 
					
						
							|  |  |  |                         rpc_client = Some(RpcClient::new_socket(node.rpc));
 | 
					
						
							|  |  |  |                         None
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                     &_ => panic!("Unknown mode"),
 | 
					
						
							|  |  |  |                 };
 | 
					
						
							|  |  |  |                 break;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     let target = target.expect("should have target");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     info!("Targetting {}", target);
 | 
					
						
							|  |  |  |     let socket = UdpSocket::bind("0.0.0.0:0").unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let mut data = Vec::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     if !nodes.is_empty() {
 | 
					
						
							|  |  |  |         let source = thread_rng().gen_range(0, nodes.len());
 | 
					
						
							|  |  |  |         let mut contact = nodes[source].clone();
 | 
					
						
							| 
									
										
										
										
											2020-10-19 12:12:08 -07:00
										 |  |  |         contact.id = solana_sdk::pubkey::new_rand();
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |         match data_type.as_str() {
 | 
					
						
							|  |  |  |             "repair_highest" => {
 | 
					
						
							|  |  |  |                 let slot = 100;
 | 
					
						
							|  |  |  |                 let req = RepairProtocol::WindowIndexWithNonce(contact, slot, 0, 0);
 | 
					
						
							|  |  |  |                 data = bincode::serialize(&req).unwrap();
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             "repair_shred" => {
 | 
					
						
							|  |  |  |                 let slot = 100;
 | 
					
						
							|  |  |  |                 let req = RepairProtocol::HighestWindowIndexWithNonce(contact, slot, 0, 0);
 | 
					
						
							|  |  |  |                 data = bincode::serialize(&req).unwrap();
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             "repair_orphan" => {
 | 
					
						
							|  |  |  |                 let slot = 100;
 | 
					
						
							|  |  |  |                 let req = RepairProtocol::OrphanWithNonce(contact, slot, 0);
 | 
					
						
							|  |  |  |                 data = bincode::serialize(&req).unwrap();
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             "random" => {
 | 
					
						
							|  |  |  |                 data.resize(data_size, 0);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             "get_account_info" => {}
 | 
					
						
							|  |  |  |             "get_program_accounts" => {}
 | 
					
						
							|  |  |  |             &_ => {
 | 
					
						
							|  |  |  |                 panic!("unknown data type");
 | 
					
						
							|  |  |  |             }
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let mut last_log = Instant::now();
 | 
					
						
							|  |  |  |     let mut count = 0;
 | 
					
						
							|  |  |  |     let mut error_count = 0;
 | 
					
						
							|  |  |  |     loop {
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |         if mode == "rpc" {
 | 
					
						
							|  |  |  |             match data_type.as_str() {
 | 
					
						
							|  |  |  |                 "get_account_info" => {
 | 
					
						
							|  |  |  |                     let res = rpc_client
 | 
					
						
							|  |  |  |                         .as_ref()
 | 
					
						
							|  |  |  |                         .unwrap()
 | 
					
						
							|  |  |  |                         .get_account(&Pubkey::from_str(&data_input.as_ref().unwrap()).unwrap());
 | 
					
						
							|  |  |  |                     if res.is_err() {
 | 
					
						
							|  |  |  |                         error_count += 1;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 "get_program_accounts" => {
 | 
					
						
							|  |  |  |                     let res = rpc_client.as_ref().unwrap().get_program_accounts(
 | 
					
						
							|  |  |  |                         &Pubkey::from_str(&data_input.as_ref().unwrap()).unwrap(),
 | 
					
						
							|  |  |  |                     );
 | 
					
						
							|  |  |  |                     if res.is_err() {
 | 
					
						
							|  |  |  |                         error_count += 1;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |                 &_ => {
 | 
					
						
							|  |  |  |                     panic!("unsupported data type");
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             if data_type == "random" {
 | 
					
						
							|  |  |  |                 thread_rng().fill(&mut data[..]);
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |             let res = socket.send_to(&data, target);
 | 
					
						
							|  |  |  |             if res.is_err() {
 | 
					
						
							|  |  |  |                 error_count += 1;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         }
 | 
					
						
							|  |  |  |         count += 1;
 | 
					
						
							|  |  |  |         if last_log.elapsed().as_secs() > 5 {
 | 
					
						
							|  |  |  |             info!("count: {} errors: {}", count, error_count);
 | 
					
						
							|  |  |  |             last_log = Instant::now();
 | 
					
						
							|  |  |  |             count = 0;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |         if iterations != 0 && count >= iterations {
 | 
					
						
							|  |  |  |             break;
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | fn main() {
 | 
					
						
							| 
									
										
										
										
											2020-04-23 11:46:12 -07:00
										 |  |  |     solana_logger::setup_with_default("solana=info");
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |     let matches = App::new(crate_name!())
 | 
					
						
							|  |  |  |         .about(crate_description!())
 | 
					
						
							| 
									
										
										
										
											2020-05-11 15:02:01 -07:00
										 |  |  |         .version(solana_version::version!())
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |         .arg(
 | 
					
						
							|  |  |  |             Arg::with_name("entrypoint")
 | 
					
						
							|  |  |  |                 .long("entrypoint")
 | 
					
						
							|  |  |  |                 .takes_value(true)
 | 
					
						
							|  |  |  |                 .value_name("HOST:PORT")
 | 
					
						
							|  |  |  |                 .help("Gossip entrypoint address. Usually <ip>:8001"),
 | 
					
						
							|  |  |  |         )
 | 
					
						
							|  |  |  |         .arg(
 | 
					
						
							|  |  |  |             Arg::with_name("mode")
 | 
					
						
							|  |  |  |                 .long("mode")
 | 
					
						
							|  |  |  |                 .takes_value(true)
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .value_name("MODE")
 | 
					
						
							|  |  |  |                 .possible_values(&[
 | 
					
						
							|  |  |  |                     "gossip",
 | 
					
						
							|  |  |  |                     "tvu",
 | 
					
						
							|  |  |  |                     "tvu_forwards",
 | 
					
						
							|  |  |  |                     "tpu",
 | 
					
						
							|  |  |  |                     "tpu_forwards",
 | 
					
						
							|  |  |  |                     "repair",
 | 
					
						
							|  |  |  |                     "serve_repair",
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |                     "rpc",
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 ])
 | 
					
						
							|  |  |  |                 .help("Interface to DoS"),
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |         )
 | 
					
						
							|  |  |  |         .arg(
 | 
					
						
							|  |  |  |             Arg::with_name("data_size")
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .long("data-size")
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |                 .takes_value(true)
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .value_name("BYTES")
 | 
					
						
							|  |  |  |                 .help("Size of packet to DoS with"),
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |         )
 | 
					
						
							|  |  |  |         .arg(
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |             Arg::with_name("data_type")
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .long("data-type")
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |                 .takes_value(true)
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .value_name("TYPE")
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |                 .possible_values(&[
 | 
					
						
							|  |  |  |                     "repair_highest",
 | 
					
						
							|  |  |  |                     "repair_shred",
 | 
					
						
							|  |  |  |                     "repair_orphan",
 | 
					
						
							|  |  |  |                     "random",
 | 
					
						
							|  |  |  |                     "get_account_info",
 | 
					
						
							|  |  |  |                     "get_program_accounts",
 | 
					
						
							|  |  |  |                 ])
 | 
					
						
							| 
									
										
										
										
											2020-04-21 16:58:30 -07:00
										 |  |  |                 .help("Type of data to send"),
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |         )
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |         .arg(
 | 
					
						
							|  |  |  |             Arg::with_name("data_input")
 | 
					
						
							|  |  |  |                 .long("data-input")
 | 
					
						
							|  |  |  |                 .takes_value(true)
 | 
					
						
							|  |  |  |                 .value_name("TYPE")
 | 
					
						
							|  |  |  |                 .help("Data to send"),
 | 
					
						
							|  |  |  |         )
 | 
					
						
							|  |  |  |         .arg(
 | 
					
						
							|  |  |  |             Arg::with_name("skip_gossip")
 | 
					
						
							|  |  |  |                 .long("skip-gossip")
 | 
					
						
							|  |  |  |                 .help("Just use entrypoint address directly"),
 | 
					
						
							|  |  |  |         )
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |         .get_matches();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let mut entrypoint_addr = SocketAddr::from(([127, 0, 0, 1], 8001));
 | 
					
						
							|  |  |  |     if let Some(addr) = matches.value_of("entrypoint") {
 | 
					
						
							|  |  |  |         entrypoint_addr = solana_net_utils::parse_host_port(addr).unwrap_or_else(|e| {
 | 
					
						
							|  |  |  |             eprintln!("failed to parse entrypoint address: {}", e);
 | 
					
						
							|  |  |  |             exit(1)
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  |     let data_size = value_t!(matches, "data_size", usize).unwrap_or(128);
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     let skip_gossip = matches.is_present("skip_gossip");
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     let mode = value_t_or_exit!(matches, "mode", String);
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |     let data_type = value_t_or_exit!(matches, "data_type", String);
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     let data_input = value_t!(matches, "data_input", String).ok();
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     let mut nodes = vec![];
 | 
					
						
							|  |  |  |     if !skip_gossip {
 | 
					
						
							|  |  |  |         info!("Finding cluster entry: {:?}", entrypoint_addr);
 | 
					
						
							|  |  |  |         let (gossip_nodes, _validators) = discover(
 | 
					
						
							| 
									
										
										
										
											2021-05-22 12:02:13 -04:00
										 |  |  |             None, // keypair
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |             Some(&entrypoint_addr),
 | 
					
						
							| 
									
										
										
										
											2021-05-22 12:02:13 -04:00
										 |  |  |             None,                    // num_nodes
 | 
					
						
							|  |  |  |             Duration::from_secs(60), // timeout
 | 
					
						
							|  |  |  |             None,                    // find_node_by_pubkey
 | 
					
						
							|  |  |  |             Some(&entrypoint_addr),  // find_node_by_gossip_addr
 | 
					
						
							|  |  |  |             None,                    // my_gossip_addr
 | 
					
						
							|  |  |  |             0,                       // my_shred_version
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |         )
 | 
					
						
							|  |  |  |         .unwrap_or_else(|err| {
 | 
					
						
							|  |  |  |             eprintln!("Failed to discover {} node: {:?}", entrypoint_addr, err);
 | 
					
						
							|  |  |  |             exit(1);
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  |         nodes = gossip_nodes;
 | 
					
						
							|  |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     info!("done found {} nodes", nodes.len());
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |     run_dos(
 | 
					
						
							|  |  |  |         &nodes,
 | 
					
						
							|  |  |  |         0,
 | 
					
						
							|  |  |  |         entrypoint_addr,
 | 
					
						
							|  |  |  |         data_type,
 | 
					
						
							|  |  |  |         data_size,
 | 
					
						
							|  |  |  |         mode,
 | 
					
						
							|  |  |  |         data_input,
 | 
					
						
							|  |  |  |     );
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  | #[cfg(test)]
 | 
					
						
							|  |  |  | pub mod test {
 | 
					
						
							|  |  |  |     use super::*;
 | 
					
						
							|  |  |  |     use solana_sdk::timing::timestamp;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #[test]
 | 
					
						
							|  |  |  |     fn test_dos() {
 | 
					
						
							| 
									
										
										
										
											2020-10-19 12:23:14 -07:00
										 |  |  |         let nodes = [ContactInfo::new_localhost(
 | 
					
						
							|  |  |  |             &solana_sdk::pubkey::new_rand(),
 | 
					
						
							|  |  |  |             timestamp(),
 | 
					
						
							|  |  |  |         )];
 | 
					
						
							| 
									
										
										
										
											2020-05-15 17:35:43 +01:00
										 |  |  |         let entrypoint_addr = nodes[0].gossip;
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         run_dos(
 | 
					
						
							|  |  |  |             &nodes,
 | 
					
						
							|  |  |  |             1,
 | 
					
						
							|  |  |  |             entrypoint_addr,
 | 
					
						
							|  |  |  |             "random".to_string(),
 | 
					
						
							|  |  |  |             10,
 | 
					
						
							|  |  |  |             "tvu".to_string(),
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |             None,
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         run_dos(
 | 
					
						
							|  |  |  |             &nodes,
 | 
					
						
							|  |  |  |             1,
 | 
					
						
							|  |  |  |             entrypoint_addr,
 | 
					
						
							|  |  |  |             "repair_highest".to_string(),
 | 
					
						
							|  |  |  |             10,
 | 
					
						
							|  |  |  |             "repair".to_string(),
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |             None,
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         );
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         run_dos(
 | 
					
						
							|  |  |  |             &nodes,
 | 
					
						
							|  |  |  |             1,
 | 
					
						
							|  |  |  |             entrypoint_addr,
 | 
					
						
							|  |  |  |             "repair_shred".to_string(),
 | 
					
						
							|  |  |  |             10,
 | 
					
						
							|  |  |  |             "serve_repair".to_string(),
 | 
					
						
							| 
									
										
										
										
											2020-09-09 08:21:48 -07:00
										 |  |  |             None,
 | 
					
						
							| 
									
										
										
										
											2020-03-29 14:44:25 -07:00
										 |  |  |         );
 | 
					
						
							| 
									
										
										
										
											2020-03-20 12:55:38 -07:00
										 |  |  |     }
 | 
					
						
							|  |  |  | }
 |