From 9e658cfb7158f7706a3543fb78e4d3ad7fd46614 Mon Sep 17 00:00:00 2001 From: sword-smith Date: Fri, 22 Nov 2024 00:49:23 +0100 Subject: [PATCH] Block: Add test that double-spending blocks are rejected Includes a rewrite of helper function for constructing a new block that makes it easier to create a new block from a BlockPrimitiveWitness. Specifically, a new function `block_template_from_primitive_witness` is added which is used in this test as well as in the call graph for creating a new block from a transaction and a predecessor. Also adds an "unsafe" version of MutatorSetUpdate's "apply" function that allows the caller to calculate a new MS accumulator which ignores double spends. This was needed to allow the BlockPrimitiveWitness constructor to complete the construction of a block with a double- spending transaction. --- src/models/blockchain/block/mod.rs | 74 +++++++++++-------- .../blockchain/block/mutator_set_update.rs | 20 ++++- .../block/validity/block_primitive_witness.rs | 43 +++++++++-- .../block/validity/block_program.rs | 63 +++++++++++++++- 4 files changed, 161 insertions(+), 39 deletions(-) diff --git a/src/models/blockchain/block/mod.rs b/src/models/blockchain/block/mod.rs index 2227bfb25..8bc95f9fc 100644 --- a/src/models/blockchain/block/mod.rs +++ b/src/models/blockchain/block/mod.rs @@ -169,26 +169,26 @@ impl Eq for Block {} impl Block { fn template_header( - predecessor: &Block, + predecessor_header: &BlockHeader, + predecessor_digest: Digest, timestamp: Timestamp, nonce: Digest, target_block_interval: Option, ) -> BlockHeader { let difficulty = difficulty_control( timestamp, - predecessor.header().timestamp, - predecessor.header().difficulty, + predecessor_header.timestamp, + predecessor_header.difficulty, target_block_interval, - predecessor.header().height, + predecessor_header.height, ); let new_cumulative_proof_of_work: ProofOfWork = - predecessor.kernel.header.cumulative_proof_of_work - + predecessor.kernel.header.difficulty; + predecessor_header.cumulative_proof_of_work + predecessor_header.difficulty; BlockHeader { version: BLOCK_HEADER_VERSION, - height: predecessor.kernel.header.height.next(), - prev_block_digest: predecessor.hash(), + height: predecessor_header.height.next(), + prev_block_digest: predecessor_digest, timestamp, nonce, cumulative_proof_of_work: new_cumulative_proof_of_work, @@ -209,8 +209,7 @@ impl Block { ) -> Block { let primitive_witness = BlockPrimitiveWitness::new(predecessor.to_owned(), transaction); let body = primitive_witness.body().to_owned(); - let header = Self::template_header( - predecessor, + let header = primitive_witness.header( block_timestamp, nonce_preimage.hash(), target_block_interval, @@ -220,32 +219,17 @@ impl Block { Block::new(header, body, appendix, proof) } - async fn make_block_template_with_valid_proof( - predecessor: &Block, - transaction: Transaction, - block_timestamp: Timestamp, + pub(crate) async fn block_template_from_primitive_witness( + primitive_witness: BlockPrimitiveWitness, + timestamp: Timestamp, nonce_preimage: Digest, target_block_interval: Option, triton_vm_job_queue: &TritonVmJobQueue, proof_job_options: TritonVmProofJobOptions, ) -> anyhow::Result { - let tx_claim = SingleProof::claim(transaction.kernel.mast_hash()); - assert!( - triton_vm::verify( - Stark::default(), - &tx_claim, - &transaction.proof.clone().into_single_proof() - ), - "Transaction proof must be valid to generate a block" - ); - let primitive_witness = BlockPrimitiveWitness::new(predecessor.to_owned(), transaction); let body = primitive_witness.body().to_owned(); - let header = Self::template_header( - predecessor, - block_timestamp, - nonce_preimage.hash(), - target_block_interval, - ); + let header = + primitive_witness.header(timestamp, nonce_preimage.hash(), target_block_interval); let (appendix, proof) = { let appendix_witness = AppendixWitness::produce(primitive_witness, triton_vm_job_queue).await?; @@ -265,6 +249,36 @@ impl Block { Ok(Block::new(header, body, appendix, proof)) } + async fn make_block_template_with_valid_proof( + predecessor: &Block, + transaction: Transaction, + block_timestamp: Timestamp, + nonce_preimage: Digest, + target_block_interval: Option, + triton_vm_job_queue: &TritonVmJobQueue, + proof_job_options: TritonVmProofJobOptions, + ) -> anyhow::Result { + let tx_claim = SingleProof::claim(transaction.kernel.mast_hash()); + assert!( + triton_vm::verify( + Stark::default(), + &tx_claim, + &transaction.proof.clone().into_single_proof() + ), + "Transaction proof must be valid to generate a block" + ); + let primitive_witness = BlockPrimitiveWitness::new(predecessor.to_owned(), transaction); + Self::block_template_from_primitive_witness( + primitive_witness, + block_timestamp, + nonce_preimage, + target_block_interval, + triton_vm_job_queue, + proof_job_options, + ) + .await + } + /// Compose a block. /// /// Create a block with valid block proof, but without proof-of-work. diff --git a/src/models/blockchain/block/mutator_set_update.rs b/src/models/blockchain/block/mutator_set_update.rs index 393885195..88ad3c47e 100644 --- a/src/models/blockchain/block/mutator_set_update.rs +++ b/src/models/blockchain/block/mutator_set_update.rs @@ -23,6 +23,13 @@ impl MutatorSetUpdate { } } + /// Like `apply_to_accumulator` but does not verify that the removal records + /// could be removed. In other words: This does not check if double spend is + /// happening. + pub(crate) fn apply_to_accumulator_unsafe(&self, ms_accumulator: &mut MutatorSetAccumulator) { + self.apply_to_accumulator_and_records_inner(ms_accumulator, &mut [], false).expect("This function shouldn't be allowed to fail, as we're not checking for double spends") + } + /// Apply a mutator-set-update to a mutator-set-accumulator. /// /// Changes the mutator @@ -32,7 +39,7 @@ impl MutatorSetUpdate { /// /// Returns an error if some removal record could not be removed. pub fn apply_to_accumulator(&self, ms_accumulator: &mut MutatorSetAccumulator) -> Result<()> { - self.apply_to_accumulator_and_records(ms_accumulator, &mut []) + self.apply_to_accumulator_and_records_inner(ms_accumulator, &mut [], true) } /// Apply a mutator-set-update to a mutator-set-accumulator and a bunch of @@ -50,6 +57,15 @@ impl MutatorSetUpdate { &self, ms_accumulator: &mut MutatorSetAccumulator, removal_records: &mut [&mut RemovalRecord], + ) -> Result<()> { + self.apply_to_accumulator_and_records_inner(ms_accumulator, removal_records, true) + } + + fn apply_to_accumulator_and_records_inner( + &self, + ms_accumulator: &mut MutatorSetAccumulator, + removal_records: &mut [&mut RemovalRecord], + check_for_double_spend: bool, ) -> Result<()> { let mut cloned_removals = self.removals.clone(); let mut applied_removal_records = cloned_removals.iter_mut().rev().collect::>(); @@ -69,7 +85,7 @@ impl MutatorSetUpdate { RemovalRecord::batch_update_from_remove(removal_records, applied_removal_record); - if !ms_accumulator.can_remove(applied_removal_record) { + if check_for_double_spend && !ms_accumulator.can_remove(applied_removal_record) { bail!("Cannot remove item from mutator set."); } ms_accumulator.remove(applied_removal_record); diff --git a/src/models/blockchain/block/validity/block_primitive_witness.rs b/src/models/blockchain/block/validity/block_primitive_witness.rs index 8367f0f57..71bae992d 100644 --- a/src/models/blockchain/block/validity/block_primitive_witness.rs +++ b/src/models/blockchain/block/validity/block_primitive_witness.rs @@ -1,11 +1,14 @@ use std::sync::OnceLock; use tasm_lib::twenty_first::prelude::Mmr; +use tasm_lib::Digest; use crate::models::blockchain::block::block_body::BlockBody; +use crate::models::blockchain::block::block_header::BlockHeader; use crate::models::blockchain::block::mutator_set_update::MutatorSetUpdate; use crate::models::blockchain::block::Block; use crate::models::blockchain::transaction::Transaction; +use crate::models::proof_abstractions::timestamp::Timestamp; /// Wraps all information necessary to produce a block. /// @@ -57,20 +60,48 @@ impl BlockPrimitiveWitness { &self.transaction } + pub(crate) fn header( + &self, + timestamp: Timestamp, + nonce: Digest, + target_block_interval: Option, + ) -> BlockHeader { + let parent_header = self.predecessor_block.header(); + let parent_digest = self.predecessor_block.hash(); + Block::template_header( + parent_header, + parent_digest, + timestamp, + nonce, + target_block_interval, + ) + } + + #[cfg(test)] + pub(crate) fn predecessor_block(&self) -> &Block { + &self.predecessor_block + } + pub(crate) fn body(&self) -> &BlockBody { - self.maybe_body.get_or_init(||{ + self.maybe_body.get_or_init(|| { assert_eq!( - self.predecessor_block.mutator_set_accumulator_after().hash(), + self.predecessor_block + .mutator_set_accumulator_after() + .hash(), self.transaction.kernel.mutator_set_hash, "Mutator set of transaction must agree with mutator set after previous block." ); let mut mutator_set = self.predecessor_block.mutator_set_accumulator_after(); - let mutator_set_update = MutatorSetUpdate::new(self.transaction.kernel.inputs.clone(), self.transaction.kernel.outputs.clone()); + let mutator_set_update = MutatorSetUpdate::new( + self.transaction.kernel.inputs.clone(), + self.transaction.kernel.outputs.clone(), + ); - mutator_set_update.apply_to_accumulator(&mut mutator_set).unwrap_or_else(|e| { - panic!("attempting to produce a block body from a transaction whose mutator set update is incompatible: {e:?}"); - }); + // Due to tests, we don't verify that the removal records can be applied. That is + // the caller's responsibility to ensure by e.g. calling block.is_valid() after + // constructing a block. + mutator_set_update.apply_to_accumulator_unsafe(&mut mutator_set); let predecessor_body = self.predecessor_block.body(); let lock_free_mmr = predecessor_body.lock_free_mmr_accumulator.clone(); diff --git a/src/models/blockchain/block/validity/block_program.rs b/src/models/blockchain/block/validity/block_program.rs index 1c2c88c90..eb93282cf 100644 --- a/src/models/blockchain/block/validity/block_program.rs +++ b/src/models/blockchain/block/validity/block_program.rs @@ -179,11 +179,15 @@ pub(crate) mod test { use itertools::Itertools; use tasm_lib::triton_vm::vm::PublicInput; use tracing_test::traced_test; + use triton_vm::prelude::Digest; use super::*; - use crate::job_queue::triton_vm::TritonVmJobQueue; + use crate::job_queue::triton_vm::{TritonVmJobPriority, TritonVmJobQueue}; use crate::models::blockchain::block::validity::block_primitive_witness::test::deterministic_block_primitive_witness; + use crate::models::blockchain::block::{Block, BlockPrimitiveWitness, TritonVmProofJobOptions}; + use crate::models::blockchain::transaction::Transaction; use crate::models::proof_abstractions::mast_hash::MastHash; + use crate::models::proof_abstractions::timestamp::Timestamp; use crate::models::proof_abstractions::SecretWitness; #[traced_test] @@ -228,4 +232,61 @@ pub(crate) mod test { .collect_vec(); assert_eq!(expected_output, tasm_output); } + + #[traced_test] + #[test] + fn disallow_double_spends() { + let current_pw = deterministic_block_primitive_witness(); + let tx = current_pw.transaction().to_owned(); + assert!( + !tx.kernel.inputs.is_empty(), + "Transaction in double-spend test cannot be empty" + ); + let predecessor = current_pw.predecessor_block().to_owned(); + let mock_now = predecessor.header().timestamp + Timestamp::months(12); + + let rt = tokio::runtime::Runtime::new().unwrap(); + let _guard = rt.enter(); + let current_block = rt + .block_on(Block::block_template_from_primitive_witness( + current_pw, + mock_now, + Digest::default(), + None, + &TritonVmJobQueue::dummy(), + TritonVmProofJobOptions::default(), + )) + .unwrap(); + + assert!(current_block.is_valid(&predecessor, mock_now)); + + let mutator_set_update = current_block.mutator_set_update(); + let updated_tx = rt + .block_on( + Transaction::new_with_updated_mutator_set_records_given_proof( + tx.kernel, + &predecessor.mutator_set_accumulator_after(), + &mutator_set_update, + tx.proof.into_single_proof(), + &TritonVmJobQueue::dummy(), + TritonVmJobPriority::default().into(), + ), + ) + .unwrap(); + assert!(rt.block_on(updated_tx.is_valid())); + + let mock_later = mock_now + Timestamp::hours(3); + let next_pw = BlockPrimitiveWitness::new(current_block.clone(), updated_tx); + let next_block = rt + .block_on(Block::block_template_from_primitive_witness( + next_pw, + mock_later, + Digest::default(), + None, + &TritonVmJobQueue::dummy(), + TritonVmProofJobOptions::default(), + )) + .unwrap(); + assert!(!next_block.is_valid(¤t_block, mock_later)); + } }