mirror of
https://github.com/hinto-janai/cuprate.git
synced 2025-01-24 11:26:03 +00:00
278 lines
8 KiB
Rust
278 lines
8 KiB
Rust
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<u8>,
|
|
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<Option<(TxRingMembersInfo, ReOrgToken)>>,
|
|
}
|
|
|
|
impl TransactionVerificationData {
|
|
pub fn new(
|
|
tx: Transaction,
|
|
hf: &HardFork,
|
|
verifier: Arc<MultiThreadedBatchVerifier>,
|
|
) -> Result<TransactionVerificationData, ConsensusError> {
|
|
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<Arc<TransactionVerificationData>>,
|
|
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<Transaction>,
|
|
hf: HardFork,
|
|
re_org_token: ReOrgToken,
|
|
},
|
|
}
|
|
|
|
pub enum VerifyTxResponse {
|
|
BatchSetupOk(Vec<Arc<TransactionVerificationData>>),
|
|
Ok,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
pub struct TxVerifierService<D: Clone> {
|
|
database: D,
|
|
}
|
|
|
|
impl<D> TxVerifierService<D>
|
|
where
|
|
D: Database + Clone + Send + 'static,
|
|
D::Future: Send + 'static,
|
|
{
|
|
pub fn new(database: D) -> TxVerifierService<D> {
|
|
TxVerifierService { database }
|
|
}
|
|
}
|
|
|
|
impl<D> Service<VerifyTxRequest> for TxVerifierService<D>
|
|
where
|
|
D: Database + Clone + Send + Sync + 'static,
|
|
D::Future: Send + 'static,
|
|
{
|
|
type Response = VerifyTxResponse;
|
|
type Error = ExtendedConsensusError;
|
|
type Future =
|
|
Pin<Box<dyn Future<Output = Result<Self::Response, Self::Error>> + Send + 'static>>;
|
|
|
|
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
|
|
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<D>(
|
|
database: D,
|
|
txs: Vec<Transaction>,
|
|
hf: HardFork,
|
|
re_org_token: ReOrgToken,
|
|
) -> Result<VerifyTxResponse, ExtendedConsensusError>
|
|
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::<Result<Vec<_>, 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<D>(
|
|
database: D,
|
|
txs: Vec<Arc<TransactionVerificationData>>,
|
|
current_chain_height: u64,
|
|
time_for_time_lock: u64,
|
|
hf: HardFork,
|
|
re_org_token: ReOrgToken,
|
|
) -> Result<VerifyTxResponse, ExtendedConsensusError>
|
|
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<std::sync::Mutex<HashSet<[u8; 32]>>>,
|
|
) -> 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(())
|
|
}
|