From 4fd539b6470f7f771e3b5e09e3287952fa7a1825 Mon Sep 17 00:00:00 2001 From: Antoine Poinsot Date: Tue, 28 Nov 2023 18:08:49 +0100 Subject: [PATCH 1/4] feat(chain)!: `KeychainTxOutIndex` uses a universal lookahead MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit The wallet is currently created without setting any lookahead value for the keychain. This implicitly makes it a lookahead of 0. As this is a high-level interface we should avoid footguns and aim for a reasonable default. Instead of simply patching it for wallet, we alter `KeychainTxOutIndex` to have a default lookahead value. Additionally, instead of a per-keychain lookahead, the constructor asks for a `lookahead` value. This avoids the footguns of having methods which allows the caller the decrease the `lookahead` (and therefore panicing). This also simplifies the API. Co-authored-by: Antoine Poisot Co-authored-by: 志宇 --- crates/chain/src/keychain/txout_index.rs | 100 ++++++------------ crates/chain/src/spk_iter.rs | 2 +- crates/chain/tests/test_indexed_tx_graph.rs | 11 +- .../chain/tests/test_keychain_txout_index.rs | 32 +++--- .../example_bitcoind_rpc_polling/src/main.rs | 14 +-- 5 files changed, 56 insertions(+), 103 deletions(-) diff --git a/crates/chain/src/keychain/txout_index.rs b/crates/chain/src/keychain/txout_index.rs index 2089673ba..ed307de77 100644 --- a/crates/chain/src/keychain/txout_index.rs +++ b/crates/chain/src/keychain/txout_index.rs @@ -5,12 +5,13 @@ use crate::{ spk_iter::BIP32_MAX_INDEX, SpkIterator, SpkTxOutIndex, }; -use alloc::vec::Vec; use bitcoin::{OutPoint, Script, TxOut}; use core::{fmt::Debug, ops::Deref}; use crate::Append; +const DEFAULT_LOOKAHEAD: u32 = 1_000; + /// A convenient wrapper around [`SpkTxOutIndex`] that relates script pubkeys to miniscript public /// [`Descriptor`]s. /// @@ -46,7 +47,7 @@ use crate::Append; /// # let secp = bdk_chain::bitcoin::secp256k1::Secp256k1::signing_only(); /// # let (external_descriptor,_) = Descriptor::::parse_descriptor(&secp, "tr([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/0/*)").unwrap(); /// # let (internal_descriptor,_) = Descriptor::::parse_descriptor(&secp, "tr([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/1/*)").unwrap(); -/// # let descriptor_for_user_42 = external_descriptor.clone(); +/// # let (descriptor_for_user_42, _) = Descriptor::::parse_descriptor(&secp, "tr([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/2/*)").unwrap(); /// txout_index.add_keychain(MyKeychain::External, external_descriptor); /// txout_index.add_keychain(MyKeychain::Internal, internal_descriptor); /// txout_index.add_keychain(MyKeychain::MyAppUser { user_id: 42 }, descriptor_for_user_42); @@ -65,17 +66,12 @@ pub struct KeychainTxOutIndex { // last revealed indexes last_revealed: BTreeMap, // lookahead settings for each keychain - lookahead: BTreeMap, + lookahead: u32, } impl Default for KeychainTxOutIndex { fn default() -> Self { - Self { - inner: SpkTxOutIndex::default(), - keychains: BTreeMap::default(), - last_revealed: BTreeMap::default(), - lookahead: BTreeMap::default(), - } + Self::new(DEFAULT_LOOKAHEAD) } } @@ -118,6 +114,24 @@ impl Indexer for KeychainTxOutIndex { } } +impl KeychainTxOutIndex { + /// Construct a [`KeychainTxOutIndex`] with the given `lookahead`. + /// + /// The lookahead is the number of scripts to cache ahead of the last revealed script index. + /// This is useful to find outputs you own when processing block data that lie beyond the last + /// revealed index. In certain situations, such as when performing an initial scan of the + /// blockchain during wallet import, it may be uncertain or unknown what the last revealed index + /// is. + pub fn new(lookahead: u32) -> Self { + Self { + inner: SpkTxOutIndex::default(), + keychains: BTreeMap::new(), + last_revealed: BTreeMap::new(), + lookahead, + } + } +} + impl KeychainTxOutIndex { /// Return a reference to the internal [`SpkTxOutIndex`]. pub fn inner(&self) -> &SpkTxOutIndex<(K, u32)> { @@ -145,54 +159,22 @@ impl KeychainTxOutIndex { pub fn add_keychain(&mut self, keychain: K, descriptor: Descriptor) { let old_descriptor = &*self .keychains - .entry(keychain) + .entry(keychain.clone()) .or_insert_with(|| descriptor.clone()); assert_eq!( &descriptor, old_descriptor, "keychain already contains a different descriptor" ); + self.replenish_lookahead(&keychain, self.lookahead); } - /// Return the lookahead setting for each keychain. + /// Get the lookahead setting. /// - /// Refer to [`set_lookahead`] for a deeper explanation of the `lookahead`. + /// Refer to [`new`] for more information on the `lookahead`. /// - /// [`set_lookahead`]: Self::set_lookahead - pub fn lookaheads(&self) -> &BTreeMap { - &self.lookahead - } - - /// Convenience method to call [`set_lookahead`] for all keychains. - /// - /// [`set_lookahead`]: Self::set_lookahead - pub fn set_lookahead_for_all(&mut self, lookahead: u32) { - for keychain in &self.keychains.keys().cloned().collect::>() { - self.set_lookahead(keychain, lookahead); - } - } - - /// Set the lookahead count for `keychain`. - /// - /// The lookahead is the number of scripts to cache ahead of the last revealed script index. This - /// is useful to find outputs you own when processing block data that lie beyond the last revealed - /// index. In certain situations, such as when performing an initial scan of the blockchain during - /// wallet import, it may be uncertain or unknown what the last revealed index is. - /// - /// # Panics - /// - /// This will panic if the `keychain` does not exist. - pub fn set_lookahead(&mut self, keychain: &K, lookahead: u32) { - self.lookahead.insert(keychain.clone(), lookahead); - self.replenish_lookahead(keychain); - } - - /// Convenience method to call [`lookahead_to_target`] for multiple keychains. - /// - /// [`lookahead_to_target`]: Self::lookahead_to_target - pub fn lookahead_to_target_multi(&mut self, target_indexes: BTreeMap) { - for (keychain, target_index) in target_indexes { - self.lookahead_to_target(&keychain, target_index) - } + /// [`new`]: Self::new + pub fn lookahead(&self) -> u32 { + self.lookahead } /// Store lookahead scripts until `target_index`. @@ -201,22 +183,14 @@ impl KeychainTxOutIndex { pub fn lookahead_to_target(&mut self, keychain: &K, target_index: u32) { let next_index = self.next_store_index(keychain); if let Some(temp_lookahead) = target_index.checked_sub(next_index).filter(|&v| v > 0) { - let old_lookahead = self.lookahead.insert(keychain.clone(), temp_lookahead); - self.replenish_lookahead(keychain); - - // revert - match old_lookahead { - Some(lookahead) => self.lookahead.insert(keychain.clone(), lookahead), - None => self.lookahead.remove(keychain), - }; + self.replenish_lookahead(keychain, temp_lookahead); } } - fn replenish_lookahead(&mut self, keychain: &K) { + fn replenish_lookahead(&mut self, keychain: &K, lookahead: u32) { let descriptor = self.keychains.get(keychain).expect("keychain must exist"); let next_store_index = self.next_store_index(keychain); let next_reveal_index = self.last_revealed.get(keychain).map_or(0, |v| *v + 1); - let lookahead = self.lookahead.get(keychain).map_or(0, |v| *v); for (new_index, new_spk) in SpkIterator::new_with_range(descriptor, next_store_index..next_reveal_index + lookahead) @@ -388,12 +362,8 @@ impl KeychainTxOutIndex { let target_index = if has_wildcard { target_index } else { 0 }; let next_reveal_index = self.last_revealed.get(keychain).map_or(0, |v| *v + 1); - let lookahead = self.lookahead.get(keychain).map_or(0, |v| *v); - debug_assert_eq!( - next_reveal_index + lookahead, - self.next_store_index(keychain) - ); + debug_assert!(next_reveal_index + self.lookahead >= self.next_store_index(keychain)); // if we need to reveal new indices, the latest revealed index goes here let mut reveal_to_index = None; @@ -401,12 +371,12 @@ impl KeychainTxOutIndex { // if the target is not yet revealed, but is already stored (due to lookahead), we need to // set the `reveal_to_index` as target here (as the `for` loop below only updates // `reveal_to_index` for indexes that are NOT stored) - if next_reveal_index <= target_index && target_index < next_reveal_index + lookahead { + if next_reveal_index <= target_index && target_index < next_reveal_index + self.lookahead { reveal_to_index = Some(target_index); } // we range over indexes that are not stored - let range = next_reveal_index + lookahead..=target_index + lookahead; + let range = next_reveal_index + self.lookahead..=target_index + self.lookahead; for (new_index, new_spk) in SpkIterator::new_with_range(descriptor, range) { let _inserted = self .inner diff --git a/crates/chain/src/spk_iter.rs b/crates/chain/src/spk_iter.rs index a80cd6082..a28944f1e 100644 --- a/crates/chain/src/spk_iter.rs +++ b/crates/chain/src/spk_iter.rs @@ -148,7 +148,7 @@ mod test { Descriptor, Descriptor, ) { - let mut txout_index = KeychainTxOutIndex::::default(); + let mut txout_index = KeychainTxOutIndex::::new(0); let secp = Secp256k1::signing_only(); let (external_descriptor,_) = Descriptor::::parse_descriptor(&secp, "tr([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/0/*)").unwrap(); diff --git a/crates/chain/tests/test_indexed_tx_graph.rs b/crates/chain/tests/test_indexed_tx_graph.rs index ec250c95c..41b1d4d3e 100644 --- a/crates/chain/tests/test_indexed_tx_graph.rs +++ b/crates/chain/tests/test_indexed_tx_graph.rs @@ -27,9 +27,10 @@ fn insert_relevant_txs() { let spk_0 = descriptor.at_derivation_index(0).unwrap().script_pubkey(); let spk_1 = descriptor.at_derivation_index(9).unwrap().script_pubkey(); - let mut graph = IndexedTxGraph::>::default(); + let mut graph = IndexedTxGraph::>::new( + KeychainTxOutIndex::new(10), + ); graph.index.add_keychain((), descriptor); - graph.index.set_lookahead(&(), 10); let tx_a = Transaction { output: vec![ @@ -118,12 +119,12 @@ fn test_list_owned_txouts() { let (desc_1, _) = Descriptor::parse_descriptor(&Secp256k1::signing_only(), "tr(tprv8ZgxMBicQKsPd3krDUsBAmtnRsK3rb8u5yi1zhQgMhF1tR8MW7xfE4rnrbbsrbPR52e7rKapu6ztw1jXveJSCGHEriUGZV7mCe88duLp5pj/86'/1'/0'/0/*)").unwrap(); let (desc_2, _) = Descriptor::parse_descriptor(&Secp256k1::signing_only(), "tr(tprv8ZgxMBicQKsPd3krDUsBAmtnRsK3rb8u5yi1zhQgMhF1tR8MW7xfE4rnrbbsrbPR52e7rKapu6ztw1jXveJSCGHEriUGZV7mCe88duLp5pj/86'/1'/0'/1/*)").unwrap(); - let mut graph = - IndexedTxGraph::>::default(); + let mut graph = IndexedTxGraph::>::new( + KeychainTxOutIndex::new(10), + ); graph.index.add_keychain("keychain_1".into(), desc_1); graph.index.add_keychain("keychain_2".into(), desc_2); - graph.index.set_lookahead_for_all(10); // Get trusted and untrusted addresses diff --git a/crates/chain/tests/test_keychain_txout_index.rs b/crates/chain/tests/test_keychain_txout_index.rs index 161aad06d..e1f55dc31 100644 --- a/crates/chain/tests/test_keychain_txout_index.rs +++ b/crates/chain/tests/test_keychain_txout_index.rs @@ -18,12 +18,14 @@ enum TestKeychain { Internal, } -fn init_txout_index() -> ( +fn init_txout_index( + lookahead: u32, +) -> ( bdk_chain::keychain::KeychainTxOutIndex, Descriptor, Descriptor, ) { - let mut txout_index = bdk_chain::keychain::KeychainTxOutIndex::::default(); + let mut txout_index = bdk_chain::keychain::KeychainTxOutIndex::::new(lookahead); let secp = bdk_chain::bitcoin::secp256k1::Secp256k1::signing_only(); let (external_descriptor,_) = Descriptor::::parse_descriptor(&secp, "tr([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/0/*)").unwrap(); @@ -46,7 +48,7 @@ fn spk_at_index(descriptor: &Descriptor, index: u32) -> Scr fn test_set_all_derivation_indices() { use bdk_chain::indexed_tx_graph::Indexer; - let (mut txout_index, _, _) = init_txout_index(); + let (mut txout_index, _, _) = init_txout_index(0); let derive_to: BTreeMap<_, _> = [(TestKeychain::External, 12), (TestKeychain::Internal, 24)].into(); assert_eq!( @@ -64,19 +66,10 @@ fn test_set_all_derivation_indices() { #[test] fn test_lookahead() { - let (mut txout_index, external_desc, internal_desc) = init_txout_index(); - - // ensure it does not break anything if lookahead is set multiple times - (0..=10).for_each(|lookahead| txout_index.set_lookahead(&TestKeychain::External, lookahead)); - (0..=20) - .filter(|v| v % 2 == 0) - .for_each(|lookahead| txout_index.set_lookahead(&TestKeychain::Internal, lookahead)); - - assert_eq!(txout_index.inner().all_spks().len(), 30); + let (mut txout_index, external_desc, internal_desc) = init_txout_index(10); // given: // - external lookahead set to 10 - // - internal lookahead set to 20 // when: // - set external derivation index to value higher than last, but within the lookahead value // expect: @@ -97,7 +90,7 @@ fn test_lookahead() { assert_eq!( txout_index.inner().all_spks().len(), 10 /* external lookahead */ + - 20 /* internal lookahead */ + + 10 /* internal lookahead */ + index as usize + 1 /* `derived` count */ ); assert_eq!( @@ -127,7 +120,7 @@ fn test_lookahead() { } // given: - // - internal lookahead is 20 + // - internal lookahead is 10 // - internal derivation index is `None` // when: // - derivation index is set ahead of current derivation index + lookahead @@ -148,7 +141,7 @@ fn test_lookahead() { assert_eq!( txout_index.inner().all_spks().len(), 10 /* external lookahead */ + - 20 /* internal lookahead */ + + 10 /* internal lookahead */ + 20 /* external stored index count */ + 25 /* internal stored index count */ ); @@ -226,8 +219,7 @@ fn test_lookahead() { // - last used index should change as expected #[test] fn test_scan_with_lookahead() { - let (mut txout_index, external_desc, _) = init_txout_index(); - txout_index.set_lookahead_for_all(10); + let (mut txout_index, external_desc, _) = init_txout_index(10); let spks: BTreeMap = [0, 10, 20, 30] .into_iter() @@ -281,7 +273,7 @@ fn test_scan_with_lookahead() { #[test] #[rustfmt::skip] fn test_wildcard_derivations() { - let (mut txout_index, external_desc, _) = init_txout_index(); + let (mut txout_index, external_desc, _) = init_txout_index(0); let external_spk_0 = external_desc.at_derivation_index(0).unwrap().script_pubkey(); let external_spk_16 = external_desc.at_derivation_index(16).unwrap().script_pubkey(); let external_spk_26 = external_desc.at_derivation_index(26).unwrap().script_pubkey(); @@ -339,7 +331,7 @@ fn test_wildcard_derivations() { #[test] fn test_non_wildcard_derivations() { - let mut txout_index = KeychainTxOutIndex::::default(); + let mut txout_index = KeychainTxOutIndex::::new(0); let secp = bitcoin::secp256k1::Secp256k1::signing_only(); let (no_wildcard_descriptor, _) = Descriptor::::parse_descriptor(&secp, "wpkh([73c5da0a/86'/0'/0']xprv9xgqHN7yz9MwCkxsBPN5qetuNdQSUttZNKw1dcYTV4mkaAFiBVGQziHs3NRSWMkCzvgjEe3n9xV8oYywvM8at9yRqyaZVz6TYYhX98VjsUk/1/0)").unwrap(); diff --git a/example-crates/example_bitcoind_rpc_polling/src/main.rs b/example-crates/example_bitcoind_rpc_polling/src/main.rs index 35aa76907..93ef53f2b 100644 --- a/example-crates/example_bitcoind_rpc_polling/src/main.rs +++ b/example-crates/example_bitcoind_rpc_polling/src/main.rs @@ -64,9 +64,6 @@ struct RpcArgs { /// Starting block height to fallback to if no point of agreement if found #[clap(env = "FALLBACK_HEIGHT", long, default_value = "0")] fallback_height: u32, - /// The unused-scripts lookahead will be kept at this size - #[clap(long, default_value = "10")] - lookahead: u32, } impl From for Auth { @@ -161,13 +158,9 @@ fn main() -> anyhow::Result<()> { match rpc_cmd { RpcCommands::Sync { rpc_args } => { let RpcArgs { - fallback_height, - lookahead, - .. + fallback_height, .. } = rpc_args; - graph.lock().unwrap().index.set_lookahead_for_all(lookahead); - let chain_tip = chain.lock().unwrap().tip(); let rpc_client = rpc_args.new_client()?; let mut emitter = Emitter::new(&rpc_client, chain_tip, fallback_height); @@ -233,13 +226,10 @@ fn main() -> anyhow::Result<()> { } RpcCommands::Live { rpc_args } => { let RpcArgs { - fallback_height, - lookahead, - .. + fallback_height, .. } = rpc_args; let sigterm_flag = start_ctrlc_handler(); - graph.lock().unwrap().index.set_lookahead_for_all(lookahead); let last_cp = chain.lock().unwrap().tip(); println!( From bc796f412acdc3d2cd96f7aec0f24fa47c1fe889 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=BF=97=E5=AE=87?= Date: Thu, 28 Dec 2023 12:49:04 +0800 Subject: [PATCH 2/4] fix(example): bitcoind_rpc_polling now initializes local_chain properly Previously, the genesis block is not initialized properly. Thank you @notmandatory for identifying this bug. --- .../example_bitcoind_rpc_polling/src/main.rs | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/example-crates/example_bitcoind_rpc_polling/src/main.rs b/example-crates/example_bitcoind_rpc_polling/src/main.rs index 93ef53f2b..449242e41 100644 --- a/example-crates/example_bitcoind_rpc_polling/src/main.rs +++ b/example-crates/example_bitcoind_rpc_polling/src/main.rs @@ -12,7 +12,7 @@ use bdk_bitcoind_rpc::{ Emitter, }; use bdk_chain::{ - bitcoin::{Block, Transaction}, + bitcoin::{constants::genesis_block, Block, Transaction}, indexed_tx_graph, keychain, local_chain::{self, CheckPoint, LocalChain}, ConfirmationTimeHeightAnchor, IndexedTxGraph, @@ -117,10 +117,11 @@ fn main() -> anyhow::Result<()> { "[{:>10}s] loaded initial changeset from db", start.elapsed().as_secs_f32() ); + let (init_chain_changeset, init_graph_changeset) = init_changeset; let graph = Mutex::new({ let mut graph = IndexedTxGraph::new(index); - graph.apply_changeset(init_changeset.1); + graph.apply_changeset(init_graph_changeset); graph }); println!( @@ -128,7 +129,16 @@ fn main() -> anyhow::Result<()> { start.elapsed().as_secs_f32() ); - let chain = Mutex::new(LocalChain::from_changeset(init_changeset.0)?); + let chain = Mutex::new(if init_chain_changeset.is_empty() { + let genesis_hash = genesis_block(args.network).block_hash(); + let (chain, chain_changeset) = LocalChain::from_genesis_hash(genesis_hash); + let mut db = db.lock().unwrap(); + db.stage((chain_changeset, Default::default())); + db.commit()?; + chain + } else { + LocalChain::from_changeset(init_chain_changeset)? + }); println!( "[{:>10}s] loaded local chain from changeset", start.elapsed().as_secs_f32() From c9467dcbb27bd497006846dcfcefb0c2cf8823c4 Mon Sep 17 00:00:00 2001 From: LLFourn Date: Fri, 29 Dec 2023 16:40:48 +1100 Subject: [PATCH 3/4] chore: improve documentation of lookahead --- crates/chain/src/keychain/txout_index.rs | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/crates/chain/src/keychain/txout_index.rs b/crates/chain/src/keychain/txout_index.rs index ed307de77..3287d4f2e 100644 --- a/crates/chain/src/keychain/txout_index.rs +++ b/crates/chain/src/keychain/txout_index.rs @@ -117,11 +117,12 @@ impl Indexer for KeychainTxOutIndex { impl KeychainTxOutIndex { /// Construct a [`KeychainTxOutIndex`] with the given `lookahead`. /// - /// The lookahead is the number of scripts to cache ahead of the last revealed script index. - /// This is useful to find outputs you own when processing block data that lie beyond the last - /// revealed index. In certain situations, such as when performing an initial scan of the - /// blockchain during wallet import, it may be uncertain or unknown what the last revealed index - /// is. + /// The `lookahead` is the number of script pubkeys to derive and cache from the internal + /// descriptors over and above the last revealed script index. Without a lookahead the index + /// will miss outputs you own when processing transactions whose output script pubkeys lie + /// beyond the last revealed index. In certain situations, such as when performing an initial + /// scan of the blockchain during wallet import, it may be uncertain or unknown what the index + /// of the last revealed script pubkey actually is. pub fn new(lookahead: u32) -> Self { Self { inner: SpkTxOutIndex::default(), From 1def76f1f17fc876534c798883099c58de57c0fd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=BF=97=E5=AE=87?= Date: Fri, 29 Dec 2023 19:15:57 +0800 Subject: [PATCH 4/4] chore: make clippy happy and bump clippy msrv --- .github/workflows/cont_integration.yml | 2 +- clippy.toml | 2 +- crates/bdk/src/descriptor/template.rs | 4 ++-- crates/bdk/src/wallet/signer.rs | 9 ++------- crates/bdk/src/wallet/tx_builder.rs | 18 ++++-------------- crates/chain/src/spk_txout_index.rs | 4 +--- crates/chain/src/tx_graph.rs | 9 +++------ crates/chain/tests/common/mod.rs | 1 + crates/electrum/src/electrum_ext.rs | 2 +- 9 files changed, 16 insertions(+), 35 deletions(-) diff --git a/.github/workflows/cont_integration.yml b/.github/workflows/cont_integration.yml index 6281c465c..3dfb9906c 100644 --- a/.github/workflows/cont_integration.yml +++ b/.github/workflows/cont_integration.yml @@ -118,7 +118,7 @@ jobs: - uses: actions/checkout@v1 - uses: actions-rs/toolchain@v1 with: - toolchain: "stable" + toolchain: stable components: clippy override: true - name: Rust Cache diff --git a/clippy.toml b/clippy.toml index 3f726dbda..69478ceab 100644 --- a/clippy.toml +++ b/clippy.toml @@ -1 +1 @@ -msrv="1.57.0" +msrv="1.63.0" diff --git a/crates/bdk/src/descriptor/template.rs b/crates/bdk/src/descriptor/template.rs index c5e8b31c5..2773795a8 100644 --- a/crates/bdk/src/descriptor/template.rs +++ b/crates/bdk/src/descriptor/template.rs @@ -575,7 +575,7 @@ mod test { if let ExtendedDescriptor::Pkh(pkh) = xdesc.0 { let path: Vec = pkh.into_inner().full_derivation_path().unwrap().into(); - let purpose = path.get(0).unwrap(); + let purpose = path.first().unwrap(); assert_matches!(purpose, Hardened { index: 44 }); let coin_type = path.get(1).unwrap(); assert_matches!(coin_type, Hardened { index: 0 }); @@ -589,7 +589,7 @@ mod test { if let ExtendedDescriptor::Pkh(pkh) = tdesc.0 { let path: Vec = pkh.into_inner().full_derivation_path().unwrap().into(); - let purpose = path.get(0).unwrap(); + let purpose = path.first().unwrap(); assert_matches!(purpose, Hardened { index: 44 }); let coin_type = path.get(1).unwrap(); assert_matches!(coin_type, Hardened { index: 1 }); diff --git a/crates/bdk/src/wallet/signer.rs b/crates/bdk/src/wallet/signer.rs index da4940bf9..18e2062fb 100644 --- a/crates/bdk/src/wallet/signer.rs +++ b/crates/bdk/src/wallet/signer.rs @@ -812,9 +812,10 @@ pub struct SignOptions { } /// Customize which taproot script-path leaves the signer should sign. -#[derive(Debug, Clone, PartialEq, Eq)] +#[derive(Default, Debug, Clone, PartialEq, Eq)] pub enum TapLeavesOptions { /// The signer will sign all the leaves it has a key for. + #[default] All, /// The signer won't sign leaves other than the ones specified. Note that it could still ignore /// some of the specified leaves, if it doesn't have the right key to sign them. @@ -825,12 +826,6 @@ pub enum TapLeavesOptions { None, } -impl Default for TapLeavesOptions { - fn default() -> Self { - TapLeavesOptions::All - } -} - #[allow(clippy::derivable_impls)] impl Default for SignOptions { fn default() -> Self { diff --git a/crates/bdk/src/wallet/tx_builder.rs b/crates/bdk/src/wallet/tx_builder.rs index 86decc8bb..f914aaefa 100644 --- a/crates/bdk/src/wallet/tx_builder.rs +++ b/crates/bdk/src/wallet/tx_builder.rs @@ -811,9 +811,10 @@ impl<'a, D> TxBuilder<'a, D, DefaultCoinSelectionAlgorithm, BumpFee> { } /// Ordering of the transaction's inputs and outputs -#[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] +#[derive(Default, Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] pub enum TxOrdering { /// Randomized (default) + #[default] Shuffle, /// Unchanged Untouched, @@ -821,12 +822,6 @@ pub enum TxOrdering { Bip69Lexicographic, } -impl Default for TxOrdering { - fn default() -> Self { - TxOrdering::Shuffle - } -} - impl TxOrdering { /// Sort transaction inputs and outputs by [`TxOrdering`] variant pub fn sort_tx(&self, tx: &mut Transaction) { @@ -880,9 +875,10 @@ impl RbfValue { } /// Policy regarding the use of change outputs when creating a transaction -#[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] +#[derive(Default, Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] pub enum ChangeSpendPolicy { /// Use both change and non-change outputs (default) + #[default] ChangeAllowed, /// Only use change outputs (see [`TxBuilder::only_spend_change`]) OnlyChange, @@ -890,12 +886,6 @@ pub enum ChangeSpendPolicy { ChangeForbidden, } -impl Default for ChangeSpendPolicy { - fn default() -> Self { - ChangeSpendPolicy::ChangeAllowed - } -} - impl ChangeSpendPolicy { pub(crate) fn is_satisfied_by(&self, utxo: &LocalOutput) -> bool { match self { diff --git a/crates/chain/src/spk_txout_index.rs b/crates/chain/src/spk_txout_index.rs index 4047f8fcb..8df1b1197 100644 --- a/crates/chain/src/spk_txout_index.rs +++ b/crates/chain/src/spk_txout_index.rs @@ -168,9 +168,7 @@ impl SpkTxOutIndex { /// /// Returns `None` if the `TxOut` hasn't been scanned or if nothing matching was found there. pub fn txout(&self, outpoint: OutPoint) -> Option<(&I, &TxOut)> { - self.txouts - .get(&outpoint) - .map(|(spk_i, txout)| (spk_i, txout)) + self.txouts.get(&outpoint).map(|v| (&v.0, &v.1)) } /// Returns the script that has been inserted at the `index`. diff --git a/crates/chain/src/tx_graph.rs b/crates/chain/src/tx_graph.rs index 10cf10446..ef15d7e7d 100644 --- a/crates/chain/src/tx_graph.rs +++ b/crates/chain/src/tx_graph.rs @@ -581,10 +581,7 @@ impl TxGraph { } for (outpoint, txout) in changeset.txouts { - let tx_entry = self - .txs - .entry(outpoint.txid) - .or_insert_with(Default::default); + let tx_entry = self.txs.entry(outpoint.txid).or_default(); match tx_entry { (TxNodeInternal::Whole(_), _, _) => { /* do nothing since we already have full tx */ @@ -597,13 +594,13 @@ impl TxGraph { for (anchor, txid) in changeset.anchors { if self.anchors.insert((anchor.clone(), txid)) { - let (_, anchors, _) = self.txs.entry(txid).or_insert_with(Default::default); + let (_, anchors, _) = self.txs.entry(txid).or_default(); anchors.insert(anchor); } } for (txid, new_last_seen) in changeset.last_seen { - let (_, _, last_seen) = self.txs.entry(txid).or_insert_with(Default::default); + let (_, _, last_seen) = self.txs.entry(txid).or_default(); if new_last_seen > *last_seen { *last_seen = new_last_seen; } diff --git a/crates/chain/tests/common/mod.rs b/crates/chain/tests/common/mod.rs index cfb8b7c44..7cc3f57a9 100644 --- a/crates/chain/tests/common/mod.rs +++ b/crates/chain/tests/common/mod.rs @@ -1,4 +1,5 @@ mod tx_template; +#[allow(unused_imports)] pub use tx_template::*; #[allow(unused_macros)] diff --git a/crates/electrum/src/electrum_ext.rs b/crates/electrum/src/electrum_ext.rs index e54007cd9..216755a13 100644 --- a/crates/electrum/src/electrum_ext.rs +++ b/crates/electrum/src/electrum_ext.rs @@ -477,7 +477,7 @@ fn populate_with_txids( let spk = tx .output - .get(0) + .first() .map(|txo| &txo.script_pubkey) .expect("tx must have an output");