From ffc82fa32c450efdbb9b5768bfb12d7c3f7d9396 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=BF=97=E5=AE=87?= Date: Thu, 12 Oct 2023 00:03:18 +0800 Subject: [PATCH] ref(chain): move `keychain::ChangeSet` into `txout_index.rs` We plan to record `Descriptor` additions into persistence. Hence, we need to add `Descriptor`s to the changeset. This depends on `miniscript`. Moving this into `txout_index.rs` makes sense as this is consistent with all the other files. The only reason why this wasn't this way before, is because the changeset didn't need miniscript. Co-Authored-By: Daniela Brozzoni --- crates/chain/src/indexed_tx_graph.rs | 1 + crates/chain/src/keychain.rs | 102 ------------------ crates/chain/src/keychain/txout_index.rs | 63 +++++++++++ .../chain/tests/test_keychain_txout_index.rs | 34 +++++- 4 files changed, 97 insertions(+), 103 deletions(-) diff --git a/crates/chain/src/indexed_tx_graph.rs b/crates/chain/src/indexed_tx_graph.rs index 0e2620e0d1..689b42e0c1 100644 --- a/crates/chain/src/indexed_tx_graph.rs +++ b/crates/chain/src/indexed_tx_graph.rs @@ -316,6 +316,7 @@ impl From> for ChangeSet { } } +#[cfg(feature = "miniscript")] impl From> for ChangeSet> { fn from(indexer: keychain::ChangeSet) -> Self { Self { diff --git a/crates/chain/src/keychain.rs b/crates/chain/src/keychain.rs index 63972a0ade..d2aa20bf7f 100644 --- a/crates/chain/src/keychain.rs +++ b/crates/chain/src/keychain.rs @@ -10,76 +10,11 @@ //! //! [`SpkTxOutIndex`]: crate::SpkTxOutIndex -use crate::{collections::BTreeMap, Append}; - #[cfg(feature = "miniscript")] mod txout_index; #[cfg(feature = "miniscript")] pub use txout_index::*; -/// Represents updates to the derivation index of a [`KeychainTxOutIndex`]. -/// It maps each keychain `K` to its last revealed index. -/// -/// It can be applied to [`KeychainTxOutIndex`] with [`apply_changeset`]. [`ChangeSet] are -/// monotone in that they will never decrease the revealed derivation index. -/// -/// [`KeychainTxOutIndex`]: crate::keychain::KeychainTxOutIndex -/// [`apply_changeset`]: crate::keychain::KeychainTxOutIndex::apply_changeset -#[derive(Clone, Debug, PartialEq)] -#[cfg_attr( - feature = "serde", - derive(serde::Deserialize, serde::Serialize), - serde( - crate = "serde_crate", - bound( - deserialize = "K: Ord + serde::Deserialize<'de>", - serialize = "K: Ord + serde::Serialize" - ) - ) -)] -#[must_use] -pub struct ChangeSet(pub BTreeMap); - -impl ChangeSet { - /// Get the inner map of the keychain to its new derivation index. - pub fn as_inner(&self) -> &BTreeMap { - &self.0 - } -} - -impl Append for ChangeSet { - /// Append another [`ChangeSet`] into self. - /// - /// If the keychain already exists, increase the index when the other's index > self's index. - /// If the keychain did not exist, append the new keychain. - fn append(&mut self, mut other: Self) { - self.0.iter_mut().for_each(|(key, index)| { - if let Some(other_index) = other.0.remove(key) { - *index = other_index.max(*index); - } - }); - - self.0.append(&mut other.0); - } - - /// Returns whether the changeset are empty. - fn is_empty(&self) -> bool { - self.0.is_empty() - } -} - -impl Default for ChangeSet { - fn default() -> Self { - Self(Default::default()) - } -} - -impl AsRef> for ChangeSet { - fn as_ref(&self) -> &BTreeMap { - &self.0 - } -} - /// Balance, differentiated into various categories. #[derive(Debug, PartialEq, Eq, Clone, Default)] #[cfg_attr( @@ -135,40 +70,3 @@ impl core::ops::Add for Balance { } } } - -#[cfg(test)] -mod test { - use super::*; - - #[test] - fn append_keychain_derivation_indices() { - #[derive(Ord, PartialOrd, Eq, PartialEq, Clone, Debug)] - enum Keychain { - One, - Two, - Three, - Four, - } - let mut lhs_di = BTreeMap::::default(); - let mut rhs_di = BTreeMap::::default(); - lhs_di.insert(Keychain::One, 7); - lhs_di.insert(Keychain::Two, 0); - rhs_di.insert(Keychain::One, 3); - rhs_di.insert(Keychain::Two, 5); - lhs_di.insert(Keychain::Three, 3); - rhs_di.insert(Keychain::Four, 4); - - let mut lhs = ChangeSet(lhs_di); - let rhs = ChangeSet(rhs_di); - lhs.append(rhs); - - // Exiting index doesn't update if the new index in `other` is lower than `self`. - assert_eq!(lhs.0.get(&Keychain::One), Some(&7)); - // Existing index updates if the new index in `other` is higher than `self`. - assert_eq!(lhs.0.get(&Keychain::Two), Some(&5)); - // Existing index is unchanged if keychain doesn't exist in `other`. - assert_eq!(lhs.0.get(&Keychain::Three), Some(&3)); - // New keychain gets added if the keychain is in `other` but not in `self`. - assert_eq!(lhs.0.get(&Keychain::Four), Some(&4)); - } -} diff --git a/crates/chain/src/keychain/txout_index.rs b/crates/chain/src/keychain/txout_index.rs index 2089673bae..fcfca5fda1 100644 --- a/crates/chain/src/keychain/txout_index.rs +++ b/crates/chain/src/keychain/txout_index.rs @@ -11,6 +11,69 @@ use core::{fmt::Debug, ops::Deref}; use crate::Append; +/// Represents updates to the derivation index of a [`KeychainTxOutIndex`]. +/// It maps each keychain `K` to its last revealed index. +/// +/// It can be applied to [`KeychainTxOutIndex`] with [`apply_changeset`]. [`ChangeSet] are +/// monotone in that they will never decrease the revealed derivation index. +/// +/// [`KeychainTxOutIndex`]: crate::keychain::KeychainTxOutIndex +/// [`apply_changeset`]: crate::keychain::KeychainTxOutIndex::apply_changeset +#[derive(Clone, Debug, PartialEq)] +#[cfg_attr( + feature = "serde", + derive(serde::Deserialize, serde::Serialize), + serde( + crate = "serde_crate", + bound( + deserialize = "K: Ord + serde::Deserialize<'de>", + serialize = "K: Ord + serde::Serialize" + ) + ) +)] +#[must_use] +pub struct ChangeSet(pub BTreeMap); + +impl ChangeSet { + /// Get the inner map of the keychain to its new derivation index. + pub fn as_inner(&self) -> &BTreeMap { + &self.0 + } +} + +impl Append for ChangeSet { + /// Append another [`ChangeSet`] into self. + /// + /// If the keychain already exists, increase the index when the other's index > self's index. + /// If the keychain did not exist, append the new keychain. + fn append(&mut self, mut other: Self) { + self.0.iter_mut().for_each(|(key, index)| { + if let Some(other_index) = other.0.remove(key) { + *index = other_index.max(*index); + } + }); + + self.0.append(&mut other.0); + } + + /// Returns whether the changeset are empty. + fn is_empty(&self) -> bool { + self.0.is_empty() + } +} + +impl Default for ChangeSet { + fn default() -> Self { + Self(Default::default()) + } +} + +impl AsRef> for ChangeSet { + fn as_ref(&self) -> &BTreeMap { + &self.0 + } +} + /// A convenient wrapper around [`SpkTxOutIndex`] that relates script pubkeys to miniscript public /// [`Descriptor`]s. /// diff --git a/crates/chain/tests/test_keychain_txout_index.rs b/crates/chain/tests/test_keychain_txout_index.rs index c4f4347155..a6edcd12e8 100644 --- a/crates/chain/tests/test_keychain_txout_index.rs +++ b/crates/chain/tests/test_keychain_txout_index.rs @@ -5,7 +5,7 @@ mod common; use bdk_chain::{ collections::BTreeMap, indexed_tx_graph::Indexer, - keychain::{self, KeychainTxOutIndex}, + keychain::{self, ChangeSet, KeychainTxOutIndex}, Append, }; @@ -42,6 +42,38 @@ fn spk_at_index(descriptor: &Descriptor, index: u32) -> Scr .script_pubkey() } +#[test] +fn append_keychain_derivation_indices() { + #[derive(Ord, PartialOrd, Eq, PartialEq, Clone, Debug)] + enum Keychain { + One, + Two, + Three, + Four, + } + let mut lhs_di = BTreeMap::::default(); + let mut rhs_di = BTreeMap::::default(); + lhs_di.insert(Keychain::One, 7); + lhs_di.insert(Keychain::Two, 0); + rhs_di.insert(Keychain::One, 3); + rhs_di.insert(Keychain::Two, 5); + lhs_di.insert(Keychain::Three, 3); + rhs_di.insert(Keychain::Four, 4); + + let mut lhs = ChangeSet(lhs_di); + let rhs = ChangeSet(rhs_di); + lhs.append(rhs); + + // Exiting index doesn't update if the new index in `other` is lower than `self`. + assert_eq!(lhs.0.get(&Keychain::One), Some(&7)); + // Existing index updates if the new index in `other` is higher than `self`. + assert_eq!(lhs.0.get(&Keychain::Two), Some(&5)); + // Existing index is unchanged if keychain doesn't exist in `other`. + assert_eq!(lhs.0.get(&Keychain::Three), Some(&3)); + // New keychain gets added if the keychain is in `other` but not in `self`. + assert_eq!(lhs.0.get(&Keychain::Four), Some(&4)); +} + #[test] fn test_set_all_derivation_indices() { use bdk_chain::indexed_tx_graph::Indexer;