diff --git a/crates/chain/src/tx_graph.rs b/crates/chain/src/tx_graph.rs index 4dd7f0a25..34cbccf5c 100644 --- a/crates/chain/src/tx_graph.rs +++ b/crates/chain/src/tx_graph.rs @@ -454,6 +454,21 @@ impl TxGraph { } } +impl TxGraph { + /// Transform the [`TxGraph`] to have [`Anchor`]s of another type. + /// + /// This takes in a closure of signature `FnMut(A) -> A2` which is called for each [`Anchor`] to + /// transform it. + pub fn map_anchors(self, f: F) -> TxGraph + where + F: FnMut(A) -> A2, + { + let mut new_graph = TxGraph::::default(); + new_graph.apply_changeset(self.initial_changeset().map_anchors(f)); + new_graph + } +} + impl TxGraph { /// Construct a new [`TxGraph`] from a list of transactions. pub fn new(txs: impl IntoIterator) -> Self { @@ -1294,6 +1309,26 @@ impl Append for ChangeSet { } } +impl ChangeSet { + /// Transform the [`ChangeSet`] to have [`Anchor`]s of another type. + /// + /// This takes in a closure of signature `FnMut(A) -> A2` which is called for each [`Anchor`] to + /// transform it. + pub fn map_anchors(self, mut f: F) -> ChangeSet + where + F: FnMut(A) -> A2, + { + ChangeSet { + txs: self.txs, + txouts: self.txouts, + anchors: BTreeSet::<(A2, Txid)>::from_iter( + self.anchors.into_iter().map(|(a, txid)| (f(a), txid)), + ), + last_seen: self.last_seen, + } + } +} + impl AsRef> for TxGraph { fn as_ref(&self) -> &TxGraph { self diff --git a/crates/chain/tests/common/tx_template.rs b/crates/chain/tests/common/tx_template.rs index 605a0ba7e..ec2eb1159 100644 --- a/crates/chain/tests/common/tx_template.rs +++ b/crates/chain/tests/common/tx_template.rs @@ -1,7 +1,7 @@ use rand::distributions::{Alphanumeric, DistString}; use std::collections::HashMap; -use bdk_chain::{tx_graph::TxGraph, BlockId, SpkTxOutIndex}; +use bdk_chain::{tx_graph::TxGraph, Anchor, SpkTxOutIndex}; use bitcoin::{ locktime::absolute::LockTime, secp256k1::Secp256k1, OutPoint, ScriptBuf, Sequence, Transaction, TxIn, TxOut, Txid, Witness, @@ -49,11 +49,11 @@ impl TxOutTemplate { } #[allow(dead_code)] -pub fn init_graph<'a>( - tx_templates: impl IntoIterator>, -) -> (TxGraph, SpkTxOutIndex, HashMap<&'a str, Txid>) { +pub fn init_graph<'a, A: Anchor + Clone + 'a>( + tx_templates: impl IntoIterator>, +) -> (TxGraph, SpkTxOutIndex, HashMap<&'a str, Txid>) { let (descriptor, _) = Descriptor::parse_descriptor(&Secp256k1::signing_only(), "tr(tprv8ZgxMBicQKsPd3krDUsBAmtnRsK3rb8u5yi1zhQgMhF1tR8MW7xfE4rnrbbsrbPR52e7rKapu6ztw1jXveJSCGHEriUGZV7mCe88duLp5pj/86'/1'/0'/0/*)").unwrap(); - let mut graph = TxGraph::::default(); + let mut graph = TxGraph::::default(); let mut spk_index = SpkTxOutIndex::default(); (0..10).for_each(|index| { spk_index.insert_spk( @@ -126,7 +126,7 @@ pub fn init_graph<'a>( spk_index.scan(&tx); let _ = graph.insert_tx(tx.clone()); for anchor in tx_tmp.anchors.iter() { - let _ = graph.insert_anchor(tx.txid(), *anchor); + let _ = graph.insert_anchor(tx.txid(), anchor.clone()); } if let Some(seen_at) = tx_tmp.last_seen { let _ = graph.insert_seen_at(tx.txid(), seen_at); diff --git a/crates/chain/tests/test_tx_graph.rs b/crates/chain/tests/test_tx_graph.rs index 4afdd66e6..37e8c7192 100644 --- a/crates/chain/tests/test_tx_graph.rs +++ b/crates/chain/tests/test_tx_graph.rs @@ -10,7 +10,9 @@ use bdk_chain::{ use bitcoin::{ absolute, hashes::Hash, BlockHash, OutPoint, ScriptBuf, Transaction, TxIn, TxOut, Txid, }; +use common::*; use core::iter; +use rand::RngCore; use std::vec; #[test] @@ -1178,3 +1180,86 @@ fn test_missing_blocks() { ), ]); } + +#[test] +/// The `map_anchors` allow a caller to pass a function to reconstruct the [`TxGraph`] with any [`Anchor`], +/// even though the function is non-deterministic. +fn call_map_anchors_with_non_deterministic_anchor() { + #[derive(Debug, Default, Clone, PartialEq, Eq, Copy, PartialOrd, Ord, core::hash::Hash)] + /// A non-deterministic anchor + pub struct NonDeterministicAnchor { + pub anchor_block: BlockId, + pub non_deterministic_field: u32, + } + + let template = [ + TxTemplate { + tx_name: "tx1", + inputs: &[TxInTemplate::Bogus], + outputs: &[TxOutTemplate::new(10000, Some(1))], + anchors: &[block_id!(1, "A")], + last_seen: None, + }, + TxTemplate { + tx_name: "tx2", + inputs: &[TxInTemplate::PrevTx("tx1", 0)], + outputs: &[TxOutTemplate::new(20000, Some(2))], + anchors: &[block_id!(2, "B")], + ..Default::default() + }, + TxTemplate { + tx_name: "tx3", + inputs: &[TxInTemplate::PrevTx("tx2", 0)], + outputs: &[TxOutTemplate::new(30000, Some(3))], + anchors: &[block_id!(3, "C"), block_id!(4, "D")], + ..Default::default() + }, + ]; + let (graph, _, _) = init_graph(&template); + let new_graph = graph.clone().map_anchors(|a| NonDeterministicAnchor { + anchor_block: a, + // A non-deterministic value + non_deterministic_field: rand::thread_rng().next_u32(), + }); + + // Check all the details in new_graph reconstruct as well + + let mut full_txs_vec: Vec<_> = graph.full_txs().collect(); + full_txs_vec.sort(); + let mut new_txs_vec: Vec<_> = new_graph.full_txs().collect(); + new_txs_vec.sort(); + let mut new_txs = new_txs_vec.iter(); + + for tx_node in full_txs_vec.iter() { + let new_txnode = new_txs.next().unwrap(); + assert_eq!(new_txnode.txid, tx_node.txid); + assert_eq!(new_txnode.tx, tx_node.tx); + assert_eq!( + new_txnode.last_seen_unconfirmed, + tx_node.last_seen_unconfirmed + ); + assert_eq!(new_txnode.anchors.len(), tx_node.anchors.len()); + + let mut new_anchors: Vec<_> = new_txnode.anchors.iter().map(|a| a.anchor_block).collect(); + new_anchors.sort(); + let mut old_anchors: Vec<_> = tx_node.anchors.iter().copied().collect(); + old_anchors.sort(); + assert_eq!(new_anchors, old_anchors); + } + assert!(new_txs.next().is_none()); + + let new_graph_anchors: Vec<_> = new_graph + .all_anchors() + .iter() + .map(|i| i.0.anchor_block) + .collect(); + assert_eq!( + new_graph_anchors, + vec![ + block_id!(1, "A"), + block_id!(2, "B"), + block_id!(3, "C"), + block_id!(4, "D"), + ] + ); +}