use core::fmt::Debug; use std::{ sync::{Arc, RwLock}, io, collections::HashMap, }; use async_trait::async_trait; use zeroize::Zeroizing; use ciphersuite::{Ciphersuite, Ristretto}; use scale::Decode; use futures::SinkExt; use ::tendermint::{ ext::{BlockNumber, Commit, Block as BlockTrait, Network as NetworkTrait}, SignedMessageFor, SyncedBlock, SyncedBlockSender, MessageSender, TendermintMachine, TendermintHandle, }; mod merkle; pub(crate) use merkle::*; mod transaction; pub use transaction::*; mod provided; pub(crate) use provided::*; mod block; pub use block::*; mod blockchain; pub use blockchain::*; mod mempool; pub use mempool::*; mod tendermint; pub(crate) use crate::tendermint::*; #[cfg(any(test, feature = "tests"))] pub mod tests; pub(crate) const TRANSACTION_MESSAGE: u8 = 0; pub(crate) const TENDERMINT_MESSAGE: u8 = 1; /// An item which can be read and written. pub trait ReadWrite: Sized { fn read(reader: &mut R) -> io::Result; fn write(&self, writer: &mut W) -> io::Result<()>; fn serialize(&self) -> Vec { // BlockHeader is 64 bytes and likely the smallest item in this system let mut buf = Vec::with_capacity(64); self.write(&mut buf).unwrap(); buf } } #[async_trait] pub trait P2p: 'static + Send + Sync + Clone + Debug { async fn broadcast(&self, msg: Vec); } #[async_trait] impl P2p for Arc

{ async fn broadcast(&self, msg: Vec) { (*self).broadcast(msg).await } } pub struct Tributary { network: Network, synced_block: SyncedBlockSender>, messages: MessageSender>, } impl Tributary { pub async fn new( genesis: [u8; 32], start_time: u64, key: Zeroizing<::F>, validators: HashMap<::G, u64>, p2p: P, ) -> Self { let validators_vec = validators.keys().cloned().collect::>(); let signer = Arc::new(Signer::new(genesis, key)); let validators = Arc::new(Validators::new(genesis, validators)); let mut blockchain = Blockchain::new(genesis, &validators_vec); let block_number = blockchain.block_number(); let start_time = start_time; // TODO: Get the start time from the blockchain let proposal = TendermintBlock(blockchain.build_block().serialize()); let blockchain = Arc::new(RwLock::new(blockchain)); let network = Network { genesis, signer, validators, blockchain, p2p }; // The genesis block is 0, so we're working on block #1 let block_number = BlockNumber(block_number + 1); let TendermintHandle { synced_block, messages, machine } = TendermintMachine::new(network.clone(), block_number, start_time, proposal).await; tokio::task::spawn(machine.run()); Self { network, synced_block, messages } } pub fn provide_transaction(&self, tx: T) { self.network.blockchain.write().unwrap().provide_transaction(tx) } // Returns if the transaction was valid. pub async fn add_transaction(&self, tx: T) -> bool { let mut to_broadcast = vec![TRANSACTION_MESSAGE]; tx.write(&mut to_broadcast).unwrap(); let res = self.network.blockchain.write().unwrap().add_transaction(tx); if res { self.network.p2p.broadcast(to_broadcast).await; } res } // Sync a block. // TODO: Since we have a static validator set, we should only need the tail commit? pub async fn sync_block(&mut self, block: Block, commit: Vec) -> bool { let (tip, block_number) = { let blockchain = self.network.blockchain.read().unwrap(); (blockchain.tip(), blockchain.block_number()) }; if block.header.parent != tip { return false; } let block = TendermintBlock(block.serialize()); let Ok(commit) = Commit::>::decode(&mut commit.as_ref()) else { return false; }; if !self.network.verify_commit(block.id(), &commit) { return false; } let number = BlockNumber(block_number + 1); self.synced_block.send(SyncedBlock { number, block, commit }).await.unwrap(); true } // Return true if the message should be rebroadcasted. pub async fn handle_message(&mut self, msg: Vec) -> bool { match msg[0] { TRANSACTION_MESSAGE => { let Ok(tx) = T::read::<&[u8]>(&mut &msg[1 ..]) else { return false; }; // TODO: Sync mempools with fellow peers // Can we just rebroadcast transactions not included for at least two blocks? self.network.blockchain.write().unwrap().add_transaction(tx) } TENDERMINT_MESSAGE => { let Ok(msg) = SignedMessageFor::>::decode::<&[u8]>(&mut &msg[1 ..]) else { return false; }; // If this message isn't to form consensus on the next block, ignore it if msg.block().0 != (self.network.blockchain.read().unwrap().block_number() + 1) { return false; } if !msg.verify_signature(&self.network.validators) { return false; } self.messages.send(msg).await.unwrap(); true } _ => false, } } }