2023-10-23 18:14:40 +00:00
|
|
|
use std::ops::Deref;
|
|
|
|
use std::{
|
|
|
|
collections::HashSet,
|
|
|
|
future::Future,
|
|
|
|
pin::Pin,
|
|
|
|
sync::Arc,
|
|
|
|
task::{Context, Poll},
|
|
|
|
};
|
|
|
|
|
|
|
|
use futures::FutureExt;
|
2023-10-20 00:04:26 +00:00
|
|
|
use monero_serai::transaction::Transaction;
|
2023-10-23 18:14:40 +00:00
|
|
|
use rayon::prelude::*;
|
|
|
|
use tower::Service;
|
|
|
|
use tracing::instrument;
|
2023-10-20 00:04:26 +00:00
|
|
|
|
2023-10-23 18:14:40 +00:00
|
|
|
use crate::{ConsensusError, Database, HardFork};
|
2023-10-20 00:04:26 +00:00
|
|
|
|
|
|
|
mod inputs;
|
2023-10-23 18:14:40 +00:00
|
|
|
pub(crate) mod outputs;
|
|
|
|
mod ring;
|
|
|
|
mod sigs;
|
2023-10-24 19:17:16 +00:00
|
|
|
mod time_lock;
|
2023-10-20 00:04:26 +00:00
|
|
|
|
2023-10-23 18:14:40 +00:00
|
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
|
2023-10-20 00:04:26 +00:00
|
|
|
pub enum TxVersion {
|
|
|
|
RingSignatures,
|
|
|
|
RingCT,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TxVersion {
|
|
|
|
pub fn from_raw(version: u64) -> Result<TxVersion, ConsensusError> {
|
|
|
|
match version {
|
|
|
|
1 => Ok(TxVersion::RingSignatures),
|
|
|
|
2 => Ok(TxVersion::RingCT),
|
|
|
|
_ => Err(ConsensusError::TransactionVersionInvalid),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Data needed to verify a transaction.
|
|
|
|
///
|
2023-10-23 18:14:40 +00:00
|
|
|
#[derive(Debug)]
|
2023-10-20 00:04:26 +00:00
|
|
|
pub struct TransactionVerificationData {
|
2023-10-23 18:14:40 +00:00
|
|
|
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<ring::TxRingMembersInfo>>,
|
2023-10-20 00:04:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TransactionVerificationData {
|
2023-10-23 18:14:40 +00:00
|
|
|
pub fn new(tx: Transaction) -> Result<TransactionVerificationData, ConsensusError> {
|
2023-10-20 00:04:26 +00:00
|
|
|
Ok(TransactionVerificationData {
|
2023-10-23 18:14:40 +00:00
|
|
|
tx_hash: tx.hash(),
|
|
|
|
tx_blob: tx.serialize(),
|
2023-10-20 00:04:26 +00:00
|
|
|
tx_weight: tx.weight(),
|
2023-10-23 18:14:40 +00:00
|
|
|
fee: tx.rct_signatures.base.fee,
|
|
|
|
rings_member_info: std::sync::Mutex::new(None),
|
2023-10-20 00:04:26 +00:00
|
|
|
version: TxVersion::from_raw(tx.prefix.version)?,
|
|
|
|
tx,
|
|
|
|
})
|
|
|
|
}
|
2023-10-23 18:14:40 +00:00
|
|
|
}
|
2023-10-20 00:04:26 +00:00
|
|
|
|
2023-10-23 18:14:40 +00:00
|
|
|
pub enum VerifyTxRequest {
|
|
|
|
/// Verifies transactions in the context of a block.
|
|
|
|
Block {
|
|
|
|
txs: Vec<Arc<TransactionVerificationData>>,
|
|
|
|
current_chain_height: u64,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock: u64,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf: HardFork,
|
|
|
|
},
|
2023-10-26 02:16:03 +00:00
|
|
|
/// Batches the setup of [`TransactionVerificationData`].
|
|
|
|
BatchSetup { txs: Vec<Transaction>, hf: HardFork },
|
2023-10-23 18:14:40 +00:00
|
|
|
/// Batches the setup of [`TransactionVerificationData`] and verifies the transactions
|
|
|
|
/// in the context of a block.
|
|
|
|
BatchSetupVerifyBlock {
|
2023-10-20 00:04:26 +00:00
|
|
|
txs: Vec<Transaction>,
|
2023-10-23 18:14:40 +00:00
|
|
|
current_chain_height: u64,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock: u64,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf: HardFork,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
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 = ConsensusError;
|
|
|
|
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,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf,
|
2023-10-24 19:17:16 +00:00
|
|
|
} => verify_transactions_for_block(
|
|
|
|
database,
|
|
|
|
txs,
|
|
|
|
current_chain_height,
|
|
|
|
time_for_time_lock,
|
|
|
|
hf,
|
|
|
|
)
|
|
|
|
.boxed(),
|
2023-10-28 23:39:22 +00:00
|
|
|
VerifyTxRequest::BatchSetup { txs, hf } => {
|
|
|
|
batch_setup_transactions(database, txs, hf).boxed()
|
|
|
|
}
|
2023-10-23 18:14:40 +00:00
|
|
|
VerifyTxRequest::BatchSetupVerifyBlock {
|
|
|
|
txs,
|
|
|
|
current_chain_height,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf,
|
2023-10-24 19:17:16 +00:00
|
|
|
} => batch_setup_verify_transactions_for_block(
|
|
|
|
database,
|
|
|
|
txs,
|
|
|
|
current_chain_height,
|
|
|
|
time_for_time_lock,
|
|
|
|
hf,
|
|
|
|
)
|
|
|
|
.boxed(),
|
2023-10-23 18:14:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn set_missing_ring_members<D>(
|
|
|
|
database: D,
|
|
|
|
txs: &[Arc<TransactionVerificationData>],
|
|
|
|
hf: &HardFork,
|
|
|
|
) -> Result<(), ConsensusError>
|
|
|
|
where
|
|
|
|
D: Database + Clone + Sync + Send + 'static,
|
|
|
|
{
|
|
|
|
// TODO: handle re-orgs.
|
|
|
|
|
|
|
|
let txs_needing_ring_members = txs
|
|
|
|
.iter()
|
|
|
|
// Safety: we must not hold the mutex lock for long to not block the async runtime.
|
|
|
|
.filter(|tx| tx.rings_member_info.lock().unwrap().is_none())
|
|
|
|
.cloned()
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
tracing::debug!(
|
|
|
|
"Retrieving ring members for {} txs",
|
|
|
|
txs_needing_ring_members.len()
|
|
|
|
);
|
|
|
|
|
|
|
|
ring::batch_fill_ring_member_info(&txs_needing_ring_members, hf, database).await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-10-26 02:16:03 +00:00
|
|
|
async fn batch_setup_transactions<D>(
|
|
|
|
database: D,
|
|
|
|
txs: Vec<Transaction>,
|
|
|
|
hf: HardFork,
|
|
|
|
) -> Result<VerifyTxResponse, ConsensusError>
|
2023-10-28 23:39:22 +00:00
|
|
|
where
|
|
|
|
D: Database + Clone + Sync + Send + 'static,
|
2023-10-26 02:16:03 +00:00
|
|
|
{
|
|
|
|
// Move out of the async runtime and use rayon to parallelize the serialisation and hashing of the txs.
|
|
|
|
let txs = tokio::task::spawn_blocking(|| {
|
|
|
|
txs.into_par_iter()
|
|
|
|
.map(|tx| Ok(Arc::new(TransactionVerificationData::new(tx)?)))
|
|
|
|
.collect::<Result<Vec<_>, ConsensusError>>()
|
|
|
|
})
|
2023-10-28 23:39:22 +00:00
|
|
|
.await
|
|
|
|
.unwrap()?;
|
2023-10-26 02:16:03 +00:00
|
|
|
|
|
|
|
set_missing_ring_members(database, &txs, &hf).await?;
|
|
|
|
|
|
|
|
Ok(VerifyTxResponse::BatchSetupOk(txs))
|
|
|
|
}
|
|
|
|
|
2023-10-23 18:14:40 +00:00
|
|
|
async fn batch_setup_verify_transactions_for_block<D>(
|
|
|
|
database: D,
|
|
|
|
txs: Vec<Transaction>,
|
|
|
|
current_chain_height: u64,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock: u64,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf: HardFork,
|
|
|
|
) -> Result<VerifyTxResponse, ConsensusError>
|
|
|
|
where
|
|
|
|
D: Database + Clone + Sync + Send + 'static,
|
|
|
|
{
|
|
|
|
// Move out of the async runtime and use rayon to parallelize the serialisation and hashing of the txs.
|
|
|
|
let txs = tokio::task::spawn_blocking(|| {
|
|
|
|
txs.into_par_iter()
|
|
|
|
.map(|tx| Ok(Arc::new(TransactionVerificationData::new(tx)?)))
|
|
|
|
.collect::<Result<Vec<_>, ConsensusError>>()
|
|
|
|
})
|
|
|
|
.await
|
|
|
|
.unwrap()?;
|
|
|
|
|
2023-10-24 19:17:16 +00:00
|
|
|
verify_transactions_for_block(
|
|
|
|
database,
|
|
|
|
txs.clone(),
|
|
|
|
current_chain_height,
|
|
|
|
time_for_time_lock,
|
|
|
|
hf,
|
|
|
|
)
|
|
|
|
.await?;
|
2023-10-23 18:14:40 +00:00
|
|
|
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,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock: u64,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf: HardFork,
|
|
|
|
) -> Result<VerifyTxResponse, ConsensusError>
|
|
|
|
where
|
|
|
|
D: Database + Clone + Sync + Send + 'static,
|
|
|
|
{
|
2023-10-24 01:25:11 +00:00
|
|
|
tracing::debug!("Verifying transactions for block, amount: {}", txs.len());
|
2023-10-23 18:14:40 +00:00
|
|
|
|
|
|
|
set_missing_ring_members(database, &txs, &hf).await?;
|
|
|
|
|
|
|
|
let spent_kis = Arc::new(std::sync::Mutex::new(HashSet::new()));
|
|
|
|
|
|
|
|
tokio::task::spawn_blocking(move || {
|
|
|
|
txs.par_iter().try_for_each(|tx| {
|
2023-10-24 19:17:16 +00:00
|
|
|
verify_transaction_for_block(
|
|
|
|
tx,
|
|
|
|
current_chain_height,
|
|
|
|
time_for_time_lock,
|
|
|
|
hf,
|
|
|
|
spent_kis.clone(),
|
|
|
|
)
|
2023-10-23 18:14:40 +00:00
|
|
|
})
|
2023-10-24 22:02:19 +00:00
|
|
|
})
|
|
|
|
.await
|
|
|
|
.unwrap()?;
|
2023-10-23 18:14:40 +00:00
|
|
|
|
|
|
|
Ok(VerifyTxResponse::Ok)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn verify_transaction_for_block(
|
|
|
|
tx_verification_data: &TransactionVerificationData,
|
|
|
|
current_chain_height: u64,
|
2023-10-24 19:17:16 +00:00
|
|
|
time_for_time_lock: u64,
|
2023-10-23 18:14:40 +00:00
|
|
|
hf: HardFork,
|
|
|
|
spent_kis: Arc<std::sync::Mutex<HashSet<[u8; 32]>>>,
|
|
|
|
) -> Result<(), ConsensusError> {
|
2023-10-24 19:17:16 +00:00
|
|
|
tracing::debug!(
|
2023-10-23 18:14:40 +00:00
|
|
|
"Verifying transaction: {}",
|
|
|
|
hex::encode(tx_verification_data.tx_hash)
|
|
|
|
);
|
|
|
|
|
|
|
|
let tx_version = &tx_verification_data.version;
|
|
|
|
|
|
|
|
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!"),
|
|
|
|
};
|
|
|
|
|
2023-10-24 19:17:16 +00:00
|
|
|
check_tx_version(&rings_member_info.decoy_info, tx_version, &hf)?;
|
|
|
|
|
|
|
|
time_lock::check_all_time_locks(
|
|
|
|
&rings_member_info.time_locked_outs,
|
|
|
|
current_chain_height,
|
|
|
|
time_for_time_lock,
|
|
|
|
&hf,
|
|
|
|
)?;
|
2023-10-23 18:14:40 +00:00
|
|
|
|
|
|
|
let sum_outputs =
|
|
|
|
outputs::check_outputs(&tx_verification_data.tx.prefix.outputs, &hf, tx_version)?;
|
|
|
|
|
|
|
|
let sum_inputs = inputs::check_inputs(
|
|
|
|
&tx_verification_data.tx.prefix.inputs,
|
|
|
|
rings_member_info,
|
|
|
|
current_chain_height,
|
|
|
|
&hf,
|
|
|
|
tx_version,
|
|
|
|
spent_kis,
|
|
|
|
)?;
|
|
|
|
|
2023-10-24 19:17:16 +00:00
|
|
|
if tx_version == &TxVersion::RingSignatures {
|
|
|
|
if sum_outputs >= sum_inputs {
|
|
|
|
return Err(ConsensusError::TransactionOutputsTooMuch);
|
|
|
|
}
|
|
|
|
// check that monero-serai is calculating the correct value here, why can't we just use this
|
|
|
|
// value? because we don't have this when we create the object.
|
|
|
|
assert_eq!(tx_verification_data.fee, sum_inputs - sum_outputs);
|
|
|
|
}
|
|
|
|
|
2023-10-23 18:14:40 +00:00
|
|
|
sigs::verify_signatures(&tx_verification_data.tx, &rings_member_info.rings)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks the version is in the allowed range.
|
|
|
|
///
|
|
|
|
/// https://cuprate.github.io/monero-book/consensus_rules/transactions.html#version
|
|
|
|
fn check_tx_version(
|
|
|
|
decoy_info: &Option<ring::DecoyInfo>,
|
|
|
|
version: &TxVersion,
|
|
|
|
hf: &HardFork,
|
|
|
|
) -> Result<(), ConsensusError> {
|
|
|
|
if let Some(decoy_info) = decoy_info {
|
|
|
|
let max = max_tx_version(hf);
|
|
|
|
if version > &max {
|
|
|
|
return Err(ConsensusError::TransactionVersionInvalid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Doc is wrong here
|
|
|
|
let min = min_tx_version(hf);
|
|
|
|
if version < &min && decoy_info.not_mixable != 0 {
|
|
|
|
return Err(ConsensusError::TransactionVersionInvalid);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// This will only happen for hard-fork 1 when only RingSignatures are allowed.
|
|
|
|
if version != &TxVersion::RingSignatures {
|
|
|
|
return Err(ConsensusError::TransactionVersionInvalid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn max_tx_version(hf: &HardFork) -> TxVersion {
|
|
|
|
if hf <= &HardFork::V3 {
|
|
|
|
TxVersion::RingSignatures
|
|
|
|
} else {
|
|
|
|
TxVersion::RingCT
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn min_tx_version(hf: &HardFork) -> TxVersion {
|
|
|
|
if hf >= &HardFork::V6 {
|
|
|
|
TxVersion::RingCT
|
|
|
|
} else {
|
|
|
|
TxVersion::RingSignatures
|
2023-10-20 00:04:26 +00:00
|
|
|
}
|
|
|
|
}
|