use std::{sync::{Arc, RwLock}, collections::HashMap}; use async_trait::async_trait; use rand::rngs::OsRng; use group::Group; use frost::curve::Curve; use crate::{ NetworkError, Network, Coin, coins::monero::Monero, wallet::{WalletKeys, MemCoinDb, Wallet} }; #[derive(Clone)] struct LocalNetwork { i: u16, size: u16, round: usize, rounds: Arc>>>> } impl LocalNetwork { fn new(size: u16) -> Vec { let rounds = Arc::new(RwLock::new(vec![])); let mut res = vec![]; for i in 1 ..= size { res.push(LocalNetwork { i, size, round: 0, rounds: rounds.clone() }); } res } } #[async_trait] impl Network for LocalNetwork { async fn round(&mut self, data: Vec) -> Result>, NetworkError> { { let mut rounds = self.rounds.write().unwrap(); if rounds.len() == self.round { rounds.push(HashMap::new()); } rounds[self.round].insert(self.i, data); } while { let read = self.rounds.try_read().unwrap(); read[self.round].len() != usize::from(self.size) } { tokio::task::yield_now().await; } let res = self.rounds.try_read().unwrap()[self.round].clone(); self.round += 1; Ok(res) } } async fn test_send(coin: C, fee: C::Fee) { // Mine a block so there's a confirmed height coin.mine_block(coin.address(::G::generator())).await; let height = coin.get_height().await.unwrap(); let mut keys = frost::tests::key_gen::<_, C::Curve>(&mut OsRng); let threshold = keys[&1].params().t(); let mut networks = LocalNetwork::new(threshold); let mut wallets = vec![]; for i in 1 ..= threshold { let mut wallet = Wallet::new(MemCoinDb::new(), coin.clone()); wallet.acknowledge_height(0, height); wallet.add_keys( &WalletKeys::new(Arc::try_unwrap(keys.remove(&i).take().unwrap()).unwrap(), 0) ); wallets.push(wallet); } // Get the chain to a height where blocks have sufficient confirmations while (height + C::CONFIRMATIONS) > coin.get_height().await.unwrap() { coin.mine_block(coin.address(::G::generator())).await; } for wallet in wallets.iter_mut() { // Poll to activate the keys wallet.poll().await.unwrap(); } coin.test_send(wallets[0].address()).await; let mut futures = vec![]; for (network, wallet) in networks.iter_mut().zip(wallets.iter_mut()) { wallet.poll().await.unwrap(); let height = coin.get_height().await.unwrap(); wallet.acknowledge_height(1, height - 10); let signable = wallet.prepare_sends( 1, vec![(wallet.address(), 10000000000)], fee ).await.unwrap().1.swap_remove(0); futures.push( wallet.attempt_send(network, signable, (1 ..= threshold).into_iter().collect::>()) ); } println!( "{:?}", hex::encode(futures::future::join_all(futures).await.swap_remove(0).unwrap().0) ); } #[tokio::test] async fn monero() { let monero = Monero::new("http://127.0.0.1:18081".to_string()); let fee = monero.rpc.get_fee().await.unwrap(); test_send(monero, fee).await; }