diff --git a/rita_bin/Cargo.toml b/rita_bin/Cargo.toml index 0be0b75b3..a158d9b71 100644 --- a/rita_bin/Cargo.toml +++ b/rita_bin/Cargo.toml @@ -14,8 +14,8 @@ name = "rita" path = "src/client.rs" [[bin]] -name = "rita_db_migration" -path = "src/database_migration.rs" +name = "contract-util" +path = "src/contract-util.rs" [[bin]] name = "rita_extender" diff --git a/rita_bin/src/contract-util.rs b/rita_bin/src/contract-util.rs new file mode 100644 index 000000000..8f7218225 --- /dev/null +++ b/rita_bin/src/contract-util.rs @@ -0,0 +1,247 @@ +//! This binary is a set of utilities for interacting with the Althea exit infrastructure contract. This contract contains a list of +//! all clients and exits and is used to perform key exchange between both sides. This binary contains a set of utilities for interacting +//! with this contract and performing a variety of tasks. + +#![warn(clippy::all)] +#![allow(clippy::pedantic)] +#![forbid(unsafe_code)] + +use althea_types::ExitIdentity; +use althea_types::Identity; +use althea_types::Regions; +use althea_types::SystemChain; +use clarity::PrivateKey; +use diesel::RunQueryDsl; +use docopt::Docopt; +use log::{error, info}; +use rita_client_registration::client_db::add_exits_to_registration_list; +use rita_client_registration::{ + client_db::{add_users_to_registered_list, get_all_regsitered_clients}, + register_client_batch_loop::{get_clients_hashset, MAX_BATCH_SIZE}, +}; +use rita_db_migration::{ + get_database_connection, + models::{self, Client}, + schema::clients::dsl::clients, +}; +use serde::Deserialize; +use std::collections::HashSet; +use std::{process::exit, time::Duration}; +use web30::{client::Web3, types::SendTxOption}; + +const WEB3_TIMEOUT: Duration = Duration::from_secs(15); +pub const TX_TIMEOUT: Duration = Duration::from_secs(60); +const EXIT_REGISTRATION_PORT: u16 = 4875; +const EXIT_WG_LISTEN_PORT: u16 = 59998; + +#[derive(Debug, Deserialize)] +pub struct Args { + pub cmd_migrate: bool, + pub cmd_add_exit: bool, + pub flag_dburl: String, + pub flag_address: String, + pub flag_web3url: String, + pub flag_privatekey: String, +} + +#[actix_rt::main] +async fn main() { + env_logger::Builder::default() + .filter(None, log::LevelFilter::Info) + .init(); + + let args: Args = Docopt::new(get_arg_usage()) + .and_then(|d| d.deserialize()) + .unwrap_or_else(|e| e.exit()); + + let db_url = args.flag_dburl; + let contract_addr = args + .flag_address + .parse() + .expect("Please provide a valid eth contract addr"); + let private_key: PrivateKey = args + .flag_privatekey + .parse() + .expect("Please provide a valid eth private key with funds"); + let address = private_key.to_address(); + + let web3 = Web3::new(&args.flag_web3url, WEB3_TIMEOUT); + + if args.cmd_migrate { + // get a copy of all existing clients, we do this in order to handle a potential future edgecase where more than one registration server + // is operating at a time and the same user attempts to register to more than one before the transaction can be sent. Without this check + // once a already registered user is in the queue all future transactions would fail and the server would no longer operate correctly + let all_contract_clients = + match get_all_regsitered_clients(&web3, address, contract_addr).await { + Ok(all_clients) => all_clients, + Err(e) => { + panic!("Failed to get list of already registered clients {:?}", e); + } + }; + let all_contract_clients = get_clients_hashset(all_contract_clients); + + let db_conn = get_database_connection(db_url).unwrap(); + + let database_clients_list = clients.load::(&db_conn).unwrap(); + let database_clients_list = clients_to_ids(database_clients_list); + + let mut clients_to_register = Vec::new(); + for client in database_clients_list { + if !all_contract_clients.contains(&client) { + clients_to_register.push(client); + if clients_to_register.len() > MAX_BATCH_SIZE { + break; + } + } + } + // if there is no one once we filter already registered users + if clients_to_register.is_empty() { + info!("No new clients to register! Successfully exiting"); + exit(0); + } + info!( + "Starting registration of {} clients", + clients_to_register.len() + ); + + while !clients_to_register.is_empty() { + let mut register_batch = Vec::new(); + + // build a small batch to register + while register_batch.len() < MAX_BATCH_SIZE { + if let Some(client) = clients_to_register.pop() { + register_batch.push(client); + } else { + break; + } + } + + info!("Prepped user batch sending register tx"); + match add_users_to_registered_list( + &web3, + register_batch.clone(), + contract_addr, + private_key, + Some(TX_TIMEOUT), + vec![ + SendTxOption::GasPriorityFee(100000000000u128.into()), + SendTxOption::GasMaxFee(400000000000u128.into()), + ], + ) + .await + { + Ok(_) => { + info!( + "Successfully registered {} clients!", + clients_to_register.len() + ); + } + Err(e) => { + error!("Failed to register clients with {:?}, will try again!", e); + for client in register_batch { + clients_to_register.push(client); + } + } + } + } + info!("Successfully migrated all users!"); + } else if args.cmd_add_exit { + let mut xdai = HashSet::new(); + xdai.insert(SystemChain::Xdai); + let mut usa = HashSet::new(); + usa.insert(Regions::US); + + // This command helps generate the bytes for registering a set of exits + let exits_to_register = vec![ + ExitIdentity { + mesh_ip: "fd00::2602:9000".parse().unwrap(), + wg_key: "4PsEKlDEF8gcj9oXtt3Gi+ZmaGuxBwRMxNJ/ewCZpis=" + .parse() + .unwrap(), + eth_addr: "0xdE8236B129Ae270B75DED07101727fB03C39AA5F" + .parse() + .unwrap(), + registration_port: EXIT_REGISTRATION_PORT, + wg_exit_listen_port: EXIT_WG_LISTEN_PORT, + allowed_regions: usa.clone(), + payment_types: xdai.clone(), + }, + ExitIdentity { + mesh_ip: "fd00::2602:3000".parse().unwrap(), + wg_key: "uNu3IMSgt3SY2+MvtEwjEpx45lOk7q/7sWC3ff80GXE=".parse().unwrap(), + eth_addr: "0x72d9E579f691D62aA7e0703840db6dd2fa9fAE21".parse().unwrap(), + registration_port: EXIT_REGISTRATION_PORT, + wg_exit_listen_port: EXIT_WG_LISTEN_PORT, + allowed_regions: usa, + payment_types: xdai, + }, + ]; + match add_exits_to_registration_list( + &web3, + exits_to_register.clone(), + contract_addr, + private_key, + Some(TX_TIMEOUT), + vec![ + SendTxOption::GasPriorityFee(100000000000u128.into()), + SendTxOption::GasMaxFee(400000000000u128.into()), + ], + ) + .await + { + Ok(_) => { + info!("Successfully registered {} exits!", exits_to_register.len()); + } + Err(e) => { + error!("Failed to register exits with {:?}", e); + } + } + } +} + +pub fn get_arg_usage() -> String { + "Usage: + contract-util migrate --dburl= --address=
--web3url= --privatekey= + contract-util add-exit --address=
--web3url= --privatekey= + contract-util (-h | --help) + +Options: + -u, --dburl= Postgresql db url + -a, --address=
Smart Contract address + -w, --web3url= Web3 url + -p, --privatekey= The contract state admin private key + +About: + Utilities for interacting with the Althea exit database contract".to_string() +} + +fn clients_to_ids(client_list: Vec) -> Vec { + let mut res = Vec::new(); + for c in client_list { + res.push(Identity { + mesh_ip: match c.mesh_ip.parse() { + Ok(a) => a, + Err(e) => { + error!("Cannot parse client {:?} mesh ip! with {}", c, e); + continue; + } + }, + eth_address: match c.eth_address.parse() { + Ok(a) => a, + Err(e) => { + error!("Cannot parse client {:?} eth addr! with {}", c, e); + continue; + } + }, + wg_public_key: match c.wg_pubkey.parse() { + Ok(a) => a, + Err(e) => { + error!("Cannot parse client {:?} wg key! with {}", c, e); + continue; + } + }, + nickname: None, + }); + } + res +} diff --git a/rita_bin/src/database_migration.rs b/rita_bin/src/database_migration.rs deleted file mode 100644 index 4c581512d..000000000 --- a/rita_bin/src/database_migration.rs +++ /dev/null @@ -1,88 +0,0 @@ -//! This binary is run to have access to a postgresql db, takes its client info and writes it into -//! smart contract. This is necessary setup to move existing registered clients from the previous -//! sql db format to a smart contract - -#![warn(clippy::all)] -#![allow(clippy::pedantic)] -#![forbid(unsafe_code)] - -use std::{thread, time::Duration}; - -use clarity::PrivateKey; -use docopt::Docopt; -use log::{error, info}; -use rita_client_registration::register_client_batch_loop::register_client_batch_loop; -use rita_db_migration::start_db_migration; -use serde::Deserialize; - -#[derive(Debug, Deserialize)] -pub struct Args { - pub flag_dburl: String, - pub flag_address: String, - pub flag_web3url: String, - pub flag_privatekey: String, -} - -#[actix_rt::main] -async fn main() { - env_logger::Builder::default() - .filter(None, log::LevelFilter::Info) - .init(); - - let args: Args = Docopt::new(get_arg_usage()) - .and_then(|d| d.deserialize()) - .unwrap_or_else(|e| e.exit()); - - info!("Received Args: {:?}", args); - - let db_url = args.flag_dburl; - let contract_addr = args - .flag_address - .parse() - .expect("Please provide a valid eth contract addr"); - let private_key: PrivateKey = args - .flag_privatekey - .parse() - .expect("Please provide a valid eth private key with funds"); - - info!("About to add user admin"); - - thread::sleep(Duration::from_secs(5)); - - info!("About to add start registration loop"); - // Start registration loop - register_client_batch_loop(args.flag_web3url.clone(), contract_addr, private_key); - - thread::sleep(Duration::from_secs(3)); - - info!("About to start db migration loop"); - match start_db_migration( - db_url, - args.flag_web3url, - private_key.to_address(), - contract_addr, - ) - .await - { - Ok(_) => info!( - "Successfully queued all clients for migration! Close the program once this is done" - ), - Err(e) => error!("Failed to migrate clients with {}", e), - } - - thread::sleep(Duration::from_secs(60000)); -} - -pub fn get_arg_usage() -> String { - "Usage: - rita_db_migration --dburl= --address=
--web3url= --privatekey= - -Options: - -u, --dburl= Postgresql db url - -a, --address=
Smart Contract address - -w, --web3url= Web3 url - -p, --privatekey= The contract state admin private key - -About: - Db migration binary".to_string() -} diff --git a/rita_client_registration/src/register_client_batch_loop.rs b/rita_client_registration/src/register_client_batch_loop.rs index 833901108..c527cb258 100644 --- a/rita_client_registration/src/register_client_batch_loop.rs +++ b/rita_client_registration/src/register_client_batch_loop.rs @@ -15,7 +15,7 @@ use web30::{client::Web3, types::SendTxOption}; pub const MAX_BATCH_SIZE: usize = 75; /// Utility function used to easily perform O(1) lookups against the identities list -fn get_clients_hashset(input: Vec) -> HashSet { +pub fn get_clients_hashset(input: Vec) -> HashSet { let mut output = HashSet::new(); for i in input { output.insert(i);