use std::{ collections::HashSet, future::Future, ops::Deref, pin::Pin, sync::Arc, task::{Context, Poll}, }; use futures::FutureExt; use monero_serai::ringct::RctType; use monero_serai::transaction::Transaction; use rayon::prelude::*; use tower::{Service, ServiceExt}; use tracing::instrument; use monero_consensus::{ transactions::{ check_transaction_contextual, check_transaction_semantic, RingCTError, TransactionError, TxRingMembersInfo, }, ConsensusError, HardFork, TxVersion, }; use crate::{ batch_verifier::MultiThreadedBatchVerifier, context::ReOrgToken, helper::rayon_spawn_async, Database, DatabaseRequest, DatabaseResponse, ExtendedConsensusError, }; mod contextual_data; /// Data needed to verify a transaction. /// #[derive(Debug)] pub struct TransactionVerificationData { pub tx: Transaction, pub version: TxVersion, pub tx_blob: Vec, pub tx_weight: usize, pub fee: u64, pub tx_hash: [u8; 32], /// We put this behind a mutex as the information is not constant and is based of past outputs idxs /// which could change on re-orgs. rings_member_info: std::sync::Mutex>, } impl TransactionVerificationData { pub fn new( tx: Transaction, hf: &HardFork, verifier: Arc, ) -> Result { let tx_hash = tx.hash(); let tx_blob = tx.serialize(); // the tx weight is only different from the blobs length for bp(+) txs. let tx_weight = match tx.rct_signatures.rct_type() { RctType::Bulletproofs | RctType::BulletproofsCompactAmount | RctType::Clsag | RctType::BulletproofsPlus => tx.weight(), _ => tx_blob.len(), }; let fee = verifier.queue_statement(|verifier| { check_transaction_semantic(&tx, tx_blob.len(), tx_weight, &tx_hash, hf, verifier) .map_err(ConsensusError::Transaction) })?; Ok(TransactionVerificationData { tx_hash, tx_blob, tx_weight, fee, rings_member_info: std::sync::Mutex::new(None), version: TxVersion::from_raw(tx.prefix.version) .ok_or(TransactionError::TransactionVersionInvalid)?, tx, }) } } pub enum VerifyTxRequest { /// Verifies transactions in the context of a block. Block { txs: Vec>, current_chain_height: u64, time_for_time_lock: u64, hf: HardFork, re_org_token: ReOrgToken, }, /// Batches the setup of [`TransactionVerificationData`], does *some* verification, you need to call [`VerifyTxRequest::Block`] /// with the returned data. BatchSetup { txs: Vec, hf: HardFork, re_org_token: ReOrgToken, }, } pub enum VerifyTxResponse { BatchSetupOk(Vec>), Ok, } #[derive(Clone)] pub struct TxVerifierService { database: D, } impl TxVerifierService where D: Database + Clone + Send + 'static, D::Future: Send + 'static, { pub fn new(database: D) -> TxVerifierService { TxVerifierService { database } } } impl Service for TxVerifierService where D: Database + Clone + Send + Sync + 'static, D::Future: Send + 'static, { type Response = VerifyTxResponse; type Error = ExtendedConsensusError; type Future = Pin> + Send + 'static>>; fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll> { self.database.poll_ready(cx).map_err(Into::into) } fn call(&mut self, req: VerifyTxRequest) -> Self::Future { let database = self.database.clone(); match req { VerifyTxRequest::Block { txs, current_chain_height, time_for_time_lock, hf, re_org_token, } => verify_transactions_for_block( database, txs, current_chain_height, time_for_time_lock, hf, re_org_token, ) .boxed(), VerifyTxRequest::BatchSetup { txs, hf, re_org_token, } => batch_setup_transactions(database, txs, hf, re_org_token).boxed(), } } } async fn batch_setup_transactions( database: D, txs: Vec, hf: HardFork, re_org_token: ReOrgToken, ) -> Result where D: Database + Clone + Sync + Send + 'static, { let batch_verifier = Arc::new(MultiThreadedBatchVerifier::new(rayon::current_num_threads())); let cloned_verifier = batch_verifier.clone(); // Move out of the async runtime and use rayon to parallelize the serialisation and hashing of the txs. let txs = rayon_spawn_async(move || { txs.into_par_iter() .map(|tx| { Ok(Arc::new(TransactionVerificationData::new( tx, &hf, cloned_verifier.clone(), )?)) }) .collect::, ConsensusError>>() }) .await?; if !Arc::into_inner(batch_verifier).unwrap().verify() { Err(ConsensusError::Transaction(TransactionError::RingCTError( RingCTError::BulletproofsRangeInvalid, )))? } contextual_data::batch_fill_ring_member_info(&txs, &hf, re_org_token, database).await?; Ok(VerifyTxResponse::BatchSetupOk(txs)) } #[instrument(name = "verify_txs", skip_all, level = "info")] async fn verify_transactions_for_block( database: D, txs: Vec>, current_chain_height: u64, time_for_time_lock: u64, hf: HardFork, re_org_token: ReOrgToken, ) -> Result where D: Database + Clone + Sync + Send + 'static, { tracing::debug!("Verifying transactions for block, amount: {}", txs.len()); contextual_data::batch_refresh_ring_member_info(&txs, &hf, re_org_token, database.clone()) .await?; let spent_kis = Arc::new(std::sync::Mutex::new(HashSet::new())); let cloned_spent_kis = spent_kis.clone(); rayon_spawn_async(move || { txs.par_iter().try_for_each(|tx| { verify_transaction_for_block( tx, current_chain_height, time_for_time_lock, hf, cloned_spent_kis.clone(), ) }) }) .await?; let DatabaseResponse::CheckKIsNotSpent(kis_spent) = database .oneshot(DatabaseRequest::CheckKIsNotSpent( Arc::into_inner(spent_kis).unwrap().into_inner().unwrap(), )) .await? else { panic!("Database sent incorrect response!"); }; if kis_spent { Err(ConsensusError::Transaction(TransactionError::KeyImageSpent))?; } Ok(VerifyTxResponse::Ok) } fn verify_transaction_for_block( tx_verification_data: &TransactionVerificationData, current_chain_height: u64, time_for_time_lock: u64, hf: HardFork, spent_kis: Arc>>, ) -> Result<(), ConsensusError> { tracing::debug!( "Verifying transaction: {}", hex::encode(tx_verification_data.tx_hash) ); let rings_member_info_lock = tx_verification_data.rings_member_info.lock().unwrap(); let rings_member_info = match rings_member_info_lock.deref() { Some(rings_member_info) => rings_member_info, None => panic!("rings_member_info needs to be set to be able to verify!"), }; check_transaction_contextual( &tx_verification_data.tx, &rings_member_info.0, current_chain_height, time_for_time_lock, &hf, spent_kis, )?; Ok(()) }