Merge bitcoindevkit/bdk#770: Upgrade to rust-bitcoin 0.29
c7a43d941fd0f457f615f2fe43e71586a253e46a Remove unused code (Alekos Filini) 1ffd59d469ff28f673d854eaf15c992c15541bfe Upgrade to rust-bitcoin 0.29 (Alekos Filini) ae4f4e541671e85be04fdddc97cb7f65af1eeea1 Upgrade `rand` to `0.8` (Alekos Filini) 9854fd34eaa688e5acd601119b8f448a2d77fbd9 Remove deprecated address validators (Alekos Filini) Pull request description: ### Description Upgrade BDK to rust-bitcoin 0.29 Missing pieces: - [x] rust-miniscript `update_output_with_descriptor` - rust-bitcoin/rust-miniscript#465 - [x] rust-miniscript 8.0.0 release - rust-bitcoin/rust-miniscript#462 - [x] Upgrade rust-hwi to bitcoin 0.29 bitcoindevkit/rust-hwi#50 - [x] Upgrade esplora-client to bitcoin 0.29 https://github.com/bitcoindevkit/rust-esplora-client/pull/20 - [x] Upgrade rand to 0.8 like secp256k1 did ### Notes to the reviewers The commits still need to be reordered and cleaned up ### Changelog notice - Upgrade rust-bitcoin to 0.29 - Remove deprecated "address validators" ### Checklists #### All Submissions: * [x] I've signed all my commits * [x] I followed the [contribution guidelines](https://github.com/bitcoindevkit/bdk/blob/master/CONTRIBUTING.md) * [x] I ran `cargo fmt` and `cargo clippy` before committing ACKs for top commit: notmandatory: ACK c7a43d941fd0f457f615f2fe43e71586a253e46a Tree-SHA512: 718a1baf3613b31ec1de39fe63467ebee38617963a4ce0670a617e20fe4f46a57c5786933cdde6cfad9fc76ce0af08843f58844fb4a89f5948cb42c697f802ef
This commit is contained in:
		
						commit
						5720e38033
					
				
							
								
								
									
										2
									
								
								.github/workflows/cont_integration.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/cont_integration.yml
									
									
									
									
										vendored
									
									
								
							| @ -154,7 +154,7 @@ jobs: | |||||||
|       - name: Update toolchain |       - name: Update toolchain | ||||||
|         run: rustup update |         run: rustup update | ||||||
|       - name: Check |       - name: Check | ||||||
|         run: cargo check --target wasm32-unknown-unknown --features use-esplora-async --no-default-features |         run: cargo check --target wasm32-unknown-unknown --features use-esplora-async,dev-getrandom-wasm --no-default-features | ||||||
| 
 | 
 | ||||||
|   fmt: |   fmt: | ||||||
|     name: Rust fmt |     name: Rust fmt | ||||||
|  | |||||||
							
								
								
									
										27
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								Cargo.toml
									
									
									
									
									
								
							| @ -14,16 +14,16 @@ license = "MIT OR Apache-2.0" | |||||||
| [dependencies] | [dependencies] | ||||||
| bdk-macros = "^0.6" | bdk-macros = "^0.6" | ||||||
| log = "^0.4" | log = "^0.4" | ||||||
| miniscript = { version = "7.0", features = ["use-serde"] } | miniscript = { version = "8.0", features = ["serde"] } | ||||||
| bitcoin = { version = "0.28.1", features = ["use-serde", "base64", "rand"] } | bitcoin = { version = "0.29.1", features = ["serde", "base64", "rand"] } | ||||||
| serde = { version = "^1.0", features = ["derive"] } | serde = { version = "^1.0", features = ["derive"] } | ||||||
| serde_json = { version = "^1.0" } | serde_json = { version = "^1.0" } | ||||||
| rand = "^0.7" | rand = "^0.8" | ||||||
| 
 | 
 | ||||||
| # Optional dependencies | # Optional dependencies | ||||||
| sled = { version = "0.34", optional = true } | sled = { version = "0.34", optional = true } | ||||||
| electrum-client = { version = "0.11", optional = true } | electrum-client = { version = "0.12", optional = true } | ||||||
| esplora-client = { version = "0.1.1", default-features = false, optional = true } | esplora-client = { version = "0.2", default-features = false, optional = true } | ||||||
| rusqlite = { version = "0.27.0", optional = true } | rusqlite = { version = "0.27.0", optional = true } | ||||||
| ahash = { version = "0.7.6", optional = true } | ahash = { version = "0.7.6", optional = true } | ||||||
| futures = { version = "0.3", optional = true } | futures = { version = "0.3", optional = true } | ||||||
| @ -31,22 +31,22 @@ async-trait = { version = "0.1", optional = true } | |||||||
| rocksdb = { version = "0.14", default-features = false, features = ["snappy"], optional = true } | rocksdb = { version = "0.14", default-features = false, features = ["snappy"], optional = true } | ||||||
| cc = { version = ">=1.0.64", optional = true } | cc = { version = ">=1.0.64", optional = true } | ||||||
| socks = { version = "0.3", optional = true } | socks = { version = "0.3", optional = true } | ||||||
| hwi = { version = "0.2.3", optional = true } | hwi = { version = "0.3.0", optional = true } | ||||||
| 
 | 
 | ||||||
| bip39 = { version = "1.0.1", optional = true } | bip39 = { version = "1.0.1", optional = true } | ||||||
| bitcoinconsensus = { version = "0.19.0-3", optional = true } | bitcoinconsensus = { version = "0.19.0-3", optional = true } | ||||||
| 
 | 
 | ||||||
| # Needed by bdk_blockchain_tests macro and the `rpc` feature | # Needed by bdk_blockchain_tests macro and the `rpc` feature | ||||||
| bitcoincore-rpc = { version = "0.15", optional = true } | bitcoincore-rpc = { version = "0.16", optional = true } | ||||||
| 
 | 
 | ||||||
| # Platform-specific dependencies | # Platform-specific dependencies | ||||||
| [target.'cfg(not(target_arch = "wasm32"))'.dependencies] | [target.'cfg(not(target_arch = "wasm32"))'.dependencies] | ||||||
| tokio = { version = "1", features = ["rt"] } | tokio = { version = "1", features = ["rt"] } | ||||||
| 
 | 
 | ||||||
| [target.'cfg(target_arch = "wasm32")'.dependencies] | [target.'cfg(target_arch = "wasm32")'.dependencies] | ||||||
|  | getrandom = "0.2" | ||||||
| async-trait = "0.1" | async-trait = "0.1" | ||||||
| js-sys = "0.3" | js-sys = "0.3" | ||||||
| rand = { version = "^0.7", features = ["wasm-bindgen"] } |  | ||||||
| 
 | 
 | ||||||
| [features] | [features] | ||||||
| minimal = [] | minimal = [] | ||||||
| @ -98,13 +98,18 @@ test-esplora = ["electrsd/legacy", "electrsd/esplora_a33e97e1", "electrsd/bitcoi | |||||||
| test-md-docs = ["electrum"] | test-md-docs = ["electrum"] | ||||||
| test-hardware-signer = ["hardware-signer"] | test-hardware-signer = ["hardware-signer"] | ||||||
| 
 | 
 | ||||||
|  | # This feature is used to run `cargo check` in our CI targeting wasm. It's not recommended | ||||||
|  | # for libraries to explicitly include the "getrandom/js" feature, so we only do it when | ||||||
|  | # necessary for running our CI. See: https://docs.rs/getrandom/0.2.8/getrandom/#webassembly-support | ||||||
|  | dev-getrandom-wasm = ["getrandom/js"] | ||||||
|  | 
 | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| lazy_static = "1.4" | lazy_static = "1.4" | ||||||
| env_logger = "0.7" | env_logger = "0.7" | ||||||
| electrsd = "0.20" | electrsd = "0.21" | ||||||
|  | # Move back to importing from rust-bitcoin once https://github.com/rust-bitcoin/rust-bitcoin/pull/1342 is released | ||||||
|  | base64 = "^0.13" | ||||||
| 
 | 
 | ||||||
| [[example]] |  | ||||||
| name = "address_validator" |  | ||||||
| [[example]] | [[example]] | ||||||
| name = "compact_filters_balance" | name = "compact_filters_balance" | ||||||
| required-features = ["compact_filters"] | required-features = ["compact_filters"] | ||||||
|  | |||||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							| @ -95,7 +95,7 @@ use bdk::blockchain::ElectrumBlockchain; | |||||||
| use bdk::electrum_client::Client; | use bdk::electrum_client::Client; | ||||||
| use bdk::wallet::AddressIndex::New; | use bdk::wallet::AddressIndex::New; | ||||||
| 
 | 
 | ||||||
| use bitcoin::base64; | use base64; | ||||||
| use bitcoin::consensus::serialize; | use bitcoin::consensus::serialize; | ||||||
| 
 | 
 | ||||||
| fn main() -> Result<(), bdk::Error> { | fn main() -> Result<(), bdk::Error> { | ||||||
| @ -132,7 +132,7 @@ fn main() -> Result<(), bdk::Error> { | |||||||
| ```rust,no_run | ```rust,no_run | ||||||
| use bdk::{Wallet, SignOptions, database::MemoryDatabase}; | use bdk::{Wallet, SignOptions, database::MemoryDatabase}; | ||||||
| 
 | 
 | ||||||
| use bitcoin::base64; | use base64; | ||||||
| use bitcoin::consensus::deserialize; | use bitcoin::consensus::deserialize; | ||||||
| 
 | 
 | ||||||
| fn main() -> Result<(), bdk::Error> { | fn main() -> Result<(), bdk::Error> { | ||||||
| @ -171,6 +171,17 @@ cargo test --features test-electrum | |||||||
| The other options are `test-esplora`, `test-rpc` or `test-rpc-legacy` which runs against an older version of Bitcoin Core. | The other options are `test-esplora`, `test-rpc` or `test-rpc-legacy` which runs against an older version of Bitcoin Core. | ||||||
| Note that `electrs` and `bitcoind` binaries are automatically downloaded (on mac and linux), to specify you already have installed binaries you must use `--no-default-features` and provide `BITCOIND_EXE` and `ELECTRS_EXE` as environment variables. | Note that `electrs` and `bitcoind` binaries are automatically downloaded (on mac and linux), to specify you already have installed binaries you must use `--no-default-features` and provide `BITCOIND_EXE` and `ELECTRS_EXE` as environment variables. | ||||||
| 
 | 
 | ||||||
|  | ## Running under WASM | ||||||
|  | 
 | ||||||
|  | If you want to run this library under WASM you will probably have to add the following lines to you `Cargo.toml`: | ||||||
|  | 
 | ||||||
|  | ```toml | ||||||
|  | [dependencies] | ||||||
|  | getrandom = { version = "0.2", features = ["js"] } | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | This enables the `rand` crate to work in environments where JavaScript is available. See [this link](https://docs.rs/getrandom/0.2.8/getrandom/#webassembly-support) to learn more. | ||||||
|  | 
 | ||||||
| ## License | ## License | ||||||
| 
 | 
 | ||||||
| Licensed under either of | Licensed under either of | ||||||
|  | |||||||
| @ -1,63 +0,0 @@ | |||||||
| // Bitcoin Dev Kit
 |  | ||||||
| // Written in 2020 by Alekos Filini <alekos.filini@gmail.com>
 |  | ||||||
| //
 |  | ||||||
| // Copyright (c) 2020-2021 Bitcoin Dev Kit Developers
 |  | ||||||
| //
 |  | ||||||
| // This file is licensed under the Apache License, Version 2.0 <LICENSE-APACHE
 |  | ||||||
| // or http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
 |  | ||||||
| // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your option.
 |  | ||||||
| // You may not use this file except in accordance with one or both of these
 |  | ||||||
| // licenses.
 |  | ||||||
| 
 |  | ||||||
| use std::sync::Arc; |  | ||||||
| 
 |  | ||||||
| use bdk::bitcoin; |  | ||||||
| use bdk::database::MemoryDatabase; |  | ||||||
| use bdk::descriptor::HdKeyPaths; |  | ||||||
| #[allow(deprecated)] |  | ||||||
| use bdk::wallet::address_validator::{AddressValidator, AddressValidatorError}; |  | ||||||
| use bdk::KeychainKind; |  | ||||||
| use bdk::Wallet; |  | ||||||
| 
 |  | ||||||
| use bdk::wallet::AddressIndex::New; |  | ||||||
| use bitcoin::hashes::hex::FromHex; |  | ||||||
| use bitcoin::util::bip32::Fingerprint; |  | ||||||
| use bitcoin::{Network, Script}; |  | ||||||
| 
 |  | ||||||
| #[derive(Debug)] |  | ||||||
| struct DummyValidator; |  | ||||||
| #[allow(deprecated)] |  | ||||||
| impl AddressValidator for DummyValidator { |  | ||||||
|     fn validate( |  | ||||||
|         &self, |  | ||||||
|         keychain: KeychainKind, |  | ||||||
|         hd_keypaths: &HdKeyPaths, |  | ||||||
|         script: &Script, |  | ||||||
|     ) -> Result<(), AddressValidatorError> { |  | ||||||
|         let (_, path) = hd_keypaths |  | ||||||
|             .values() |  | ||||||
|             .find(|(fing, _)| fing == &Fingerprint::from_hex("bc123c3e").unwrap()) |  | ||||||
|             .ok_or(AddressValidatorError::InvalidScript)?; |  | ||||||
| 
 |  | ||||||
|         println!( |  | ||||||
|             "Validating `{:?}` {} address, script: {}", |  | ||||||
|             keychain, path, script |  | ||||||
|         ); |  | ||||||
| 
 |  | ||||||
|         Ok(()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| fn main() -> Result<(), bdk::Error> { |  | ||||||
|     let descriptor = "sh(and_v(v:pk(tpubDDpWvmUrPZrhSPmUzCMBHffvC3HyMAPnWDSAQNBTnj1iZeJa7BZQEttFiP4DS4GCcXQHezdXhn86Hj6LHX5EDstXPWrMaSneRWM8yUf6NFd/*),after(630000)))"; |  | ||||||
|     let mut wallet = Wallet::new(descriptor, None, Network::Regtest, MemoryDatabase::new())?; |  | ||||||
| 
 |  | ||||||
|     #[allow(deprecated)] |  | ||||||
|     wallet.add_address_validator(Arc::new(DummyValidator)); |  | ||||||
| 
 |  | ||||||
|     wallet.get_address(New)?; |  | ||||||
|     wallet.get_address(New)?; |  | ||||||
|     wallet.get_address(New)?; |  | ||||||
| 
 |  | ||||||
|     Ok(()) |  | ||||||
| } |  | ||||||
| @ -33,8 +33,8 @@ fn main() -> Result<(), Box<dyn Error>> { | |||||||
|     let internal_secret_xkey = DescriptorSecretKey::from_str("[e9824965/84'/1'/0']tprv8fvem7qWxY3SGCQczQpRpqTKg455wf1zgixn6MZ4ze8gRfHjov5gXBQTadNfDgqs9ERbZZ3Bi1PNYrCCusFLucT39K525MWLpeURjHwUsfX/1/*").unwrap(); |     let internal_secret_xkey = DescriptorSecretKey::from_str("[e9824965/84'/1'/0']tprv8fvem7qWxY3SGCQczQpRpqTKg455wf1zgixn6MZ4ze8gRfHjov5gXBQTadNfDgqs9ERbZZ3Bi1PNYrCCusFLucT39K525MWLpeURjHwUsfX/1/*").unwrap(); | ||||||
| 
 | 
 | ||||||
|     let secp = Secp256k1::new(); |     let secp = Secp256k1::new(); | ||||||
|     let external_public_xkey = external_secret_xkey.as_public(&secp).unwrap(); |     let external_public_xkey = external_secret_xkey.to_public(&secp).unwrap(); | ||||||
|     let internal_public_xkey = internal_secret_xkey.as_public(&secp).unwrap(); |     let internal_public_xkey = internal_secret_xkey.to_public(&secp).unwrap(); | ||||||
| 
 | 
 | ||||||
|     let signing_external_descriptor = descriptor!(wpkh(external_secret_xkey)).unwrap(); |     let signing_external_descriptor = descriptor!(wpkh(external_secret_xkey)).unwrap(); | ||||||
|     let signing_internal_descriptor = descriptor!(wpkh(internal_secret_xkey)).unwrap(); |     let signing_internal_descriptor = descriptor!(wpkh(internal_secret_xkey)).unwrap(); | ||||||
|  | |||||||
| @ -194,7 +194,7 @@ impl_from!(boxed rpc::RpcBlockchain, AnyBlockchain, Rpc, #[cfg(feature = "rpc")] | |||||||
| /// );
 | /// );
 | ||||||
| /// # }
 | /// # }
 | ||||||
| /// ```
 | /// ```
 | ||||||
| #[derive(Debug, serde::Serialize, serde::Deserialize, Clone, PartialEq)] | #[derive(Debug, serde::Serialize, serde::Deserialize, Clone, PartialEq, Eq)] | ||||||
| #[serde(tag = "type", rename_all = "snake_case")] | #[serde(tag = "type", rename_all = "snake_case")] | ||||||
| pub enum AnyBlockchainConfig { | pub enum AnyBlockchainConfig { | ||||||
|     #[cfg(feature = "electrum")] |     #[cfg(feature = "electrum")] | ||||||
|  | |||||||
| @ -479,7 +479,7 @@ impl WalletSync for CompactFiltersBlockchain { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Data to connect to a Bitcoin P2P peer
 | /// Data to connect to a Bitcoin P2P peer
 | ||||||
| #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq)] | #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq, Eq)] | ||||||
| pub struct BitcoinPeerConfig { | pub struct BitcoinPeerConfig { | ||||||
|     /// Peer address such as 127.0.0.1:18333
 |     /// Peer address such as 127.0.0.1:18333
 | ||||||
|     pub address: String, |     pub address: String, | ||||||
| @ -490,7 +490,7 @@ pub struct BitcoinPeerConfig { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Configuration for a [`CompactFiltersBlockchain`]
 | /// Configuration for a [`CompactFiltersBlockchain`]
 | ||||||
| #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq)] | #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq, Eq)] | ||||||
| pub struct CompactFiltersBlockchainConfig { | pub struct CompactFiltersBlockchainConfig { | ||||||
|     /// List of peers to try to connect to for asking headers and filters
 |     /// List of peers to try to connect to for asking headers and filters
 | ||||||
|     pub peers: Vec<BitcoinPeerConfig>, |     pub peers: Vec<BitcoinPeerConfig>, | ||||||
|  | |||||||
| @ -75,7 +75,10 @@ impl Mempool { | |||||||
|     /// Look-up a transaction in the mempool given an [`Inventory`] request
 |     /// Look-up a transaction in the mempool given an [`Inventory`] request
 | ||||||
|     pub fn get_tx(&self, inventory: &Inventory) -> Option<Transaction> { |     pub fn get_tx(&self, inventory: &Inventory) -> Option<Transaction> { | ||||||
|         let identifer = match inventory { |         let identifer = match inventory { | ||||||
|             Inventory::Error | Inventory::Block(_) | Inventory::WitnessBlock(_) => return None, |             Inventory::Error | ||||||
|  |             | Inventory::Block(_) | ||||||
|  |             | Inventory::WitnessBlock(_) | ||||||
|  |             | Inventory::CompactBlock(_) => return None, | ||||||
|             Inventory::Transaction(txid) => TxIdentifier::Txid(*txid), |             Inventory::Transaction(txid) => TxIdentifier::Txid(*txid), | ||||||
|             Inventory::WitnessTransaction(txid) => TxIdentifier::Txid(*txid), |             Inventory::WitnessTransaction(txid) => TxIdentifier::Txid(*txid), | ||||||
|             Inventory::WTx(wtxid) => TxIdentifier::Wtxid(*wtxid), |             Inventory::WTx(wtxid) => TxIdentifier::Wtxid(*wtxid), | ||||||
|  | |||||||
| @ -103,42 +103,42 @@ where | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Encodable for BundleStatus { | impl Encodable for BundleStatus { | ||||||
|     fn consensus_encode<W: Write>(&self, mut e: W) -> Result<usize, std::io::Error> { |     fn consensus_encode<W: Write + ?Sized>(&self, e: &mut W) -> Result<usize, std::io::Error> { | ||||||
|         let mut written = 0; |         let mut written = 0; | ||||||
| 
 | 
 | ||||||
|         match self { |         match self { | ||||||
|             BundleStatus::Init => { |             BundleStatus::Init => { | ||||||
|                 written += 0x00u8.consensus_encode(&mut e)?; |                 written += 0x00u8.consensus_encode(e)?; | ||||||
|             } |             } | ||||||
|             BundleStatus::CfHeaders { cf_headers } => { |             BundleStatus::CfHeaders { cf_headers } => { | ||||||
|                 written += 0x01u8.consensus_encode(&mut e)?; |                 written += 0x01u8.consensus_encode(e)?; | ||||||
|                 written += VarInt(cf_headers.len() as u64).consensus_encode(&mut e)?; |                 written += VarInt(cf_headers.len() as u64).consensus_encode(e)?; | ||||||
|                 for header in cf_headers { |                 for header in cf_headers { | ||||||
|                     written += header.consensus_encode(&mut e)?; |                     written += header.consensus_encode(e)?; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             BundleStatus::CFilters { cf_filters } => { |             BundleStatus::CFilters { cf_filters } => { | ||||||
|                 written += 0x02u8.consensus_encode(&mut e)?; |                 written += 0x02u8.consensus_encode(e)?; | ||||||
|                 written += VarInt(cf_filters.len() as u64).consensus_encode(&mut e)?; |                 written += VarInt(cf_filters.len() as u64).consensus_encode(e)?; | ||||||
|                 for filter in cf_filters { |                 for filter in cf_filters { | ||||||
|                     written += filter.consensus_encode(&mut e)?; |                     written += filter.consensus_encode(e)?; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             BundleStatus::Processed { cf_filters } => { |             BundleStatus::Processed { cf_filters } => { | ||||||
|                 written += 0x03u8.consensus_encode(&mut e)?; |                 written += 0x03u8.consensus_encode(e)?; | ||||||
|                 written += VarInt(cf_filters.len() as u64).consensus_encode(&mut e)?; |                 written += VarInt(cf_filters.len() as u64).consensus_encode(e)?; | ||||||
|                 for filter in cf_filters { |                 for filter in cf_filters { | ||||||
|                     written += filter.consensus_encode(&mut e)?; |                     written += filter.consensus_encode(e)?; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             BundleStatus::Pruned => { |             BundleStatus::Pruned => { | ||||||
|                 written += 0x04u8.consensus_encode(&mut e)?; |                 written += 0x04u8.consensus_encode(e)?; | ||||||
|             } |             } | ||||||
|             BundleStatus::Tip { cf_filters } => { |             BundleStatus::Tip { cf_filters } => { | ||||||
|                 written += 0x05u8.consensus_encode(&mut e)?; |                 written += 0x05u8.consensus_encode(e)?; | ||||||
|                 written += VarInt(cf_filters.len() as u64).consensus_encode(&mut e)?; |                 written += VarInt(cf_filters.len() as u64).consensus_encode(e)?; | ||||||
|                 for filter in cf_filters { |                 for filter in cf_filters { | ||||||
|                     written += filter.consensus_encode(&mut e)?; |                     written += filter.consensus_encode(e)?; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -148,51 +148,53 @@ impl Encodable for BundleStatus { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Decodable for BundleStatus { | impl Decodable for BundleStatus { | ||||||
|     fn consensus_decode<D: Read>(mut d: D) -> Result<Self, bitcoin::consensus::encode::Error> { |     fn consensus_decode<D: Read + ?Sized>( | ||||||
|         let byte_type = u8::consensus_decode(&mut d)?; |         d: &mut D, | ||||||
|  |     ) -> Result<Self, bitcoin::consensus::encode::Error> { | ||||||
|  |         let byte_type = u8::consensus_decode(d)?; | ||||||
|         match byte_type { |         match byte_type { | ||||||
|             0x00 => Ok(BundleStatus::Init), |             0x00 => Ok(BundleStatus::Init), | ||||||
|             0x01 => { |             0x01 => { | ||||||
|                 let num = VarInt::consensus_decode(&mut d)?; |                 let num = VarInt::consensus_decode(d)?; | ||||||
|                 let num = num.0 as usize; |                 let num = num.0 as usize; | ||||||
| 
 | 
 | ||||||
|                 let mut cf_headers = Vec::with_capacity(num); |                 let mut cf_headers = Vec::with_capacity(num); | ||||||
|                 for _ in 0..num { |                 for _ in 0..num { | ||||||
|                     cf_headers.push(FilterHeader::consensus_decode(&mut d)?); |                     cf_headers.push(FilterHeader::consensus_decode(d)?); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 Ok(BundleStatus::CfHeaders { cf_headers }) |                 Ok(BundleStatus::CfHeaders { cf_headers }) | ||||||
|             } |             } | ||||||
|             0x02 => { |             0x02 => { | ||||||
|                 let num = VarInt::consensus_decode(&mut d)?; |                 let num = VarInt::consensus_decode(d)?; | ||||||
|                 let num = num.0 as usize; |                 let num = num.0 as usize; | ||||||
| 
 | 
 | ||||||
|                 let mut cf_filters = Vec::with_capacity(num); |                 let mut cf_filters = Vec::with_capacity(num); | ||||||
|                 for _ in 0..num { |                 for _ in 0..num { | ||||||
|                     cf_filters.push(Vec::<u8>::consensus_decode(&mut d)?); |                     cf_filters.push(Vec::<u8>::consensus_decode(d)?); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 Ok(BundleStatus::CFilters { cf_filters }) |                 Ok(BundleStatus::CFilters { cf_filters }) | ||||||
|             } |             } | ||||||
|             0x03 => { |             0x03 => { | ||||||
|                 let num = VarInt::consensus_decode(&mut d)?; |                 let num = VarInt::consensus_decode(d)?; | ||||||
|                 let num = num.0 as usize; |                 let num = num.0 as usize; | ||||||
| 
 | 
 | ||||||
|                 let mut cf_filters = Vec::with_capacity(num); |                 let mut cf_filters = Vec::with_capacity(num); | ||||||
|                 for _ in 0..num { |                 for _ in 0..num { | ||||||
|                     cf_filters.push(Vec::<u8>::consensus_decode(&mut d)?); |                     cf_filters.push(Vec::<u8>::consensus_decode(d)?); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 Ok(BundleStatus::Processed { cf_filters }) |                 Ok(BundleStatus::Processed { cf_filters }) | ||||||
|             } |             } | ||||||
|             0x04 => Ok(BundleStatus::Pruned), |             0x04 => Ok(BundleStatus::Pruned), | ||||||
|             0x05 => { |             0x05 => { | ||||||
|                 let num = VarInt::consensus_decode(&mut d)?; |                 let num = VarInt::consensus_decode(d)?; | ||||||
|                 let num = num.0 as usize; |                 let num = num.0 as usize; | ||||||
| 
 | 
 | ||||||
|                 let mut cf_filters = Vec::with_capacity(num); |                 let mut cf_filters = Vec::with_capacity(num); | ||||||
|                 for _ in 0..num { |                 for _ in 0..num { | ||||||
|                     cf_filters.push(Vec::<u8>::consensus_decode(&mut d)?); |                     cf_filters.push(Vec::<u8>::consensus_decode(d)?); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 Ok(BundleStatus::Tip { cf_filters }) |                 Ok(BundleStatus::Tip { cf_filters }) | ||||||
| @ -276,7 +278,11 @@ impl ChainStore<Full> { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn start_snapshot(&self, from: usize) -> Result<ChainStore<Snapshot>, CompactFiltersError> { |     pub fn start_snapshot(&self, from: usize) -> Result<ChainStore<Snapshot>, CompactFiltersError> { | ||||||
|         let new_cf_name: String = thread_rng().sample_iter(&Alphanumeric).take(16).collect(); |         let new_cf_name: String = thread_rng() | ||||||
|  |             .sample_iter(&Alphanumeric) | ||||||
|  |             .map(|byte| byte as char) | ||||||
|  |             .take(16) | ||||||
|  |             .collect(); | ||||||
|         let new_cf_name = format!("_headers:{}", new_cf_name); |         let new_cf_name = format!("_headers:{}", new_cf_name); | ||||||
| 
 | 
 | ||||||
|         let mut write_store = self.store.write().unwrap(); |         let mut write_store = self.store.write().unwrap(); | ||||||
| @ -647,7 +653,7 @@ impl CfStore { | |||||||
|                     &first_key, |                     &first_key, | ||||||
|                     ( |                     ( | ||||||
|                         BundleStatus::Init, |                         BundleStatus::Init, | ||||||
|                         filter.filter_header(&FilterHeader::from_hash(Default::default())), |                         filter.filter_header(&FilterHeader::from_hash(Hash::all_zeros())), | ||||||
|                     ) |                     ) | ||||||
|                         .serialize(), |                         .serialize(), | ||||||
|                 )?; |                 )?; | ||||||
|  | |||||||
| @ -14,6 +14,7 @@ use std::sync::{Arc, Mutex}; | |||||||
| use std::time::Duration; | use std::time::Duration; | ||||||
| 
 | 
 | ||||||
| use bitcoin::hash_types::{BlockHash, FilterHeader}; | use bitcoin::hash_types::{BlockHash, FilterHeader}; | ||||||
|  | use bitcoin::hashes::Hash; | ||||||
| use bitcoin::network::message::NetworkMessage; | use bitcoin::network::message::NetworkMessage; | ||||||
| use bitcoin::network::message_blockdata::GetHeadersMessage; | use bitcoin::network::message_blockdata::GetHeadersMessage; | ||||||
| use bitcoin::util::bip158::BlockFilter; | use bitcoin::util::bip158::BlockFilter; | ||||||
| @ -254,7 +255,7 @@ where | |||||||
| 
 | 
 | ||||||
|     peer.send(NetworkMessage::GetHeaders(GetHeadersMessage::new( |     peer.send(NetworkMessage::GetHeaders(GetHeadersMessage::new( | ||||||
|         locators_vec, |         locators_vec, | ||||||
|         Default::default(), |         Hash::all_zeros(), | ||||||
|     )))?; |     )))?; | ||||||
|     let (mut snapshot, mut last_hash) = if let NetworkMessage::Headers(headers) = peer |     let (mut snapshot, mut last_hash) = if let NetworkMessage::Headers(headers) = peer | ||||||
|         .recv("headers", Some(Duration::from_secs(TIMEOUT_SECS)))? |         .recv("headers", Some(Duration::from_secs(TIMEOUT_SECS)))? | ||||||
| @ -276,7 +277,7 @@ where | |||||||
|     while sync_height < peer.get_version().start_height as usize { |     while sync_height < peer.get_version().start_height as usize { | ||||||
|         peer.send(NetworkMessage::GetHeaders(GetHeadersMessage::new( |         peer.send(NetworkMessage::GetHeaders(GetHeadersMessage::new( | ||||||
|             vec![last_hash], |             vec![last_hash], | ||||||
|             Default::default(), |             Hash::all_zeros(), | ||||||
|         )))?; |         )))?; | ||||||
|         if let NetworkMessage::Headers(headers) = peer |         if let NetworkMessage::Headers(headers) = peer | ||||||
|             .recv("headers", Some(Duration::from_secs(TIMEOUT_SECS)))? |             .recv("headers", Some(Duration::from_secs(TIMEOUT_SECS)))? | ||||||
|  | |||||||
| @ -296,7 +296,7 @@ impl<'a, 'b, D: Database> TxCache<'a, 'b, D> { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Configuration for an [`ElectrumBlockchain`]
 | /// Configuration for an [`ElectrumBlockchain`]
 | ||||||
| #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq)] | #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq, Eq)] | ||||||
| pub struct ElectrumBlockchainConfig { | pub struct ElectrumBlockchainConfig { | ||||||
|     /// URL of the Electrum server (such as ElectrumX, Esplora, BWT) may start with `ssl://` or `tcp://` and include a port
 |     /// URL of the Electrum server (such as ElectrumX, Esplora, BWT) may start with `ssl://` or `tcp://` and include a port
 | ||||||
|     ///
 |     ///
 | ||||||
|  | |||||||
| @ -125,8 +125,9 @@ impl GetTx for EsploraBlockchain { | |||||||
| #[maybe_async] | #[maybe_async] | ||||||
| impl GetBlockHash for EsploraBlockchain { | impl GetBlockHash for EsploraBlockchain { | ||||||
|     fn get_block_hash(&self, height: u64) -> Result<BlockHash, Error> { |     fn get_block_hash(&self, height: u64) -> Result<BlockHash, Error> { | ||||||
|         let block_header = await_or_block!(self.url_client.get_header(height as u32))?; |         Ok(await_or_block!(self | ||||||
|         Ok(block_header.block_hash()) |             .url_client | ||||||
|  |             .get_block_hash(height as u32))?) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -110,8 +110,7 @@ impl GetTx for EsploraBlockchain { | |||||||
| 
 | 
 | ||||||
| impl GetBlockHash for EsploraBlockchain { | impl GetBlockHash for EsploraBlockchain { | ||||||
|     fn get_block_hash(&self, height: u64) -> Result<BlockHash, Error> { |     fn get_block_hash(&self, height: u64) -> Result<BlockHash, Error> { | ||||||
|         let block_header = self.url_client.get_header(height as u32)?; |         Ok(self.url_client.get_block_hash(height as u32)?) | ||||||
|         Ok(block_header.block_hash()) |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -33,7 +33,7 @@ mod blocking; | |||||||
| pub use self::blocking::*; | pub use self::blocking::*; | ||||||
| 
 | 
 | ||||||
| /// Configuration for an [`EsploraBlockchain`]
 | /// Configuration for an [`EsploraBlockchain`]
 | ||||||
| #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq)] | #[derive(Debug, serde::Deserialize, serde::Serialize, Clone, PartialEq, Eq)] | ||||||
| pub struct EsploraBlockchainConfig { | pub struct EsploraBlockchainConfig { | ||||||
|     /// Base URL of the esplora service
 |     /// Base URL of the esplora service
 | ||||||
|     ///
 |     ///
 | ||||||
|  | |||||||
| @ -77,7 +77,7 @@ impl Deref for RpcBlockchain { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// RpcBlockchain configuration options
 | /// RpcBlockchain configuration options
 | ||||||
| #[derive(Debug, Serialize, Deserialize, Clone, PartialEq)] | #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)] | ||||||
| pub struct RpcConfig { | pub struct RpcConfig { | ||||||
|     /// The bitcoin node url
 |     /// The bitcoin node url
 | ||||||
|     pub url: String, |     pub url: String, | ||||||
| @ -96,7 +96,7 @@ pub struct RpcConfig { | |||||||
| /// In general, BDK tries to sync `scriptPubKey`s cached in [`crate::database::Database`] with
 | /// In general, BDK tries to sync `scriptPubKey`s cached in [`crate::database::Database`] with
 | ||||||
| /// `scriptPubKey`s imported in the Bitcoin Core Wallet. These parameters are used for determining
 | /// `scriptPubKey`s imported in the Bitcoin Core Wallet. These parameters are used for determining
 | ||||||
| /// how the `importdescriptors` RPC calls are to be made.
 | /// how the `importdescriptors` RPC calls are to be made.
 | ||||||
| #[derive(Debug, Serialize, Deserialize, Clone, PartialEq)] | #[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)] | ||||||
| pub struct RpcSyncParams { | pub struct RpcSyncParams { | ||||||
|     /// The minimum number of scripts to scan for on initial sync.
 |     /// The minimum number of scripts to scan for on initial sync.
 | ||||||
|     pub start_script_count: usize, |     pub start_script_count: usize, | ||||||
| @ -167,7 +167,7 @@ impl Blockchain for RpcBlockchain { | |||||||
|             .estimate_smart_fee(target as u16, None)? |             .estimate_smart_fee(target as u16, None)? | ||||||
|             .fee_rate |             .fee_rate | ||||||
|             .ok_or(Error::FeeRateUnavailable)? |             .ok_or(Error::FeeRateUnavailable)? | ||||||
|             .as_sat() as f64; |             .to_sat() as f64; | ||||||
| 
 | 
 | ||||||
|         Ok(FeeRate::from_sat_per_vb((sat_per_kb / 1000f64) as f32)) |         Ok(FeeRate::from_sat_per_vb((sat_per_kb / 1000f64) as f32)) | ||||||
|     } |     } | ||||||
| @ -410,7 +410,12 @@ impl<'a, D: BatchDatabase> DbState<'a, D> { | |||||||
|                 updated = true; |                 updated = true; | ||||||
|                 TransactionDetails { |                 TransactionDetails { | ||||||
|                     txid: tx_res.info.txid, |                     txid: tx_res.info.txid, | ||||||
|                     ..Default::default() |                     transaction: None, | ||||||
|  | 
 | ||||||
|  |                     received: 0, | ||||||
|  |                     sent: 0, | ||||||
|  |                     fee: None, | ||||||
|  |                     confirmation_time: None, | ||||||
|                 } |                 } | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
| @ -430,7 +435,7 @@ impl<'a, D: BatchDatabase> DbState<'a, D> { | |||||||
|             // update fee (if needed)
 |             // update fee (if needed)
 | ||||||
|             if let (None, Some(new_fee)) = (db_tx.fee, tx_res.detail.fee) { |             if let (None, Some(new_fee)) = (db_tx.fee, tx_res.detail.fee) { | ||||||
|                 updated = true; |                 updated = true; | ||||||
|                 db_tx.fee = Some(new_fee.as_sat().unsigned_abs()); |                 db_tx.fee = Some(new_fee.to_sat().unsigned_abs()); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             // update confirmation time (if needed)
 |             // update confirmation time (if needed)
 | ||||||
| @ -603,7 +608,7 @@ impl<'a, D: BatchDatabase> DbState<'a, D> { | |||||||
|         LocalUtxo { |         LocalUtxo { | ||||||
|             outpoint: OutPoint::new(entry.txid, entry.vout), |             outpoint: OutPoint::new(entry.txid, entry.vout), | ||||||
|             txout: TxOut { |             txout: TxOut { | ||||||
|                 value: entry.amount.as_sat(), |                 value: entry.amount.to_sat(), | ||||||
|                 script_pubkey: entry.script_pub_key, |                 script_pubkey: entry.script_pub_key, | ||||||
|             }, |             }, | ||||||
|             keychain, |             keychain, | ||||||
| @ -873,15 +878,13 @@ impl BlockchainFactory for RpcBlockchainFactory { | |||||||
| mod test { | mod test { | ||||||
|     use super::*; |     use super::*; | ||||||
|     use crate::{ |     use crate::{ | ||||||
|         descriptor::{into_wallet_descriptor_checked, AsDerived}, |         descriptor::into_wallet_descriptor_checked, testutils::blockchain_tests::TestClient, | ||||||
|         testutils::blockchain_tests::TestClient, |  | ||||||
|         wallet::utils::SecpCtx, |         wallet::utils::SecpCtx, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     use bitcoin::{Address, Network}; |     use bitcoin::{Address, Network}; | ||||||
|     use bitcoincore_rpc::RpcApi; |     use bitcoincore_rpc::RpcApi; | ||||||
|     use log::LevelFilter; |     use log::LevelFilter; | ||||||
|     use miniscript::DescriptorTrait; |  | ||||||
| 
 | 
 | ||||||
|     crate::bdk_blockchain_tests! { |     crate::bdk_blockchain_tests! { | ||||||
|         fn test_instance(test_client: &TestClient) -> RpcBlockchain { |         fn test_instance(test_client: &TestClient) -> RpcBlockchain { | ||||||
| @ -958,7 +961,7 @@ mod test { | |||||||
| 
 | 
 | ||||||
|         // generate scripts (1 tx per script)
 |         // generate scripts (1 tx per script)
 | ||||||
|         let scripts = (0..TX_COUNT) |         let scripts = (0..TX_COUNT) | ||||||
|             .map(|index| desc.as_derived(index, &secp).script_pubkey()) |             .map(|index| desc.at_derivation_index(index).script_pubkey()) | ||||||
|             .collect::<Vec<_>>(); |             .collect::<Vec<_>>(); | ||||||
| 
 | 
 | ||||||
|         // import scripts and wait
 |         // import scripts and wait
 | ||||||
|  | |||||||
| @ -497,7 +497,7 @@ macro_rules! populate_test_db { | |||||||
|         } |         } | ||||||
|         let tx = $crate::bitcoin::Transaction { |         let tx = $crate::bitcoin::Transaction { | ||||||
|             version: 1, |             version: 1, | ||||||
|             lock_time: 0, |             lock_time: bitcoin::PackedLockTime(0), | ||||||
|             input, |             input, | ||||||
|             output: tx_meta |             output: tx_meta | ||||||
|                 .output |                 .output | ||||||
|  | |||||||
| @ -1,210 +0,0 @@ | |||||||
| // Bitcoin Dev Kit
 |  | ||||||
| // Written in 2020 by Alekos Filini <alekos.filini@gmail.com>
 |  | ||||||
| //
 |  | ||||||
| // Copyright (c) 2020-2021 Bitcoin Dev Kit Developers
 |  | ||||||
| //
 |  | ||||||
| // This file is licensed under the Apache License, Version 2.0 <LICENSE-APACHE
 |  | ||||||
| // or http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
 |  | ||||||
| // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your option.
 |  | ||||||
| // You may not use this file except in accordance with one or both of these
 |  | ||||||
| // licenses.
 |  | ||||||
| 
 |  | ||||||
| //! Derived descriptor keys
 |  | ||||||
| //!
 |  | ||||||
| //! The [`DerivedDescriptorKey`] type is a wrapper over the standard [`DescriptorPublicKey`] which
 |  | ||||||
| //! guarantees that all the extended keys have a fixed derivation path, i.e. all the wildcards have
 |  | ||||||
| //! been replaced by actual derivation indexes.
 |  | ||||||
| //!
 |  | ||||||
| //! The [`AsDerived`] trait provides a quick way to derive descriptors to obtain a
 |  | ||||||
| //! `Descriptor<DerivedDescriptorKey>` type. This, in turn, can be used to derive public
 |  | ||||||
| //! keys for arbitrary derivation indexes.
 |  | ||||||
| //!
 |  | ||||||
| //! Combining this with [`Wallet::get_signers`], secret keys can also be derived.
 |  | ||||||
| //!
 |  | ||||||
| //! # Example
 |  | ||||||
| //!
 |  | ||||||
| //! ```
 |  | ||||||
| //! # use std::str::FromStr;
 |  | ||||||
| //! # use bitcoin::secp256k1::Secp256k1;
 |  | ||||||
| //! use bdk::descriptor::{AsDerived, DescriptorPublicKey};
 |  | ||||||
| //! use bdk::miniscript::{ToPublicKey, TranslatePk, MiniscriptKey};
 |  | ||||||
| //!
 |  | ||||||
| //! let secp = Secp256k1::gen_new();
 |  | ||||||
| //!
 |  | ||||||
| //! let key = DescriptorPublicKey::from_str("[aa600a45/84'/0'/0']tpubDCbDXFKoLTQp44wQuC12JgSn5g9CWGjZdpBHeTqyypZ4VvgYjTJmK9CkyR5bFvG9f4PutvwmvpYCLkFx2rpx25hiMs4sUgxJveW8ZzSAVAc/0/*")?;
 |  | ||||||
| //! let (descriptor, _, _) = bdk::descriptor!(wpkh(key))?;
 |  | ||||||
| //!
 |  | ||||||
| //! // derived: wpkh([aa600a45/84'/0'/0']tpubDCbDXFKoLTQp44wQuC12JgSn5g9CWGjZdpBHeTqyypZ4VvgYjTJmK9CkyR5bFvG9f4PutvwmvpYCLkFx2rpx25hiMs4sUgxJveW8ZzSAVAc/0/42)#3ladd0t2
 |  | ||||||
| //! let derived = descriptor.as_derived(42, &secp);
 |  | ||||||
| //! println!("derived: {}", derived);
 |  | ||||||
| //!
 |  | ||||||
| //! // with_pks: wpkh(02373ecb54c5e83bd7e0d40adf78b65efaf12fafb13571f0261fc90364eee22e1e)#p4jjgvll
 |  | ||||||
| //! let with_pks = derived.translate_pk_infallible(|pk| pk.to_public_key(), |pkh| pkh.to_public_key().to_pubkeyhash());
 |  | ||||||
| //! println!("with_pks: {}", with_pks);
 |  | ||||||
| //! # Ok::<(), Box<dyn std::error::Error>>(())
 |  | ||||||
| //! ```
 |  | ||||||
| //!
 |  | ||||||
| //! [`Wallet::get_signers`]: crate::wallet::Wallet::get_signers
 |  | ||||||
| 
 |  | ||||||
| use std::cmp::Ordering; |  | ||||||
| use std::fmt; |  | ||||||
| use std::hash::{Hash, Hasher}; |  | ||||||
| use std::ops::Deref; |  | ||||||
| 
 |  | ||||||
| use bitcoin::hashes::hash160; |  | ||||||
| use bitcoin::{PublicKey, XOnlyPublicKey}; |  | ||||||
| 
 |  | ||||||
| use miniscript::descriptor::{DescriptorSinglePub, SinglePubKey, Wildcard}; |  | ||||||
| use miniscript::{Descriptor, DescriptorPublicKey, MiniscriptKey, ToPublicKey, TranslatePk}; |  | ||||||
| 
 |  | ||||||
| use crate::wallet::utils::SecpCtx; |  | ||||||
| 
 |  | ||||||
| /// Extended [`DescriptorPublicKey`] that has been derived
 |  | ||||||
| ///
 |  | ||||||
| /// Derived keys are guaranteed to never contain wildcards of any kind
 |  | ||||||
| #[derive(Debug, Clone)] |  | ||||||
| pub struct DerivedDescriptorKey<'s>(DescriptorPublicKey, &'s SecpCtx); |  | ||||||
| 
 |  | ||||||
| impl<'s> DerivedDescriptorKey<'s> { |  | ||||||
|     /// Construct a new derived key
 |  | ||||||
|     ///
 |  | ||||||
|     /// Panics if the key is wildcard
 |  | ||||||
|     pub fn new(key: DescriptorPublicKey, secp: &'s SecpCtx) -> DerivedDescriptorKey<'s> { |  | ||||||
|         if let DescriptorPublicKey::XPub(xpub) = &key { |  | ||||||
|             assert!(xpub.wildcard == Wildcard::None) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         DerivedDescriptorKey(key, secp) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> Deref for DerivedDescriptorKey<'s> { |  | ||||||
|     type Target = DescriptorPublicKey; |  | ||||||
| 
 |  | ||||||
|     fn deref(&self) -> &Self::Target { |  | ||||||
|         &self.0 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> PartialEq for DerivedDescriptorKey<'s> { |  | ||||||
|     fn eq(&self, other: &Self) -> bool { |  | ||||||
|         self.0 == other.0 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> Eq for DerivedDescriptorKey<'s> {} |  | ||||||
| 
 |  | ||||||
| impl<'s> PartialOrd for DerivedDescriptorKey<'s> { |  | ||||||
|     fn partial_cmp(&self, other: &Self) -> Option<Ordering> { |  | ||||||
|         self.0.partial_cmp(&other.0) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> Ord for DerivedDescriptorKey<'s> { |  | ||||||
|     fn cmp(&self, other: &Self) -> Ordering { |  | ||||||
|         self.0.cmp(&other.0) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> fmt::Display for DerivedDescriptorKey<'s> { |  | ||||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |  | ||||||
|         self.0.fmt(f) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> Hash for DerivedDescriptorKey<'s> { |  | ||||||
|     fn hash<H: Hasher>(&self, state: &mut H) { |  | ||||||
|         self.0.hash(state); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> MiniscriptKey for DerivedDescriptorKey<'s> { |  | ||||||
|     type Hash = Self; |  | ||||||
| 
 |  | ||||||
|     fn to_pubkeyhash(&self) -> Self::Hash { |  | ||||||
|         DerivedDescriptorKey(self.0.to_pubkeyhash(), self.1) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn is_uncompressed(&self) -> bool { |  | ||||||
|         self.0.is_uncompressed() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> ToPublicKey for DerivedDescriptorKey<'s> { |  | ||||||
|     fn to_public_key(&self) -> PublicKey { |  | ||||||
|         match &self.0 { |  | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |  | ||||||
|                 key: SinglePubKey::XOnly(_), |  | ||||||
|                 .. |  | ||||||
|             }) => panic!("Found x-only public key in non-tr descriptor"), |  | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |  | ||||||
|                 key: SinglePubKey::FullKey(ref pk), |  | ||||||
|                 .. |  | ||||||
|             }) => *pk, |  | ||||||
|             DescriptorPublicKey::XPub(ref xpub) => PublicKey::new( |  | ||||||
|                 xpub.xkey |  | ||||||
|                     .derive_pub(self.1, &xpub.derivation_path) |  | ||||||
|                     .expect("Shouldn't fail, only normal derivations") |  | ||||||
|                     .public_key, |  | ||||||
|             ), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn to_x_only_pubkey(&self) -> XOnlyPublicKey { |  | ||||||
|         match &self.0 { |  | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |  | ||||||
|                 key: SinglePubKey::XOnly(ref pk), |  | ||||||
|                 .. |  | ||||||
|             }) => *pk, |  | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |  | ||||||
|                 key: SinglePubKey::FullKey(ref pk), |  | ||||||
|                 .. |  | ||||||
|             }) => XOnlyPublicKey::from(pk.inner), |  | ||||||
|             DescriptorPublicKey::XPub(ref xpub) => XOnlyPublicKey::from( |  | ||||||
|                 xpub.xkey |  | ||||||
|                     .derive_pub(self.1, &xpub.derivation_path) |  | ||||||
|                     .expect("Shouldn't fail, only normal derivations") |  | ||||||
|                     .public_key, |  | ||||||
|             ), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn hash_to_hash160(hash: &Self::Hash) -> hash160::Hash { |  | ||||||
|         hash.to_public_key().to_pubkeyhash() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Utilities to derive descriptors
 |  | ||||||
| ///
 |  | ||||||
| /// Check out the [module level] documentation for more.
 |  | ||||||
| ///
 |  | ||||||
| /// [module level]: crate::descriptor::derived
 |  | ||||||
| pub trait AsDerived { |  | ||||||
|     /// Derive a descriptor and transform all of its keys to `DerivedDescriptorKey`
 |  | ||||||
|     fn as_derived<'s>(&self, index: u32, secp: &'s SecpCtx) |  | ||||||
|         -> Descriptor<DerivedDescriptorKey<'s>>; |  | ||||||
| 
 |  | ||||||
|     /// Transform the keys into `DerivedDescriptorKey`.
 |  | ||||||
|     ///
 |  | ||||||
|     /// Panics if the descriptor is not "fixed", i.e. if it's derivable
 |  | ||||||
|     fn as_derived_fixed<'s>(&self, secp: &'s SecpCtx) -> Descriptor<DerivedDescriptorKey<'s>>; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl AsDerived for Descriptor<DescriptorPublicKey> { |  | ||||||
|     fn as_derived<'s>( |  | ||||||
|         &self, |  | ||||||
|         index: u32, |  | ||||||
|         secp: &'s SecpCtx, |  | ||||||
|     ) -> Descriptor<DerivedDescriptorKey<'s>> { |  | ||||||
|         self.derive(index).translate_pk_infallible( |  | ||||||
|             |key| DerivedDescriptorKey::new(key.clone(), secp), |  | ||||||
|             |key| DerivedDescriptorKey::new(key.clone(), secp), |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn as_derived_fixed<'s>(&self, secp: &'s SecpCtx) -> Descriptor<DerivedDescriptorKey<'s>> { |  | ||||||
|         assert!(!self.is_deriveable()); |  | ||||||
| 
 |  | ||||||
|         self.as_derived(0, secp) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -700,10 +700,10 @@ macro_rules! fragment { | |||||||
|         $crate::keys::make_pkh($key, &secp) |         $crate::keys::make_pkh($key, &secp) | ||||||
|     }); |     }); | ||||||
|     ( after ( $value:expr ) ) => ({ |     ( after ( $value:expr ) ) => ({ | ||||||
|         $crate::impl_leaf_opcode_value!(After, $value) |         $crate::impl_leaf_opcode_value!(After, $crate::bitcoin::PackedLockTime($value)) // TODO!! https://github.com/rust-bitcoin/rust-bitcoin/issues/1302
 | ||||||
|     }); |     }); | ||||||
|     ( older ( $value:expr ) ) => ({ |     ( older ( $value:expr ) ) => ({ | ||||||
|         $crate::impl_leaf_opcode_value!(Older, $value) |         $crate::impl_leaf_opcode_value!(Older, $crate::bitcoin::Sequence($value)) // TODO!!
 | ||||||
|     }); |     }); | ||||||
|     ( sha256 ( $hash:expr ) ) => ({ |     ( sha256 ( $hash:expr ) ) => ({ | ||||||
|         $crate::impl_leaf_opcode_value!(Sha256, $hash) |         $crate::impl_leaf_opcode_value!(Sha256, $hash) | ||||||
| @ -795,7 +795,7 @@ macro_rules! fragment { | |||||||
| mod test { | mod test { | ||||||
|     use bitcoin::hashes::hex::ToHex; |     use bitcoin::hashes::hex::ToHex; | ||||||
|     use bitcoin::secp256k1::Secp256k1; |     use bitcoin::secp256k1::Secp256k1; | ||||||
|     use miniscript::descriptor::{DescriptorPublicKey, DescriptorTrait, KeyMap}; |     use miniscript::descriptor::{DescriptorPublicKey, KeyMap}; | ||||||
|     use miniscript::{Descriptor, Legacy, Segwitv0}; |     use miniscript::{Descriptor, Legacy, Segwitv0}; | ||||||
| 
 | 
 | ||||||
|     use std::str::FromStr; |     use std::str::FromStr; | ||||||
| @ -806,8 +806,6 @@ mod test { | |||||||
|     use bitcoin::util::bip32; |     use bitcoin::util::bip32; | ||||||
|     use bitcoin::PrivateKey; |     use bitcoin::PrivateKey; | ||||||
| 
 | 
 | ||||||
|     use crate::descriptor::derived::AsDerived; |  | ||||||
| 
 |  | ||||||
|     // test the descriptor!() macro
 |     // test the descriptor!() macro
 | ||||||
| 
 | 
 | ||||||
|     // verify descriptor generates expected script(s) (if bare or pk) or address(es)
 |     // verify descriptor generates expected script(s) (if bare or pk) or address(es)
 | ||||||
| @ -817,17 +815,15 @@ mod test { | |||||||
|         is_fixed: bool, |         is_fixed: bool, | ||||||
|         expected: &[&str], |         expected: &[&str], | ||||||
|     ) { |     ) { | ||||||
|         let secp = Secp256k1::new(); |  | ||||||
| 
 |  | ||||||
|         let (desc, _key_map, _networks) = desc.unwrap(); |         let (desc, _key_map, _networks) = desc.unwrap(); | ||||||
|         assert_eq!(desc.is_witness(), is_witness); |         assert_eq!(desc.is_witness(), is_witness); | ||||||
|         assert_eq!(!desc.is_deriveable(), is_fixed); |         assert_eq!(!desc.has_wildcard(), is_fixed); | ||||||
|         for i in 0..expected.len() { |         for i in 0..expected.len() { | ||||||
|             let index = i as u32; |             let index = i as u32; | ||||||
|             let child_desc = if !desc.is_deriveable() { |             let child_desc = if !desc.has_wildcard() { | ||||||
|                 desc.as_derived_fixed(&secp) |                 desc.at_derivation_index(0) | ||||||
|             } else { |             } else { | ||||||
|                 desc.as_derived(index, &secp) |                 desc.at_derivation_index(index) | ||||||
|             }; |             }; | ||||||
|             let address = child_desc.address(Regtest); |             let address = child_desc.address(Regtest); | ||||||
|             if let Ok(address) = address { |             if let Ok(address) = address { | ||||||
|  | |||||||
| @ -15,24 +15,22 @@ | |||||||
| //! from [`miniscript`].
 | //! from [`miniscript`].
 | ||||||
| 
 | 
 | ||||||
| use std::collections::BTreeMap; | use std::collections::BTreeMap; | ||||||
| use std::ops::Deref; |  | ||||||
| 
 | 
 | ||||||
| use bitcoin::util::bip32::{ChildNumber, DerivationPath, ExtendedPubKey, Fingerprint, KeySource}; | use bitcoin::util::bip32::{ChildNumber, DerivationPath, ExtendedPubKey, Fingerprint, KeySource}; | ||||||
| use bitcoin::util::{psbt, taproot}; | use bitcoin::util::{psbt, taproot}; | ||||||
| use bitcoin::{secp256k1, PublicKey, XOnlyPublicKey}; | use bitcoin::{secp256k1, PublicKey, XOnlyPublicKey}; | ||||||
| use bitcoin::{Network, Script, TxOut}; | use bitcoin::{Network, TxOut}; | ||||||
| 
 | 
 | ||||||
| use miniscript::descriptor::{DescriptorType, InnerXKey, SinglePubKey}; | use miniscript::descriptor::{DefiniteDescriptorKey, DescriptorType, InnerXKey, SinglePubKey}; | ||||||
| pub use miniscript::{ | pub use miniscript::{ | ||||||
|     descriptor::DescriptorXKey, descriptor::KeyMap, descriptor::Wildcard, Descriptor, |     descriptor::DescriptorXKey, descriptor::KeyMap, descriptor::Wildcard, Descriptor, | ||||||
|     DescriptorPublicKey, Legacy, Miniscript, ScriptContext, Segwitv0, |     DescriptorPublicKey, Legacy, Miniscript, ScriptContext, Segwitv0, | ||||||
| }; | }; | ||||||
| use miniscript::{DescriptorTrait, ForEachKey, TranslatePk}; | use miniscript::{ForEachKey, MiniscriptKey, TranslatePk}; | ||||||
| 
 | 
 | ||||||
| use crate::descriptor::policy::BuildSatisfaction; | use crate::descriptor::policy::BuildSatisfaction; | ||||||
| 
 | 
 | ||||||
| pub mod checksum; | pub mod checksum; | ||||||
| pub mod derived; |  | ||||||
| #[doc(hidden)] | #[doc(hidden)] | ||||||
| pub mod dsl; | pub mod dsl; | ||||||
| pub mod error; | pub mod error; | ||||||
| @ -40,7 +38,6 @@ pub mod policy; | |||||||
| pub mod template; | pub mod template; | ||||||
| 
 | 
 | ||||||
| pub use self::checksum::get_checksum; | pub use self::checksum::get_checksum; | ||||||
| pub use self::derived::{AsDerived, DerivedDescriptorKey}; |  | ||||||
| pub use self::error::Error as DescriptorError; | pub use self::error::Error as DescriptorError; | ||||||
| pub use self::policy::Policy; | pub use self::policy::Policy; | ||||||
| use self::template::DescriptorTemplateOut; | use self::template::DescriptorTemplateOut; | ||||||
| @ -52,7 +49,7 @@ use crate::wallet::utils::SecpCtx; | |||||||
| pub type ExtendedDescriptor = Descriptor<DescriptorPublicKey>; | pub type ExtendedDescriptor = Descriptor<DescriptorPublicKey>; | ||||||
| 
 | 
 | ||||||
| /// Alias for a [`Descriptor`] that contains extended **derived** keys
 | /// Alias for a [`Descriptor`] that contains extended **derived** keys
 | ||||||
| pub type DerivedDescriptor<'s> = Descriptor<DerivedDescriptorKey<'s>>; | pub type DerivedDescriptor = Descriptor<DefiniteDescriptorKey>; | ||||||
| 
 | 
 | ||||||
| /// Alias for the type of maps that represent derivation paths in a [`psbt::Input`] or
 | /// Alias for the type of maps that represent derivation paths in a [`psbt::Input`] or
 | ||||||
| /// [`psbt::Output`]
 | /// [`psbt::Output`]
 | ||||||
| @ -132,8 +129,27 @@ impl IntoWalletDescriptor for (ExtendedDescriptor, KeyMap) { | |||||||
|     ) -> Result<(ExtendedDescriptor, KeyMap), DescriptorError> { |     ) -> Result<(ExtendedDescriptor, KeyMap), DescriptorError> { | ||||||
|         use crate::keys::DescriptorKey; |         use crate::keys::DescriptorKey; | ||||||
| 
 | 
 | ||||||
|         let check_key = |pk: &DescriptorPublicKey| { |         struct Translator<'s, 'd> { | ||||||
|             let (pk, _, networks) = if self.0.is_witness() { |             secp: &'s SecpCtx, | ||||||
|  |             descriptor: &'d ExtendedDescriptor, | ||||||
|  |             network: Network, | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         impl<'s, 'd> | ||||||
|  |             miniscript::Translator<DescriptorPublicKey, miniscript::DummyKey, DescriptorError> | ||||||
|  |             for Translator<'s, 'd> | ||||||
|  |         { | ||||||
|  |             fn pk( | ||||||
|  |                 &mut self, | ||||||
|  |                 pk: &DescriptorPublicKey, | ||||||
|  |             ) -> Result<miniscript::DummyKey, DescriptorError> { | ||||||
|  |                 let secp = &self.secp; | ||||||
|  | 
 | ||||||
|  |                 let (_, _, networks) = if self.descriptor.is_taproot() { | ||||||
|  |                     let descriptor_key: DescriptorKey<miniscript::Tap> = | ||||||
|  |                         pk.clone().into_descriptor_key()?; | ||||||
|  |                     descriptor_key.extract(secp)? | ||||||
|  |                 } else if self.descriptor.is_witness() { | ||||||
|                     let descriptor_key: DescriptorKey<miniscript::Segwitv0> = |                     let descriptor_key: DescriptorKey<miniscript::Segwitv0> = | ||||||
|                         pk.clone().into_descriptor_key()?; |                         pk.clone().into_descriptor_key()?; | ||||||
|                     descriptor_key.extract(secp)? |                     descriptor_key.extract(secp)? | ||||||
| @ -143,17 +159,46 @@ impl IntoWalletDescriptor for (ExtendedDescriptor, KeyMap) { | |||||||
|                     descriptor_key.extract(secp)? |                     descriptor_key.extract(secp)? | ||||||
|                 }; |                 }; | ||||||
| 
 | 
 | ||||||
|             if networks.contains(&network) { |                 if networks.contains(&self.network) { | ||||||
|                 Ok(pk) |                     Ok(miniscript::DummyKey) | ||||||
|                 } else { |                 } else { | ||||||
|                     Err(DescriptorError::Key(KeyError::InvalidNetwork)) |                     Err(DescriptorError::Key(KeyError::InvalidNetwork)) | ||||||
|                 } |                 } | ||||||
|         }; |             } | ||||||
|  |             fn sha256( | ||||||
|  |                 &mut self, | ||||||
|  |                 _sha256: &<DescriptorPublicKey as MiniscriptKey>::Sha256, | ||||||
|  |             ) -> Result<miniscript::DummySha256Hash, DescriptorError> { | ||||||
|  |                 Ok(Default::default()) | ||||||
|  |             } | ||||||
|  |             fn hash256( | ||||||
|  |                 &mut self, | ||||||
|  |                 _hash256: &<DescriptorPublicKey as MiniscriptKey>::Hash256, | ||||||
|  |             ) -> Result<miniscript::DummyHash256Hash, DescriptorError> { | ||||||
|  |                 Ok(Default::default()) | ||||||
|  |             } | ||||||
|  |             fn ripemd160( | ||||||
|  |                 &mut self, | ||||||
|  |                 _ripemd160: &<DescriptorPublicKey as MiniscriptKey>::Ripemd160, | ||||||
|  |             ) -> Result<miniscript::DummyRipemd160Hash, DescriptorError> { | ||||||
|  |                 Ok(Default::default()) | ||||||
|  |             } | ||||||
|  |             fn hash160( | ||||||
|  |                 &mut self, | ||||||
|  |                 _hash160: &<DescriptorPublicKey as MiniscriptKey>::Hash160, | ||||||
|  |             ) -> Result<miniscript::DummyHash160Hash, DescriptorError> { | ||||||
|  |                 Ok(Default::default()) | ||||||
|  |             } | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         // check the network for the keys
 |         // check the network for the keys
 | ||||||
|         let translated = self.0.translate_pk(check_key, check_key)?; |         self.0.translate_pk(&mut Translator { | ||||||
|  |             secp, | ||||||
|  |             network, | ||||||
|  |             descriptor: &self.0, | ||||||
|  |         })?; | ||||||
| 
 | 
 | ||||||
|         Ok((translated, self.1)) |         Ok(self) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -163,10 +208,17 @@ impl IntoWalletDescriptor for DescriptorTemplateOut { | |||||||
|         _secp: &SecpCtx, |         _secp: &SecpCtx, | ||||||
|         network: Network, |         network: Network, | ||||||
|     ) -> Result<(ExtendedDescriptor, KeyMap), DescriptorError> { |     ) -> Result<(ExtendedDescriptor, KeyMap), DescriptorError> { | ||||||
|         let valid_networks = &self.2; |         struct Translator { | ||||||
|  |             network: Network, | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         let fix_key = |pk: &DescriptorPublicKey| { |         impl miniscript::Translator<DescriptorPublicKey, DescriptorPublicKey, DescriptorError> | ||||||
|             if valid_networks.contains(&network) { |             for Translator | ||||||
|  |         { | ||||||
|  |             fn pk( | ||||||
|  |                 &mut self, | ||||||
|  |                 pk: &DescriptorPublicKey, | ||||||
|  |             ) -> Result<DescriptorPublicKey, DescriptorError> { | ||||||
|                 // workaround for xpubs generated by other key types, like bip39: since when the
 |                 // workaround for xpubs generated by other key types, like bip39: since when the
 | ||||||
|                 // conversion is made one network has to be chosen, what we generally choose
 |                 // conversion is made one network has to be chosen, what we generally choose
 | ||||||
|                 // "mainnet", but then override the set of valid networks to specify that all of
 |                 // "mainnet", but then override the set of valid networks to specify that all of
 | ||||||
| @ -175,7 +227,7 @@ impl IntoWalletDescriptor for DescriptorTemplateOut { | |||||||
|                 let pk = match pk { |                 let pk = match pk { | ||||||
|                     DescriptorPublicKey::XPub(ref xpub) => { |                     DescriptorPublicKey::XPub(ref xpub) => { | ||||||
|                         let mut xpub = xpub.clone(); |                         let mut xpub = xpub.clone(); | ||||||
|                         xpub.xkey.network = network; |                         xpub.xkey.network = self.network; | ||||||
| 
 | 
 | ||||||
|                         DescriptorPublicKey::XPub(xpub) |                         DescriptorPublicKey::XPub(xpub) | ||||||
|                     } |                     } | ||||||
| @ -183,13 +235,20 @@ impl IntoWalletDescriptor for DescriptorTemplateOut { | |||||||
|                 }; |                 }; | ||||||
| 
 | 
 | ||||||
|                 Ok(pk) |                 Ok(pk) | ||||||
|             } else { |  | ||||||
|                 Err(DescriptorError::Key(KeyError::InvalidNetwork)) |  | ||||||
|             } |             } | ||||||
|         }; |             miniscript::translate_hash_clone!( | ||||||
|  |                 DescriptorPublicKey, | ||||||
|  |                 DescriptorPublicKey, | ||||||
|  |                 DescriptorError | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if !self.2.contains(&network) { | ||||||
|  |             return Err(DescriptorError::Key(KeyError::InvalidNetwork)); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         // fixup the network for keys that need it
 |         // fixup the network for keys that need it
 | ||||||
|         let translated = self.0.translate_pk(fix_key, fix_key)?; |         let translated = self.0.translate_pk(&mut Translator { network })?; | ||||||
| 
 | 
 | ||||||
|         Ok((translated, self.1)) |         Ok((translated, self.1)) | ||||||
|     } |     } | ||||||
| @ -210,7 +269,7 @@ pub(crate) fn into_wallet_descriptor_checked<T: IntoWalletDescriptor>( | |||||||
|             derivation_path, |             derivation_path, | ||||||
|             wildcard, |             wildcard, | ||||||
|             .. |             .. | ||||||
|         }) = k.as_key() |         }) = k | ||||||
|         { |         { | ||||||
|             return *wildcard == Wildcard::Hardened |             return *wildcard == Wildcard::Hardened | ||||||
|                 || derivation_path.into_iter().any(ChildNumber::is_hardened); |                 || derivation_path.into_iter().any(ChildNumber::is_hardened); | ||||||
| @ -257,7 +316,6 @@ pub trait ExtractPolicy { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub(crate) trait XKeyUtils { | pub(crate) trait XKeyUtils { | ||||||
|     fn full_path(&self, append: &[ChildNumber]) -> DerivationPath; |  | ||||||
|     fn root_fingerprint(&self, secp: &SecpCtx) -> Fingerprint; |     fn root_fingerprint(&self, secp: &SecpCtx) -> Fingerprint; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -265,27 +323,6 @@ impl<T> XKeyUtils for DescriptorXKey<T> | |||||||
| where | where | ||||||
|     T: InnerXKey, |     T: InnerXKey, | ||||||
| { | { | ||||||
|     fn full_path(&self, append: &[ChildNumber]) -> DerivationPath { |  | ||||||
|         let full_path = match self.origin { |  | ||||||
|             Some((_, ref path)) => path |  | ||||||
|                 .into_iter() |  | ||||||
|                 .chain(self.derivation_path.into_iter()) |  | ||||||
|                 .cloned() |  | ||||||
|                 .collect(), |  | ||||||
|             None => self.derivation_path.clone(), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         if self.wildcard != Wildcard::None { |  | ||||||
|             full_path |  | ||||||
|                 .into_iter() |  | ||||||
|                 .chain(append.iter()) |  | ||||||
|                 .cloned() |  | ||||||
|                 .collect() |  | ||||||
|         } else { |  | ||||||
|             full_path |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn root_fingerprint(&self, secp: &SecpCtx) -> Fingerprint { |     fn root_fingerprint(&self, secp: &SecpCtx) -> Fingerprint { | ||||||
|         match self.origin { |         match self.origin { | ||||||
|             Some((fingerprint, _)) => fingerprint, |             Some((fingerprint, _)) => fingerprint, | ||||||
| @ -294,11 +331,6 @@ where | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub(crate) trait DerivedDescriptorMeta { |  | ||||||
|     fn get_hd_keypaths(&self, secp: &SecpCtx) -> HdKeyPaths; |  | ||||||
|     fn get_tap_key_origins(&self, secp: &SecpCtx) -> TapKeyOrigins; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| pub(crate) trait DescriptorMeta { | pub(crate) trait DescriptorMeta { | ||||||
|     fn is_witness(&self) -> bool; |     fn is_witness(&self) -> bool; | ||||||
|     fn is_taproot(&self) -> bool; |     fn is_taproot(&self) -> bool; | ||||||
| @ -307,63 +339,23 @@ pub(crate) trait DescriptorMeta { | |||||||
|         &self, |         &self, | ||||||
|         hd_keypaths: &HdKeyPaths, |         hd_keypaths: &HdKeyPaths, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>>; |     ) -> Option<DerivedDescriptor>; | ||||||
|     fn derive_from_tap_key_origins<'s>( |     fn derive_from_tap_key_origins<'s>( | ||||||
|         &self, |         &self, | ||||||
|         tap_key_origins: &TapKeyOrigins, |         tap_key_origins: &TapKeyOrigins, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>>; |     ) -> Option<DerivedDescriptor>; | ||||||
|     fn derive_from_psbt_key_origins<'s>( |     fn derive_from_psbt_key_origins<'s>( | ||||||
|         &self, |         &self, | ||||||
|         key_origins: BTreeMap<Fingerprint, (&DerivationPath, SinglePubKey)>, |         key_origins: BTreeMap<Fingerprint, (&DerivationPath, SinglePubKey)>, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>>; |     ) -> Option<DerivedDescriptor>; | ||||||
|     fn derive_from_psbt_input<'s>( |     fn derive_from_psbt_input<'s>( | ||||||
|         &self, |         &self, | ||||||
|         psbt_input: &psbt::Input, |         psbt_input: &psbt::Input, | ||||||
|         utxo: Option<TxOut>, |         utxo: Option<TxOut>, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>>; |     ) -> Option<DerivedDescriptor>; | ||||||
| } |  | ||||||
| 
 |  | ||||||
| pub(crate) trait DescriptorScripts { |  | ||||||
|     fn psbt_redeem_script(&self) -> Option<Script>; |  | ||||||
|     fn psbt_witness_script(&self) -> Option<Script>; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'s> DescriptorScripts for DerivedDescriptor<'s> { |  | ||||||
|     fn psbt_redeem_script(&self) -> Option<Script> { |  | ||||||
|         match self.desc_type() { |  | ||||||
|             DescriptorType::ShWpkh => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::ShWsh => Some(self.explicit_script().unwrap().to_v0_p2wsh()), |  | ||||||
|             DescriptorType::Sh => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::Bare => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::ShSortedMulti => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::ShWshSortedMulti => Some(self.explicit_script().unwrap().to_v0_p2wsh()), |  | ||||||
|             DescriptorType::Pkh |  | ||||||
|             | DescriptorType::Wpkh |  | ||||||
|             | DescriptorType::Tr |  | ||||||
|             | DescriptorType::Wsh |  | ||||||
|             | DescriptorType::WshSortedMulti => None, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn psbt_witness_script(&self) -> Option<Script> { |  | ||||||
|         match self.desc_type() { |  | ||||||
|             DescriptorType::Wsh => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::ShWsh => Some(self.explicit_script().unwrap()), |  | ||||||
|             DescriptorType::WshSortedMulti | DescriptorType::ShWshSortedMulti => { |  | ||||||
|                 Some(self.explicit_script().unwrap()) |  | ||||||
|             } |  | ||||||
|             DescriptorType::Bare |  | ||||||
|             | DescriptorType::Sh |  | ||||||
|             | DescriptorType::Pkh |  | ||||||
|             | DescriptorType::Wpkh |  | ||||||
|             | DescriptorType::ShSortedMulti |  | ||||||
|             | DescriptorType::Tr |  | ||||||
|             | DescriptorType::ShWpkh => None, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl DescriptorMeta for ExtendedDescriptor { | impl DescriptorMeta for ExtendedDescriptor { | ||||||
| @ -387,7 +379,7 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|         let mut answer = Vec::new(); |         let mut answer = Vec::new(); | ||||||
| 
 | 
 | ||||||
|         self.for_each_key(|pk| { |         self.for_each_key(|pk| { | ||||||
|             if let DescriptorPublicKey::XPub(xpub) = pk.as_key() { |             if let DescriptorPublicKey::XPub(xpub) = pk { | ||||||
|                 answer.push(xpub.clone()); |                 answer.push(xpub.clone()); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -401,7 +393,7 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|         &self, |         &self, | ||||||
|         key_origins: BTreeMap<Fingerprint, (&DerivationPath, SinglePubKey)>, |         key_origins: BTreeMap<Fingerprint, (&DerivationPath, SinglePubKey)>, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>> { |     ) -> Option<DerivedDescriptor> { | ||||||
|         // Ensure that deriving `xpub` with `path` yields `expected`
 |         // Ensure that deriving `xpub` with `path` yields `expected`
 | ||||||
|         let verify_key = |xpub: &DescriptorXKey<ExtendedPubKey>, |         let verify_key = |xpub: &DescriptorXKey<ExtendedPubKey>, | ||||||
|                           path: &DerivationPath, |                           path: &DerivationPath, | ||||||
| @ -423,7 +415,7 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
| 
 | 
 | ||||||
|         // using `for_any_key` should make this stop as soon as we return `true`
 |         // using `for_any_key` should make this stop as soon as we return `true`
 | ||||||
|         self.for_any_key(|key| { |         self.for_any_key(|key| { | ||||||
|             if let DescriptorPublicKey::XPub(xpub) = key.as_key().deref() { |             if let DescriptorPublicKey::XPub(xpub) = key { | ||||||
|                 // Check if the key matches one entry in our `key_origins`. If it does, `matches()` will
 |                 // Check if the key matches one entry in our `key_origins`. If it does, `matches()` will
 | ||||||
|                 // return the "prefix" that matched, so we remove that prefix from the full path
 |                 // return the "prefix" that matched, so we remove that prefix from the full path
 | ||||||
|                 // found in `key_origins` and save it in `derive_path`. We expect this to be a derivation
 |                 // found in `key_origins` and save it in `derive_path`. We expect this to be a derivation
 | ||||||
| @ -481,14 +473,14 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|             false |             false | ||||||
|         }); |         }); | ||||||
| 
 | 
 | ||||||
|         path_found.map(|path| self.as_derived(path, secp)) |         path_found.map(|path| self.at_derivation_index(path)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn derive_from_hd_keypaths<'s>( |     fn derive_from_hd_keypaths<'s>( | ||||||
|         &self, |         &self, | ||||||
|         hd_keypaths: &HdKeyPaths, |         hd_keypaths: &HdKeyPaths, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>> { |     ) -> Option<DerivedDescriptor> { | ||||||
|         // "Convert" an hd_keypaths map to the format required by `derive_from_psbt_key_origins`
 |         // "Convert" an hd_keypaths map to the format required by `derive_from_psbt_key_origins`
 | ||||||
|         let key_origins = hd_keypaths |         let key_origins = hd_keypaths | ||||||
|             .iter() |             .iter() | ||||||
| @ -506,7 +498,7 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|         &self, |         &self, | ||||||
|         tap_key_origins: &TapKeyOrigins, |         tap_key_origins: &TapKeyOrigins, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>> { |     ) -> Option<DerivedDescriptor> { | ||||||
|         // "Convert" a tap_key_origins map to the format required by `derive_from_psbt_key_origins`
 |         // "Convert" a tap_key_origins map to the format required by `derive_from_psbt_key_origins`
 | ||||||
|         let key_origins = tap_key_origins |         let key_origins = tap_key_origins | ||||||
|             .iter() |             .iter() | ||||||
| @ -520,19 +512,19 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|         psbt_input: &psbt::Input, |         psbt_input: &psbt::Input, | ||||||
|         utxo: Option<TxOut>, |         utxo: Option<TxOut>, | ||||||
|         secp: &'s SecpCtx, |         secp: &'s SecpCtx, | ||||||
|     ) -> Option<DerivedDescriptor<'s>> { |     ) -> Option<DerivedDescriptor> { | ||||||
|         if let Some(derived) = self.derive_from_hd_keypaths(&psbt_input.bip32_derivation, secp) { |         if let Some(derived) = self.derive_from_hd_keypaths(&psbt_input.bip32_derivation, secp) { | ||||||
|             return Some(derived); |             return Some(derived); | ||||||
|         } |         } | ||||||
|         if let Some(derived) = self.derive_from_tap_key_origins(&psbt_input.tap_key_origins, secp) { |         if let Some(derived) = self.derive_from_tap_key_origins(&psbt_input.tap_key_origins, secp) { | ||||||
|             return Some(derived); |             return Some(derived); | ||||||
|         } |         } | ||||||
|         if self.is_deriveable() { |         if self.has_wildcard() { | ||||||
|             // We can't try to bruteforce the derivation index, exit here
 |             // We can't try to bruteforce the derivation index, exit here
 | ||||||
|             return None; |             return None; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         let descriptor = self.as_derived_fixed(secp); |         let descriptor = self.at_derivation_index(0); | ||||||
|         match descriptor.desc_type() { |         match descriptor.desc_type() { | ||||||
|             // TODO: add pk() here
 |             // TODO: add pk() here
 | ||||||
|             DescriptorType::Pkh |             DescriptorType::Pkh | ||||||
| @ -566,86 +558,6 @@ impl DescriptorMeta for ExtendedDescriptor { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<'s> DerivedDescriptorMeta for DerivedDescriptor<'s> { |  | ||||||
|     fn get_hd_keypaths(&self, secp: &SecpCtx) -> HdKeyPaths { |  | ||||||
|         let mut answer = BTreeMap::new(); |  | ||||||
|         self.for_each_key(|key| { |  | ||||||
|             if let DescriptorPublicKey::XPub(xpub) = key.as_key().deref() { |  | ||||||
|                 let derived_pubkey = xpub |  | ||||||
|                     .xkey |  | ||||||
|                     .derive_pub(secp, &xpub.derivation_path) |  | ||||||
|                     .expect("Derivation can't fail"); |  | ||||||
| 
 |  | ||||||
|                 answer.insert( |  | ||||||
|                     derived_pubkey.public_key, |  | ||||||
|                     (xpub.root_fingerprint(secp), xpub.full_path(&[])), |  | ||||||
|                 ); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             true |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         answer |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn get_tap_key_origins(&self, secp: &SecpCtx) -> TapKeyOrigins { |  | ||||||
|         use miniscript::ToPublicKey; |  | ||||||
| 
 |  | ||||||
|         let mut answer = BTreeMap::new(); |  | ||||||
|         let mut insert_path = |pk: &DerivedDescriptorKey<'_>, lh| { |  | ||||||
|             let key_origin = match pk.deref() { |  | ||||||
|                 DescriptorPublicKey::XPub(xpub) => { |  | ||||||
|                     Some((xpub.root_fingerprint(secp), xpub.full_path(&[]))) |  | ||||||
|                 } |  | ||||||
|                 DescriptorPublicKey::SinglePub(_) => None, |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             // If this is the internal key, we only insert the key origin if it's not None.
 |  | ||||||
|             // For keys found in the tap tree we always insert a key origin (because the signer
 |  | ||||||
|             // looks for it to know which leaves to sign for), even though it may be None
 |  | ||||||
|             match (lh, key_origin) { |  | ||||||
|                 (None, Some(ko)) => { |  | ||||||
|                     answer |  | ||||||
|                         .entry(pk.to_x_only_pubkey()) |  | ||||||
|                         .or_insert_with(|| (vec![], ko)); |  | ||||||
|                 } |  | ||||||
|                 (Some(lh), origin) => { |  | ||||||
|                     answer |  | ||||||
|                         .entry(pk.to_x_only_pubkey()) |  | ||||||
|                         .or_insert_with(|| (vec![], origin.unwrap_or_default())) |  | ||||||
|                         .0 |  | ||||||
|                         .push(lh); |  | ||||||
|                 } |  | ||||||
|                 _ => {} |  | ||||||
|             } |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         if let Descriptor::Tr(tr) = &self { |  | ||||||
|             // Internal key first, then iterate the scripts
 |  | ||||||
|             insert_path(tr.internal_key(), None); |  | ||||||
| 
 |  | ||||||
|             for (_, ms) in tr.iter_scripts() { |  | ||||||
|                 // Assume always the same leaf version
 |  | ||||||
|                 let leaf_hash = taproot::TapLeafHash::from_script( |  | ||||||
|                     &ms.encode(), |  | ||||||
|                     taproot::LeafVersion::TapScript, |  | ||||||
|                 ); |  | ||||||
| 
 |  | ||||||
|                 for key in ms.iter_pk_pkh() { |  | ||||||
|                     let key = match key { |  | ||||||
|                         miniscript::miniscript::iter::PkPkh::PlainPubkey(pk) => pk, |  | ||||||
|                         miniscript::miniscript::iter::PkPkh::HashedPubkey(pk) => pk, |  | ||||||
|                     }; |  | ||||||
| 
 |  | ||||||
|                     insert_path(&key, Some(leaf_hash)); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         answer |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod test { | mod test { | ||||||
|     use std::str::FromStr; |     use std::str::FromStr; | ||||||
| @ -917,7 +829,7 @@ mod test { | |||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_sh_wsh_sortedmulti_redeemscript() { |     fn test_sh_wsh_sortedmulti_redeemscript() { | ||||||
|         use super::{AsDerived, DescriptorScripts}; |         use miniscript::psbt::PsbtInputExt; | ||||||
| 
 | 
 | ||||||
|         let secp = Secp256k1::new(); |         let secp = Secp256k1::new(); | ||||||
| 
 | 
 | ||||||
| @ -925,11 +837,16 @@ mod test { | |||||||
|         let (descriptor, _) = |         let (descriptor, _) = | ||||||
|             into_wallet_descriptor_checked(descriptor, &secp, Network::Testnet).unwrap(); |             into_wallet_descriptor_checked(descriptor, &secp, Network::Testnet).unwrap(); | ||||||
| 
 | 
 | ||||||
|         let descriptor = descriptor.as_derived(0, &secp); |         let descriptor = descriptor.at_derivation_index(0); | ||||||
| 
 | 
 | ||||||
|         let script = Script::from_str("5321022f533b667e2ea3b36e21961c9fe9dca340fbe0af5210173a83ae0337ab20a57621026bb53a98e810bd0ee61a0ed1164ba6c024786d76554e793e202dc6ce9c78c4ea2102d5b8a7d66a41ffdb6f4c53d61994022e886b4f45001fb158b95c9164d45f8ca3210324b75eead2c1f9c60e8adeb5e7009fec7a29afcdb30d829d82d09562fe8bae8521032d34f8932200833487bd294aa219dcbe000b9f9b3d824799541430009f0fa55121037468f8ea99b6c64788398b5ad25480cad08f4b0d65be54ce3a55fd206b5ae4722103f72d3d96663b0ea99b0aeb0d7f273cab11a8de37885f1dddc8d9112adb87169357ae").unwrap(); |         let script = Script::from_str("5321022f533b667e2ea3b36e21961c9fe9dca340fbe0af5210173a83ae0337ab20a57621026bb53a98e810bd0ee61a0ed1164ba6c024786d76554e793e202dc6ce9c78c4ea2102d5b8a7d66a41ffdb6f4c53d61994022e886b4f45001fb158b95c9164d45f8ca3210324b75eead2c1f9c60e8adeb5e7009fec7a29afcdb30d829d82d09562fe8bae8521032d34f8932200833487bd294aa219dcbe000b9f9b3d824799541430009f0fa55121037468f8ea99b6c64788398b5ad25480cad08f4b0d65be54ce3a55fd206b5ae4722103f72d3d96663b0ea99b0aeb0d7f273cab11a8de37885f1dddc8d9112adb87169357ae").unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(descriptor.psbt_redeem_script(), Some(script.to_v0_p2wsh())); |         let mut psbt_input = psbt::Input::default(); | ||||||
|         assert_eq!(descriptor.psbt_witness_script(), Some(script)); |         psbt_input | ||||||
|  |             .update_with_descriptor_unchecked(&descriptor) | ||||||
|  |             .unwrap(); | ||||||
|  | 
 | ||||||
|  |         assert_eq!(psbt_input.redeem_script, Some(script.to_v0_p2wsh())); | ||||||
|  |         assert_eq!(psbt_input.witness_script, Some(script)); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -43,14 +43,17 @@ use std::fmt; | |||||||
| use serde::ser::SerializeMap; | use serde::ser::SerializeMap; | ||||||
| use serde::{Serialize, Serializer}; | use serde::{Serialize, Serializer}; | ||||||
| 
 | 
 | ||||||
| use bitcoin::hashes::*; | use bitcoin::hashes::{hash160, ripemd160, sha256}; | ||||||
| use bitcoin::util::bip32::Fingerprint; | use bitcoin::util::bip32::Fingerprint; | ||||||
| use bitcoin::{PublicKey, XOnlyPublicKey}; | use bitcoin::{LockTime, PublicKey, Sequence, XOnlyPublicKey}; | ||||||
| 
 | 
 | ||||||
| use miniscript::descriptor::{ | use miniscript::descriptor::{ | ||||||
|     DescriptorPublicKey, DescriptorSinglePub, ShInner, SinglePubKey, SortedMultiVec, WshInner, |     DescriptorPublicKey, ShInner, SinglePub, SinglePubKey, SortedMultiVec, WshInner, | ||||||
|  | }; | ||||||
|  | use miniscript::hash256; | ||||||
|  | use miniscript::{ | ||||||
|  |     Descriptor, Miniscript, Satisfier, ScriptContext, SigType, Terminal, ToPublicKey, | ||||||
| }; | }; | ||||||
| use miniscript::{Descriptor, Miniscript, MiniscriptKey, Satisfier, ScriptContext, Terminal}; |  | ||||||
| 
 | 
 | ||||||
| #[allow(unused_imports)] | #[allow(unused_imports)] | ||||||
| use log::{debug, error, info, trace}; | use log::{debug, error, info, trace}; | ||||||
| @ -58,7 +61,7 @@ use log::{debug, error, info, trace}; | |||||||
| use crate::descriptor::ExtractPolicy; | use crate::descriptor::ExtractPolicy; | ||||||
| use crate::keys::ExtScriptContext; | use crate::keys::ExtScriptContext; | ||||||
| use crate::wallet::signer::{SignerId, SignersContainer}; | use crate::wallet::signer::{SignerId, SignersContainer}; | ||||||
| use crate::wallet::utils::{self, After, Older, SecpCtx}; | use crate::wallet::utils::{After, Older, SecpCtx}; | ||||||
| 
 | 
 | ||||||
| use super::checksum::get_checksum; | use super::checksum::get_checksum; | ||||||
| use super::error::Error; | use super::error::Error; | ||||||
| @ -81,11 +84,11 @@ pub enum PkOrF { | |||||||
| impl PkOrF { | impl PkOrF { | ||||||
|     fn from_key(k: &DescriptorPublicKey, secp: &SecpCtx) -> Self { |     fn from_key(k: &DescriptorPublicKey, secp: &SecpCtx) -> Self { | ||||||
|         match k { |         match k { | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |             DescriptorPublicKey::Single(SinglePub { | ||||||
|                 key: SinglePubKey::FullKey(pk), |                 key: SinglePubKey::FullKey(pk), | ||||||
|                 .. |                 .. | ||||||
|             }) => PkOrF::Pubkey(*pk), |             }) => PkOrF::Pubkey(*pk), | ||||||
|             DescriptorPublicKey::SinglePub(DescriptorSinglePub { |             DescriptorPublicKey::Single(SinglePub { | ||||||
|                 key: SinglePubKey::XOnly(pk), |                 key: SinglePubKey::XOnly(pk), | ||||||
|                 .. |                 .. | ||||||
|             }) => PkOrF::XOnlyPubkey(*pk), |             }) => PkOrF::XOnlyPubkey(*pk), | ||||||
| @ -111,7 +114,7 @@ pub enum SatisfiableItem { | |||||||
|     /// Double SHA256 preimage hash
 |     /// Double SHA256 preimage hash
 | ||||||
|     Hash256Preimage { |     Hash256Preimage { | ||||||
|         /// The digest value
 |         /// The digest value
 | ||||||
|         hash: sha256d::Hash, |         hash: hash256::Hash, | ||||||
|     }, |     }, | ||||||
|     /// RIPEMD160 preimage hash
 |     /// RIPEMD160 preimage hash
 | ||||||
|     Ripemd160Preimage { |     Ripemd160Preimage { | ||||||
| @ -125,13 +128,13 @@ pub enum SatisfiableItem { | |||||||
|     }, |     }, | ||||||
|     /// Absolute timeclock timestamp
 |     /// Absolute timeclock timestamp
 | ||||||
|     AbsoluteTimelock { |     AbsoluteTimelock { | ||||||
|         /// The timestamp value
 |         /// The timelock value
 | ||||||
|         value: u32, |         value: LockTime, | ||||||
|     }, |     }, | ||||||
|     /// Relative timelock locktime
 |     /// Relative timelock locktime
 | ||||||
|     RelativeTimelock { |     RelativeTimelock { | ||||||
|         /// The locktime value
 |         /// The timelock value
 | ||||||
|         value: u32, |         value: Sequence, | ||||||
|     }, |     }, | ||||||
|     /// Multi-signature public keys with threshold count
 |     /// Multi-signature public keys with threshold count
 | ||||||
|     Multisig { |     Multisig { | ||||||
| @ -438,32 +441,30 @@ pub struct Policy { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// An extra condition that must be satisfied but that is out of control of the user
 | /// An extra condition that must be satisfied but that is out of control of the user
 | ||||||
| #[derive(Hash, Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Default, Serialize)] | /// TODO: use `bitcoin::LockTime` and `bitcoin::Sequence`
 | ||||||
|  | #[derive(Hash, Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Default, Serialize)] | ||||||
| pub struct Condition { | pub struct Condition { | ||||||
|     /// Optional CheckSequenceVerify condition
 |     /// Optional CheckSequenceVerify condition
 | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|     pub csv: Option<u32>, |     pub csv: Option<Sequence>, | ||||||
|     /// Optional timelock condition
 |     /// Optional timelock condition
 | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|     pub timelock: Option<u32>, |     pub timelock: Option<LockTime>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Condition { | impl Condition { | ||||||
|     fn merge_nlocktime(a: u32, b: u32) -> Result<u32, PolicyError> { |     fn merge_nlocktime(a: LockTime, b: LockTime) -> Result<LockTime, PolicyError> { | ||||||
|         if (a < utils::BLOCKS_TIMELOCK_THRESHOLD) != (b < utils::BLOCKS_TIMELOCK_THRESHOLD) { |         if !a.is_same_unit(b) { | ||||||
|             Err(PolicyError::MixedTimelockUnits) |             Err(PolicyError::MixedTimelockUnits) | ||||||
|  |         } else if a > b { | ||||||
|  |             Ok(a) | ||||||
|         } else { |         } else { | ||||||
|             Ok(max(a, b)) |             Ok(b) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn merge_nsequence(a: u32, b: u32) -> Result<u32, PolicyError> { |     fn merge_nsequence(a: Sequence, b: Sequence) -> Result<Sequence, PolicyError> { | ||||||
|         let mask = utils::SEQUENCE_LOCKTIME_TYPE_FLAG | utils::SEQUENCE_LOCKTIME_MASK; |         if a.is_time_locked() != b.is_time_locked() { | ||||||
| 
 |  | ||||||
|         let a = a & mask; |  | ||||||
|         let b = b & mask; |  | ||||||
| 
 |  | ||||||
|         if (a < utils::SEQUENCE_LOCKTIME_TYPE_FLAG) != (b < utils::SEQUENCE_LOCKTIME_TYPE_FLAG) { |  | ||||||
|             Err(PolicyError::MixedTimelockUnits) |             Err(PolicyError::MixedTimelockUnits) | ||||||
|         } else { |         } else { | ||||||
|             Ok(max(a, b)) |             Ok(max(a, b)) | ||||||
| @ -720,15 +721,18 @@ impl From<SatisfiableItem> for Policy { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fn signer_id(key: &DescriptorPublicKey, secp: &SecpCtx) -> SignerId { | fn signer_id(key: &DescriptorPublicKey, secp: &SecpCtx) -> SignerId { | ||||||
|  |     // For consistency we always compute the key hash in "ecdsa" form (with the leading sign
 | ||||||
|  |     // prefix) even if we are in a taproot descriptor. We just want some kind of unique identifier
 | ||||||
|  |     // for a key, so it doesn't really matter how the identifier is computed.
 | ||||||
|     match key { |     match key { | ||||||
|         DescriptorPublicKey::SinglePub(DescriptorSinglePub { |         DescriptorPublicKey::Single(SinglePub { | ||||||
|             key: SinglePubKey::FullKey(pk), |             key: SinglePubKey::FullKey(pk), | ||||||
|             .. |             .. | ||||||
|         }) => pk.to_pubkeyhash().into(), |         }) => pk.to_pubkeyhash(SigType::Ecdsa).into(), | ||||||
|         DescriptorPublicKey::SinglePub(DescriptorSinglePub { |         DescriptorPublicKey::Single(SinglePub { | ||||||
|             key: SinglePubKey::XOnly(pk), |             key: SinglePubKey::XOnly(pk), | ||||||
|             .. |             .. | ||||||
|         }) => pk.to_pubkeyhash().into(), |         }) => pk.to_pubkeyhash(SigType::Ecdsa).into(), | ||||||
|         DescriptorPublicKey::XPub(xpub) => xpub.root_fingerprint(secp).into(), |         DescriptorPublicKey::XPub(xpub) => xpub.root_fingerprint(secp).into(), | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -779,7 +783,7 @@ fn generic_sig_in_psbt< | |||||||
| ) -> bool { | ) -> bool { | ||||||
|     //TODO check signature validity
 |     //TODO check signature validity
 | ||||||
|     psbt.inputs.iter().all(|input| match key { |     psbt.inputs.iter().all(|input| match key { | ||||||
|         DescriptorPublicKey::SinglePub(DescriptorSinglePub { key, .. }) => check(input, key), |         DescriptorPublicKey::Single(SinglePub { key, .. }) => check(input, key), | ||||||
|         DescriptorPublicKey::XPub(xpub) => { |         DescriptorPublicKey::XPub(xpub) => { | ||||||
|             //TODO check actual derivation matches
 |             //TODO check actual derivation matches
 | ||||||
|             match extract(input, xpub.root_fingerprint(secp)) { |             match extract(input, xpub.root_fingerprint(secp)) { | ||||||
| @ -891,10 +895,13 @@ impl<Ctx: ScriptContext + 'static> ExtractPolicy for Miniscript<DescriptorPublic | |||||||
|                 Some(Ctx::make_signature(pubkey_hash, signers, build_sat, secp)) |                 Some(Ctx::make_signature(pubkey_hash, signers, build_sat, secp)) | ||||||
|             } |             } | ||||||
|             Terminal::After(value) => { |             Terminal::After(value) => { | ||||||
|                 let mut policy: Policy = SatisfiableItem::AbsoluteTimelock { value: *value }.into(); |                 let mut policy: Policy = SatisfiableItem::AbsoluteTimelock { | ||||||
|  |                     value: value.into(), | ||||||
|  |                 } | ||||||
|  |                 .into(); | ||||||
|                 policy.contribution = Satisfaction::Complete { |                 policy.contribution = Satisfaction::Complete { | ||||||
|                     condition: Condition { |                     condition: Condition { | ||||||
|                         timelock: Some(*value), |                         timelock: Some(value.into()), | ||||||
|                         csv: None, |                         csv: None, | ||||||
|                     }, |                     }, | ||||||
|                 }; |                 }; | ||||||
| @ -905,9 +912,11 @@ impl<Ctx: ScriptContext + 'static> ExtractPolicy for Miniscript<DescriptorPublic | |||||||
|                 } = build_sat |                 } = build_sat | ||||||
|                 { |                 { | ||||||
|                     let after = After::new(Some(current_height), false); |                     let after = After::new(Some(current_height), false); | ||||||
|                     let after_sat = Satisfier::<bitcoin::PublicKey>::check_after(&after, *value); |                     let after_sat = | ||||||
|                     let inputs_sat = psbt_inputs_sat(psbt) |                         Satisfier::<bitcoin::PublicKey>::check_after(&after, value.into()); | ||||||
|                         .all(|sat| Satisfier::<bitcoin::PublicKey>::check_after(&sat, *value)); |                     let inputs_sat = psbt_inputs_sat(psbt).all(|sat| { | ||||||
|  |                         Satisfier::<bitcoin::PublicKey>::check_after(&sat, value.into()) | ||||||
|  |                     }); | ||||||
|                     if after_sat && inputs_sat { |                     if after_sat && inputs_sat { | ||||||
|                         policy.satisfaction = policy.contribution.clone(); |                         policy.satisfaction = policy.contribution.clone(); | ||||||
|                     } |                     } | ||||||
| @ -999,6 +1008,9 @@ impl<Ctx: ScriptContext + 'static> ExtractPolicy for Miniscript<DescriptorPublic | |||||||
| 
 | 
 | ||||||
|                 Policy::make_thresh(mapped, threshold)? |                 Policy::make_thresh(mapped, threshold)? | ||||||
|             } |             } | ||||||
|  | 
 | ||||||
|  |             // Unsupported
 | ||||||
|  |             Terminal::RawPkH(_) => None, | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -1124,14 +1136,12 @@ mod test { | |||||||
|     use crate::descriptor::{ExtractPolicy, IntoWalletDescriptor}; |     use crate::descriptor::{ExtractPolicy, IntoWalletDescriptor}; | ||||||
| 
 | 
 | ||||||
|     use super::*; |     use super::*; | ||||||
|     use crate::descriptor::derived::AsDerived; |  | ||||||
|     use crate::descriptor::policy::SatisfiableItem::{EcdsaSignature, Multisig, Thresh}; |     use crate::descriptor::policy::SatisfiableItem::{EcdsaSignature, Multisig, Thresh}; | ||||||
|     use crate::keys::{DescriptorKey, IntoDescriptorKey}; |     use crate::keys::{DescriptorKey, IntoDescriptorKey}; | ||||||
|     use crate::wallet::signer::SignersContainer; |     use crate::wallet::signer::SignersContainer; | ||||||
|     use bitcoin::secp256k1::Secp256k1; |     use bitcoin::secp256k1::Secp256k1; | ||||||
|     use bitcoin::util::bip32; |     use bitcoin::util::bip32; | ||||||
|     use bitcoin::Network; |     use bitcoin::Network; | ||||||
|     use miniscript::DescriptorTrait; |  | ||||||
|     use std::str::FromStr; |     use std::str::FromStr; | ||||||
|     use std::sync::Arc; |     use std::sync::Arc; | ||||||
| 
 | 
 | ||||||
| @ -1329,9 +1339,8 @@ mod test { | |||||||
|         let (wallet_desc, keymap) = desc |         let (wallet_desc, keymap) = desc | ||||||
|             .into_wallet_descriptor(&secp, Network::Testnet) |             .into_wallet_descriptor(&secp, Network::Testnet) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         let single_key = wallet_desc.derive(0); |  | ||||||
|         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); |         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); | ||||||
|         let policy = single_key |         let policy = wallet_desc | ||||||
|             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) |             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) | ||||||
|             .unwrap() |             .unwrap() | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
| @ -1343,16 +1352,15 @@ mod test { | |||||||
|         let (wallet_desc, keymap) = desc |         let (wallet_desc, keymap) = desc | ||||||
|             .into_wallet_descriptor(&secp, Network::Testnet) |             .into_wallet_descriptor(&secp, Network::Testnet) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         let single_key = wallet_desc.derive(0); |  | ||||||
|         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); |         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); | ||||||
|         let policy = single_key |         let policy = wallet_desc | ||||||
|             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) |             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) | ||||||
|             .unwrap() |             .unwrap() | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert!(matches!(&policy.item, EcdsaSignature(PkOrF::Fingerprint(f)) if f == &fingerprint)); |         assert!(matches!(policy.item, EcdsaSignature(PkOrF::Fingerprint(f)) if f == fingerprint)); | ||||||
|         assert!( |         assert!( | ||||||
|             matches!(&policy.contribution, Satisfaction::Complete {condition} if condition.csv == None && condition.timelock == None) |             matches!(policy.contribution, Satisfaction::Complete {condition} if condition.csv == None && condition.timelock == None) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -1368,21 +1376,20 @@ mod test { | |||||||
|         let (wallet_desc, keymap) = desc |         let (wallet_desc, keymap) = desc | ||||||
|             .into_wallet_descriptor(&secp, Network::Testnet) |             .into_wallet_descriptor(&secp, Network::Testnet) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         let single_key = wallet_desc.derive(0); |  | ||||||
|         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); |         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); | ||||||
|         let policy = single_key |         let policy = wallet_desc | ||||||
|             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) |             .extract_policy(&signers_container, BuildSatisfaction::None, &secp) | ||||||
|             .unwrap() |             .unwrap() | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert!( |         assert!( | ||||||
|             matches!(&policy.item, Multisig { keys, threshold } if threshold == &1 |             matches!(policy.item, Multisig { keys, threshold } if threshold == 1 | ||||||
|             && keys[0] == PkOrF::Fingerprint(fingerprint0) |             && keys[0] == PkOrF::Fingerprint(fingerprint0) | ||||||
|             && keys[1] == PkOrF::Fingerprint(fingerprint1)) |             && keys[1] == PkOrF::Fingerprint(fingerprint1)) | ||||||
|         ); |         ); | ||||||
|         assert!( |         assert!( | ||||||
|             matches!(&policy.contribution, Satisfaction::PartialComplete { n, m, items, conditions, .. } if n == &2 |             matches!(policy.contribution, Satisfaction::PartialComplete { n, m, items, conditions, .. } if n == 2 | ||||||
|              && m == &1 |              && m == 1 | ||||||
|              && items.len() == 2 |              && items.len() == 2 | ||||||
|              && conditions.contains_key(&vec![0]) |              && conditions.contains_key(&vec![0]) | ||||||
|              && conditions.contains_key(&vec![1]) |              && conditions.contains_key(&vec![1]) | ||||||
| @ -1427,8 +1434,8 @@ mod test { | |||||||
|              && m == &2 |              && m == &2 | ||||||
|              && items.len() == 3 |              && items.len() == 3 | ||||||
|              && conditions.get(&vec![0,1]).unwrap().iter().next().unwrap().csv.is_none() |              && conditions.get(&vec![0,1]).unwrap().iter().next().unwrap().csv.is_none() | ||||||
|              && conditions.get(&vec![0,2]).unwrap().iter().next().unwrap().csv == Some(sequence) |              && conditions.get(&vec![0,2]).unwrap().iter().next().unwrap().csv == Some(Sequence(sequence)) | ||||||
|              && conditions.get(&vec![1,2]).unwrap().iter().next().unwrap().csv == Some(sequence) |              && conditions.get(&vec![1,2]).unwrap().iter().next().unwrap().csv == Some(Sequence(sequence)) | ||||||
|             ) |             ) | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| @ -1574,7 +1581,7 @@ mod test { | |||||||
|             .unwrap(); |             .unwrap(); | ||||||
| 
 | 
 | ||||||
|         let addr = wallet_desc |         let addr = wallet_desc | ||||||
|             .as_derived(0, &secp) |             .at_derivation_index(0) | ||||||
|             .address(Network::Testnet) |             .address(Network::Testnet) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         assert_eq!( |         assert_eq!( | ||||||
| @ -1646,7 +1653,7 @@ mod test { | |||||||
|         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); |         let signers_container = Arc::new(SignersContainer::build(keymap, &wallet_desc, &secp)); | ||||||
| 
 | 
 | ||||||
|         let addr = wallet_desc |         let addr = wallet_desc | ||||||
|             .as_derived(0, &secp) |             .at_derivation_index(0) | ||||||
|             .address(Network::Testnet) |             .address(Network::Testnet) | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|         assert_eq!( |         assert_eq!( | ||||||
|  | |||||||
| @ -468,12 +468,10 @@ mod test { | |||||||
|     use std::str::FromStr; |     use std::str::FromStr; | ||||||
| 
 | 
 | ||||||
|     use super::*; |     use super::*; | ||||||
|     use crate::descriptor::derived::AsDerived; |  | ||||||
|     use crate::descriptor::{DescriptorError, DescriptorMeta}; |     use crate::descriptor::{DescriptorError, DescriptorMeta}; | ||||||
|     use crate::keys::ValidNetworks; |     use crate::keys::ValidNetworks; | ||||||
|     use bitcoin::network::constants::Network::Regtest; |     use bitcoin::network::constants::Network::Regtest; | ||||||
|     use bitcoin::secp256k1::Secp256k1; |     use miniscript::descriptor::{DescriptorPublicKey, KeyMap}; | ||||||
|     use miniscript::descriptor::{DescriptorPublicKey, DescriptorTrait, KeyMap}; |  | ||||||
|     use miniscript::Descriptor; |     use miniscript::Descriptor; | ||||||
| 
 | 
 | ||||||
|     // BIP44 `pkh(key/44'/{0,1}'/0'/{0,1}/*)`
 |     // BIP44 `pkh(key/44'/{0,1}'/0'/{0,1}/*)`
 | ||||||
| @ -517,17 +515,15 @@ mod test { | |||||||
|         is_fixed: bool, |         is_fixed: bool, | ||||||
|         expected: &[&str], |         expected: &[&str], | ||||||
|     ) { |     ) { | ||||||
|         let secp = Secp256k1::new(); |  | ||||||
| 
 |  | ||||||
|         let (desc, _key_map, _networks) = desc.unwrap(); |         let (desc, _key_map, _networks) = desc.unwrap(); | ||||||
|         assert_eq!(desc.is_witness(), is_witness); |         assert_eq!(desc.is_witness(), is_witness); | ||||||
|         assert_eq!(!desc.is_deriveable(), is_fixed); |         assert_eq!(!desc.has_wildcard(), is_fixed); | ||||||
|         for i in 0..expected.len() { |         for i in 0..expected.len() { | ||||||
|             let index = i as u32; |             let index = i as u32; | ||||||
|             let child_desc = if !desc.is_deriveable() { |             let child_desc = if !desc.has_wildcard() { | ||||||
|                 desc.as_derived_fixed(&secp) |                 desc.at_derivation_index(0) | ||||||
|             } else { |             } else { | ||||||
|                 desc.as_derived(index, &secp) |                 desc.at_derivation_index(index) | ||||||
|             }; |             }; | ||||||
|             let address = child_desc.address(Regtest).unwrap(); |             let address = child_desc.address(Regtest).unwrap(); | ||||||
|             assert_eq!(address.to_string(), *expected.get(i).unwrap()); |             assert_eq!(address.to_string(), *expected.get(i).unwrap()); | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								src/error.rs
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								src/error.rs
									
									
									
									
									
								
							| @ -12,7 +12,7 @@ | |||||||
| use std::fmt; | use std::fmt; | ||||||
| 
 | 
 | ||||||
| use crate::bitcoin::Network; | use crate::bitcoin::Network; | ||||||
| use crate::{descriptor, wallet, wallet::address_validator}; | use crate::{descriptor, wallet}; | ||||||
| use bitcoin::{OutPoint, Txid}; | use bitcoin::{OutPoint, Txid}; | ||||||
| 
 | 
 | ||||||
| /// Errors that can be thrown by the [`Wallet`](crate::wallet::Wallet)
 | /// Errors that can be thrown by the [`Wallet`](crate::wallet::Wallet)
 | ||||||
| @ -99,12 +99,12 @@ pub enum Error { | |||||||
| 
 | 
 | ||||||
|     /// Error related to the parsing and usage of descriptors
 |     /// Error related to the parsing and usage of descriptors
 | ||||||
|     Descriptor(crate::descriptor::error::Error), |     Descriptor(crate::descriptor::error::Error), | ||||||
|     /// Error that can be returned to fail the validation of an address
 |  | ||||||
|     AddressValidator(crate::wallet::address_validator::AddressValidatorError), |  | ||||||
|     /// Encoding error
 |     /// Encoding error
 | ||||||
|     Encode(bitcoin::consensus::encode::Error), |     Encode(bitcoin::consensus::encode::Error), | ||||||
|     /// Miniscript error
 |     /// Miniscript error
 | ||||||
|     Miniscript(miniscript::Error), |     Miniscript(miniscript::Error), | ||||||
|  |     /// Miniscript PSBT error
 | ||||||
|  |     MiniscriptPsbt(MiniscriptPsbtError), | ||||||
|     /// BIP32 error
 |     /// BIP32 error
 | ||||||
|     Bip32(bitcoin::util::bip32::Error), |     Bip32(bitcoin::util::bip32::Error), | ||||||
|     /// An ECDSA error
 |     /// An ECDSA error
 | ||||||
| @ -149,6 +149,14 @@ pub enum Error { | |||||||
|     Rusqlite(rusqlite::Error), |     Rusqlite(rusqlite::Error), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /// Errors returned by miniscript when updating inconsistent PSBTs
 | ||||||
|  | #[derive(Debug, Clone)] | ||||||
|  | pub enum MiniscriptPsbtError { | ||||||
|  |     Conversion(miniscript::descriptor::ConversionError), | ||||||
|  |     UtxoUpdate(miniscript::psbt::UtxoUpdateError), | ||||||
|  |     OutputUpdate(miniscript::psbt::OutputUpdateError), | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// Represents the last failed [`crate::blockchain::WalletSync`] sync attempt in which we were short
 | /// Represents the last failed [`crate::blockchain::WalletSync`] sync attempt in which we were short
 | ||||||
| /// on cached `scriptPubKey`s.
 | /// on cached `scriptPubKey`s.
 | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| @ -181,7 +189,6 @@ macro_rules! impl_error { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl_error!(descriptor::error::Error, Descriptor); | impl_error!(descriptor::error::Error, Descriptor); | ||||||
| impl_error!(address_validator::AddressValidatorError, AddressValidator); |  | ||||||
| impl_error!(descriptor::policy::PolicyError, InvalidPolicyPathError); | impl_error!(descriptor::policy::PolicyError, InvalidPolicyPathError); | ||||||
| impl_error!(wallet::signer::SignerError, Signer); | impl_error!(wallet::signer::SignerError, Signer); | ||||||
| 
 | 
 | ||||||
| @ -198,6 +205,7 @@ impl From<crate::keys::KeyError> for Error { | |||||||
| 
 | 
 | ||||||
| impl_error!(bitcoin::consensus::encode::Error, Encode); | impl_error!(bitcoin::consensus::encode::Error, Encode); | ||||||
| impl_error!(miniscript::Error, Miniscript); | impl_error!(miniscript::Error, Miniscript); | ||||||
|  | impl_error!(MiniscriptPsbtError, MiniscriptPsbt); | ||||||
| impl_error!(bitcoin::util::bip32::Error, Bip32); | impl_error!(bitcoin::util::bip32::Error, Bip32); | ||||||
| impl_error!(bitcoin::secp256k1::Error, Secp256k1); | impl_error!(bitcoin::secp256k1::Error, Secp256k1); | ||||||
| impl_error!(serde_json::Error, Json); | impl_error!(serde_json::Error, Json); | ||||||
|  | |||||||
| @ -24,8 +24,8 @@ use bitcoin::{Network, PrivateKey, PublicKey, XOnlyPublicKey}; | |||||||
| 
 | 
 | ||||||
| use miniscript::descriptor::{Descriptor, DescriptorXKey, Wildcard}; | use miniscript::descriptor::{Descriptor, DescriptorXKey, Wildcard}; | ||||||
| pub use miniscript::descriptor::{ | pub use miniscript::descriptor::{ | ||||||
|     DescriptorPublicKey, DescriptorSecretKey, DescriptorSinglePriv, DescriptorSinglePub, KeyMap, |     DescriptorPublicKey, DescriptorSecretKey, KeyMap, SinglePriv, SinglePub, SinglePubKey, | ||||||
|     SinglePubKey, SortedMultiVec, |     SortedMultiVec, | ||||||
| }; | }; | ||||||
| pub use miniscript::ScriptContext; | pub use miniscript::ScriptContext; | ||||||
| use miniscript::{Miniscript, Terminal}; | use miniscript::{Miniscript, Terminal}; | ||||||
| @ -110,7 +110,7 @@ impl<Ctx: ScriptContext> DescriptorKey<Ctx> { | |||||||
|                 let mut key_map = KeyMap::with_capacity(1); |                 let mut key_map = KeyMap::with_capacity(1); | ||||||
| 
 | 
 | ||||||
|                 let public = secret |                 let public = secret | ||||||
|                     .as_public(secp) |                     .to_public(secp) | ||||||
|                     .map_err(|e| miniscript::Error::Unexpected(e.to_string()))?; |                     .map_err(|e| miniscript::Error::Unexpected(e.to_string()))?; | ||||||
|                 key_map.insert(public.clone(), secret); |                 key_map.insert(public.clone(), secret); | ||||||
| 
 | 
 | ||||||
| @ -224,8 +224,8 @@ impl<Ctx: ScriptContext + 'static> ExtScriptContext for Ctx { | |||||||
| /// use bdk::bitcoin::PublicKey;
 | /// use bdk::bitcoin::PublicKey;
 | ||||||
| ///
 | ///
 | ||||||
| /// use bdk::keys::{
 | /// use bdk::keys::{
 | ||||||
| ///     mainnet_network, DescriptorKey, DescriptorPublicKey, DescriptorSinglePub,
 | ///     mainnet_network, DescriptorKey, DescriptorPublicKey, IntoDescriptorKey, KeyError,
 | ||||||
| ///     IntoDescriptorKey, KeyError, ScriptContext, SinglePubKey,
 | ///     ScriptContext, SinglePub, SinglePubKey,
 | ||||||
| /// };
 | /// };
 | ||||||
| ///
 | ///
 | ||||||
| /// pub struct MyKeyType {
 | /// pub struct MyKeyType {
 | ||||||
| @ -235,7 +235,7 @@ impl<Ctx: ScriptContext + 'static> ExtScriptContext for Ctx { | |||||||
| /// impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for MyKeyType {
 | /// impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for MyKeyType {
 | ||||||
| ///     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> {
 | ///     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> {
 | ||||||
| ///         Ok(DescriptorKey::from_public(
 | ///         Ok(DescriptorKey::from_public(
 | ||||||
| ///             DescriptorPublicKey::SinglePub(DescriptorSinglePub {
 | ///             DescriptorPublicKey::Single(SinglePub {
 | ||||||
| ///                 origin: None,
 | ///                 origin: None,
 | ||||||
| ///                 key: SinglePubKey::FullKey(self.pubkey),
 | ///                 key: SinglePubKey::FullKey(self.pubkey),
 | ||||||
| ///             }),
 | ///             }),
 | ||||||
| @ -842,7 +842,7 @@ impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorKey<Ctx> { | |||||||
| impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorPublicKey { | impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorPublicKey { | ||||||
|     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { |     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { | ||||||
|         let networks = match self { |         let networks = match self { | ||||||
|             DescriptorPublicKey::SinglePub(_) => any_network(), |             DescriptorPublicKey::Single(_) => any_network(), | ||||||
|             DescriptorPublicKey::XPub(DescriptorXKey { xkey, .. }) |             DescriptorPublicKey::XPub(DescriptorXKey { xkey, .. }) | ||||||
|                 if xkey.network == Network::Bitcoin => |                 if xkey.network == Network::Bitcoin => | ||||||
|             { |             { | ||||||
| @ -857,7 +857,7 @@ impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorPublicKey { | |||||||
| 
 | 
 | ||||||
| impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for PublicKey { | impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for PublicKey { | ||||||
|     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { |     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { | ||||||
|         DescriptorPublicKey::SinglePub(DescriptorSinglePub { |         DescriptorPublicKey::Single(SinglePub { | ||||||
|             key: SinglePubKey::FullKey(self), |             key: SinglePubKey::FullKey(self), | ||||||
|             origin: None, |             origin: None, | ||||||
|         }) |         }) | ||||||
| @ -867,7 +867,7 @@ impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for PublicKey { | |||||||
| 
 | 
 | ||||||
| impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for XOnlyPublicKey { | impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for XOnlyPublicKey { | ||||||
|     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { |     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { | ||||||
|         DescriptorPublicKey::SinglePub(DescriptorSinglePub { |         DescriptorPublicKey::Single(SinglePub { | ||||||
|             key: SinglePubKey::XOnly(self), |             key: SinglePubKey::XOnly(self), | ||||||
|             origin: None, |             origin: None, | ||||||
|         }) |         }) | ||||||
| @ -878,7 +878,7 @@ impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for XOnlyPublicKey { | |||||||
| impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorSecretKey { | impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for DescriptorSecretKey { | ||||||
|     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { |     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { | ||||||
|         let networks = match &self { |         let networks = match &self { | ||||||
|             DescriptorSecretKey::SinglePriv(sk) if sk.key.network == Network::Bitcoin => { |             DescriptorSecretKey::Single(sk) if sk.key.network == Network::Bitcoin => { | ||||||
|                 mainnet_network() |                 mainnet_network() | ||||||
|             } |             } | ||||||
|             DescriptorSecretKey::XPrv(DescriptorXKey { xkey, .. }) |             DescriptorSecretKey::XPrv(DescriptorXKey { xkey, .. }) | ||||||
| @ -903,7 +903,7 @@ impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for &'_ str { | |||||||
| 
 | 
 | ||||||
| impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for PrivateKey { | impl<Ctx: ScriptContext> IntoDescriptorKey<Ctx> for PrivateKey { | ||||||
|     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { |     fn into_descriptor_key(self) -> Result<DescriptorKey<Ctx>, KeyError> { | ||||||
|         DescriptorSecretKey::SinglePriv(DescriptorSinglePriv { |         DescriptorSecretKey::Single(SinglePriv { | ||||||
|             key: self, |             key: self, | ||||||
|             origin: None, |             origin: None, | ||||||
|         }) |         }) | ||||||
|  | |||||||
| @ -273,7 +273,6 @@ pub use descriptor::template; | |||||||
| pub use descriptor::HdKeyPaths; | pub use descriptor::HdKeyPaths; | ||||||
| pub use error::Error; | pub use error::Error; | ||||||
| pub use types::*; | pub use types::*; | ||||||
| pub use wallet::address_validator; |  | ||||||
| pub use wallet::signer; | pub use wallet::signer; | ||||||
| pub use wallet::signer::SignOptions; | pub use wallet::signer::SignOptions; | ||||||
| pub use wallet::tx_builder::TxBuilder; | pub use wallet::tx_builder::TxBuilder; | ||||||
|  | |||||||
| @ -1,8 +1,18 @@ | |||||||
|  | // Bitcoin Dev Kit
 | ||||||
|  | //
 | ||||||
|  | // Copyright (c) 2020-2021 Bitcoin Dev Kit Developers
 | ||||||
|  | //
 | ||||||
|  | // This file is licensed under the Apache License, Version 2.0 <LICENSE-APACHE
 | ||||||
|  | // or http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
 | ||||||
|  | // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your option.
 | ||||||
|  | // You may not use this file except in accordance with one or both of these
 | ||||||
|  | // licenses.
 | ||||||
|  | 
 | ||||||
| use crate::testutils::TestIncomingTx; | use crate::testutils::TestIncomingTx; | ||||||
| use bitcoin::consensus::encode::{deserialize, serialize}; | use bitcoin::consensus::encode::{deserialize, serialize}; | ||||||
| use bitcoin::hashes::hex::{FromHex, ToHex}; | use bitcoin::hashes::hex::{FromHex, ToHex}; | ||||||
| use bitcoin::hashes::sha256d; | use bitcoin::hashes::sha256d; | ||||||
| use bitcoin::{Address, Amount, Script, Transaction, Txid, Witness}; | use bitcoin::{Address, Amount, PackedLockTime, Script, Sequence, Transaction, Txid, Witness}; | ||||||
| pub use bitcoincore_rpc::bitcoincore_rpc_json::AddressType; | pub use bitcoincore_rpc::bitcoincore_rpc_json::AddressType; | ||||||
| pub use bitcoincore_rpc::{Auth, Client as RpcClient, RpcApi}; | pub use bitcoincore_rpc::{Auth, Client as RpcClient, RpcApi}; | ||||||
| use core::str::FromStr; | use core::str::FromStr; | ||||||
| @ -110,7 +120,7 @@ impl TestClient { | |||||||
|         if let Some(true) = meta_tx.replaceable { |         if let Some(true) = meta_tx.replaceable { | ||||||
|             // for some reason core doesn't set this field right
 |             // for some reason core doesn't set this field right
 | ||||||
|             for input in &mut tx.input { |             for input in &mut tx.input { | ||||||
|                 input.sequence = 0xFFFFFFFD; |                 input.sequence = Sequence(0xFFFFFFFD); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @ -164,6 +174,7 @@ impl TestClient { | |||||||
|         use bitcoin::blockdata::script::Builder; |         use bitcoin::blockdata::script::Builder; | ||||||
|         use bitcoin::blockdata::transaction::{OutPoint, TxIn, TxOut}; |         use bitcoin::blockdata::transaction::{OutPoint, TxIn, TxOut}; | ||||||
|         use bitcoin::hash_types::{BlockHash, TxMerkleNode}; |         use bitcoin::hash_types::{BlockHash, TxMerkleNode}; | ||||||
|  |         use bitcoin::hashes::Hash; | ||||||
| 
 | 
 | ||||||
|         let block_template: serde_json::Value = self |         let block_template: serde_json::Value = self | ||||||
|             .call("getblocktemplate", &[json!({"rules": ["segwit"]})]) |             .call("getblocktemplate", &[json!({"rules": ["segwit"]})]) | ||||||
| @ -176,7 +187,7 @@ impl TestClient { | |||||||
|                 block_template["previousblockhash"].as_str().unwrap(), |                 block_template["previousblockhash"].as_str().unwrap(), | ||||||
|             ) |             ) | ||||||
|             .unwrap(), |             .unwrap(), | ||||||
|             merkle_root: TxMerkleNode::default(), |             merkle_root: TxMerkleNode::all_zeros(), | ||||||
|             time: block_template["curtime"].as_u64().unwrap() as u32, |             time: block_template["curtime"].as_u64().unwrap() as u32, | ||||||
|             bits: u32::from_str_radix(block_template["bits"].as_str().unwrap(), 16).unwrap(), |             bits: u32::from_str_radix(block_template["bits"].as_str().unwrap(), 16).unwrap(), | ||||||
|             nonce: 0, |             nonce: 0, | ||||||
| @ -184,15 +195,15 @@ impl TestClient { | |||||||
|         debug!("header: {:#?}", header); |         debug!("header: {:#?}", header); | ||||||
| 
 | 
 | ||||||
|         let height = block_template["height"].as_u64().unwrap() as i64; |         let height = block_template["height"].as_u64().unwrap() as i64; | ||||||
|         let witness_reserved_value: Vec<u8> = sha256d::Hash::default().as_ref().into(); |         let witness_reserved_value: Vec<u8> = sha256d::Hash::all_zeros().as_ref().into(); | ||||||
|         // burn block subsidy and fees, not a big deal
 |         // burn block subsidy and fees, not a big deal
 | ||||||
|         let mut coinbase_tx = Transaction { |         let mut coinbase_tx = Transaction { | ||||||
|             version: 1, |             version: 1, | ||||||
|             lock_time: 0, |             lock_time: PackedLockTime(0), | ||||||
|             input: vec![TxIn { |             input: vec![TxIn { | ||||||
|                 previous_output: OutPoint::null(), |                 previous_output: OutPoint::null(), | ||||||
|                 script_sig: Builder::new().push_int(height).into_script(), |                 script_sig: Builder::new().push_int(height).into_script(), | ||||||
|                 sequence: 0xFFFFFFFF, |                 sequence: Sequence(0xFFFFFFFF), | ||||||
|                 witness: Witness::from_vec(vec![witness_reserved_value]), |                 witness: Witness::from_vec(vec![witness_reserved_value]), | ||||||
|             }], |             }], | ||||||
|             output: vec![], |             output: vec![], | ||||||
| @ -1184,7 +1195,7 @@ macro_rules! bdk_blockchain_tests { | |||||||
|                 // 5. Verify 25_000 sats are received by test bitcoind node taproot wallet
 |                 // 5. Verify 25_000 sats are received by test bitcoind node taproot wallet
 | ||||||
| 
 | 
 | ||||||
|                 let taproot_balance = taproot_wallet_client.get_balance(None, None).unwrap(); |                 let taproot_balance = taproot_wallet_client.get_balance(None, None).unwrap(); | ||||||
|                 assert_eq!(taproot_balance.as_sat(), 25_000, "node has incorrect taproot wallet balance"); |                 assert_eq!(taproot_balance.to_sat(), 25_000, "node has incorrect taproot wallet balance"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             #[test] |             #[test] | ||||||
|  | |||||||
| @ -101,25 +101,21 @@ impl TestIncomingTx { | |||||||
| macro_rules! testutils { | macro_rules! testutils { | ||||||
|     ( @external $descriptors:expr, $child:expr ) => ({ |     ( @external $descriptors:expr, $child:expr ) => ({ | ||||||
|         use $crate::bitcoin::secp256k1::Secp256k1; |         use $crate::bitcoin::secp256k1::Secp256k1; | ||||||
|         use $crate::miniscript::descriptor::{Descriptor, DescriptorPublicKey, DescriptorTrait}; |         use $crate::miniscript::descriptor::{Descriptor, DescriptorPublicKey}; | ||||||
| 
 |  | ||||||
|         use $crate::descriptor::AsDerived; |  | ||||||
| 
 | 
 | ||||||
|         let secp = Secp256k1::new(); |         let secp = Secp256k1::new(); | ||||||
| 
 | 
 | ||||||
|         let parsed = Descriptor::<DescriptorPublicKey>::parse_descriptor(&secp, &$descriptors.0).expect("Failed to parse descriptor in `testutils!(@external)`").0; |         let parsed = Descriptor::<DescriptorPublicKey>::parse_descriptor(&secp, &$descriptors.0).expect("Failed to parse descriptor in `testutils!(@external)`").0; | ||||||
|         parsed.as_derived($child, &secp).address(bitcoin::Network::Regtest).expect("No address form") |         parsed.at_derivation_index($child).address(bitcoin::Network::Regtest).expect("No address form") | ||||||
|     }); |     }); | ||||||
|     ( @internal $descriptors:expr, $child:expr ) => ({ |     ( @internal $descriptors:expr, $child:expr ) => ({ | ||||||
|         use $crate::bitcoin::secp256k1::Secp256k1; |         use $crate::bitcoin::secp256k1::Secp256k1; | ||||||
|         use $crate::miniscript::descriptor::{Descriptor, DescriptorPublicKey, DescriptorTrait}; |         use $crate::miniscript::descriptor::{Descriptor, DescriptorPublicKey}; | ||||||
| 
 |  | ||||||
|         use $crate::descriptor::AsDerived; |  | ||||||
| 
 | 
 | ||||||
|         let secp = Secp256k1::new(); |         let secp = Secp256k1::new(); | ||||||
| 
 | 
 | ||||||
|         let parsed = Descriptor::<DescriptorPublicKey>::parse_descriptor(&secp, &$descriptors.1.expect("Missing internal descriptor")).expect("Failed to parse descriptor in `testutils!(@internal)`").0; |         let parsed = Descriptor::<DescriptorPublicKey>::parse_descriptor(&secp, &$descriptors.1.expect("Missing internal descriptor")).expect("Failed to parse descriptor in `testutils!(@internal)`").0; | ||||||
|         parsed.as_derived($child, &secp).address($crate::bitcoin::Network::Regtest).expect("No address form") |         parsed.at_derivation_index($child).address($crate::bitcoin::Network::Regtest).expect("No address form") | ||||||
|     }); |     }); | ||||||
|     ( @e $descriptors:expr, $child:expr ) => ({ testutils!(@external $descriptors, $child) }); |     ( @e $descriptors:expr, $child:expr ) => ({ testutils!(@external $descriptors, $child) }); | ||||||
|     ( @i $descriptors:expr, $child:expr ) => ({ testutils!(@internal $descriptors, $child) }); |     ( @i $descriptors:expr, $child:expr ) => ({ testutils!(@internal $descriptors, $child) }); | ||||||
| @ -186,49 +182,50 @@ macro_rules! testutils { | |||||||
|     ( @descriptors ( $external_descriptor:expr ) $( ( $internal_descriptor:expr ) )? $( ( @keys $( $keys:tt )* ) )* ) => ({ |     ( @descriptors ( $external_descriptor:expr ) $( ( $internal_descriptor:expr ) )? $( ( @keys $( $keys:tt )* ) )* ) => ({ | ||||||
|         use std::str::FromStr; |         use std::str::FromStr; | ||||||
|         use std::collections::HashMap; |         use std::collections::HashMap; | ||||||
|  |         use std::convert::Infallible; | ||||||
|  | 
 | ||||||
|         use $crate::miniscript::descriptor::Descriptor; |         use $crate::miniscript::descriptor::Descriptor; | ||||||
|         use $crate::miniscript::TranslatePk; |         use $crate::miniscript::TranslatePk; | ||||||
| 
 | 
 | ||||||
|  |         struct Translator { | ||||||
|  |             keys: HashMap<&'static str, (String, Option<String>, Option<String>)>, | ||||||
|  |             is_internal: bool, | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         impl $crate::miniscript::Translator<String, String, Infallible> for Translator { | ||||||
|  |             fn pk(&mut self, pk: &String) -> Result<String, Infallible> { | ||||||
|  |                 match self.keys.get(pk.as_str()) { | ||||||
|  |                     Some((key, ext_path, int_path)) => { | ||||||
|  |                         let path = if self.is_internal { int_path } else { ext_path }; | ||||||
|  |                         Ok(format!("{}{}", key, path.clone().unwrap_or_default())) | ||||||
|  |                     } | ||||||
|  |                     None => Ok(pk.clone()), | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             fn sha256(&mut self, sha256: &String) -> Result<String, Infallible> { Ok(sha256.clone()) } | ||||||
|  |             fn hash256(&mut self, hash256: &String) -> Result<String, Infallible> { Ok(hash256.clone()) } | ||||||
|  |             fn ripemd160(&mut self, ripemd160: &String) -> Result<String, Infallible> { Ok(ripemd160.clone()) } | ||||||
|  |             fn hash160(&mut self, hash160: &String) -> Result<String, Infallible> { Ok(hash160.clone()) } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         #[allow(unused_assignments, unused_mut)] |         #[allow(unused_assignments, unused_mut)] | ||||||
|         let mut keys: HashMap<&'static str, (String, Option<String>, Option<String>)> = HashMap::new(); |         let mut keys = HashMap::new(); | ||||||
|         $( |         $( | ||||||
|             keys = testutils!{ @keys $( $keys )* }; |             keys = testutils!{ @keys $( $keys )* }; | ||||||
|         )* |         )* | ||||||
| 
 | 
 | ||||||
|         let external: Descriptor<String> = FromStr::from_str($external_descriptor).unwrap(); |         let mut translator = Translator { keys, is_internal: false }; | ||||||
|         let external: Descriptor<String> = external.translate_pk_infallible::<_, _>(|k| { |  | ||||||
|             if let Some((key, ext_path, _)) = keys.get(&k.as_str()) { |  | ||||||
|                 format!("{}{}", key, ext_path.as_ref().unwrap_or(&"".into())) |  | ||||||
|             } else { |  | ||||||
|                 k.clone() |  | ||||||
|             } |  | ||||||
|         }, |kh| { |  | ||||||
|             if let Some((key, ext_path, _)) = keys.get(&kh.as_str()) { |  | ||||||
|                 format!("{}{}", key, ext_path.as_ref().unwrap_or(&"".into())) |  | ||||||
|             } else { |  | ||||||
|                 kh.clone() |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|         }); |         let external: Descriptor<String> = FromStr::from_str($external_descriptor).unwrap(); | ||||||
|  |         let external = external.translate_pk(&mut translator).expect("Infallible conversion"); | ||||||
|         let external = external.to_string(); |         let external = external.to_string(); | ||||||
| 
 | 
 | ||||||
|         let internal = None::<String>$(.or({ |         translator.is_internal = true; | ||||||
|             let string_internal: Descriptor<String> = FromStr::from_str($internal_descriptor).unwrap(); |  | ||||||
| 
 | 
 | ||||||
|             let string_internal: Descriptor<String> = string_internal.translate_pk_infallible::<_, _>(|k| { |         let internal = None::<String>$(.or({ | ||||||
|                 if let Some((key, _, int_path)) = keys.get(&k.as_str()) { |             let internal: Descriptor<String> = FromStr::from_str($internal_descriptor).unwrap(); | ||||||
|                     format!("{}{}", key, int_path.as_ref().unwrap_or(&"".into())) |             let internal = internal.translate_pk(&mut translator).expect("Infallible conversion"); | ||||||
|                 } else { |             Some(internal.to_string()) | ||||||
|                     k.clone() |  | ||||||
|                 } |  | ||||||
|             }, |kh| { |  | ||||||
|                 if let Some((key, _, int_path)) = keys.get(&kh.as_str()) { |  | ||||||
|                     format!("{}{}", key, int_path.as_ref().unwrap_or(&"".into())) |  | ||||||
|                 } else { |  | ||||||
|                     kh.clone() |  | ||||||
|                 } |  | ||||||
|             }); |  | ||||||
|             Some(string_internal.to_string()) |  | ||||||
|         }))?; |         }))?; | ||||||
| 
 | 
 | ||||||
|         (external, internal) |         (external, internal) | ||||||
|  | |||||||
| @ -166,7 +166,7 @@ pub struct LocalUtxo { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// A [`Utxo`] with its `satisfaction_weight`.
 | /// A [`Utxo`] with its `satisfaction_weight`.
 | ||||||
| #[derive(Debug, Clone, PartialEq)] | #[derive(Debug, Clone, PartialEq, Eq)] | ||||||
| pub struct WeightedUtxo { | pub struct WeightedUtxo { | ||||||
|     /// The weight of the witness data and `scriptSig` expressed in [weight units]. This is used to
 |     /// The weight of the witness data and `scriptSig` expressed in [weight units]. This is used to
 | ||||||
|     /// properly maintain the feerate when adding this input to a transaction during coin selection.
 |     /// properly maintain the feerate when adding this input to a transaction during coin selection.
 | ||||||
| @ -177,7 +177,7 @@ pub struct WeightedUtxo { | |||||||
|     pub utxo: Utxo, |     pub utxo: Utxo, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Debug, Clone, PartialEq)] | #[derive(Debug, Clone, PartialEq, Eq)] | ||||||
| /// An unspent transaction output (UTXO).
 | /// An unspent transaction output (UTXO).
 | ||||||
| pub enum Utxo { | pub enum Utxo { | ||||||
|     /// A UTXO owned by the local wallet.
 |     /// A UTXO owned by the local wallet.
 | ||||||
| @ -224,7 +224,7 @@ impl Utxo { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// A wallet transaction
 | /// A wallet transaction
 | ||||||
| #[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq, Default)] | #[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq)] | ||||||
| pub struct TransactionDetails { | pub struct TransactionDetails { | ||||||
|     /// Optional transaction
 |     /// Optional transaction
 | ||||||
|     pub transaction: Option<Transaction>, |     pub transaction: Option<Transaction>, | ||||||
|  | |||||||
| @ -1,158 +0,0 @@ | |||||||
| // Bitcoin Dev Kit
 |  | ||||||
| // Written in 2020 by Alekos Filini <alekos.filini@gmail.com>
 |  | ||||||
| //
 |  | ||||||
| // Copyright (c) 2020-2021 Bitcoin Dev Kit Developers
 |  | ||||||
| //
 |  | ||||||
| // This file is licensed under the Apache License, Version 2.0 <LICENSE-APACHE
 |  | ||||||
| // or http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
 |  | ||||||
| // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your option.
 |  | ||||||
| // You may not use this file except in accordance with one or both of these
 |  | ||||||
| // licenses.
 |  | ||||||
| 
 |  | ||||||
| //! Address validation callbacks
 |  | ||||||
| //!
 |  | ||||||
| //! The typical usage of those callbacks is for displaying the newly-generated address on a
 |  | ||||||
| //! hardware wallet, so that the user can cross-check its correctness.
 |  | ||||||
| //!
 |  | ||||||
| //! More generally speaking though, these callbacks can also be used to "do something" every time
 |  | ||||||
| //! an address is generated, without necessarily checking or validating it.
 |  | ||||||
| //!
 |  | ||||||
| //! An address validator can be attached to a [`Wallet`](super::Wallet) by using the
 |  | ||||||
| //! [`Wallet::add_address_validator`](super::Wallet::add_address_validator) method, and
 |  | ||||||
| //! whenever a new address is generated (either explicitly by the user with
 |  | ||||||
| //! [`Wallet::get_address`](super::Wallet::get_address) or internally to create a change
 |  | ||||||
| //! address) all the attached validators will be polled, in sequence. All of them must complete
 |  | ||||||
| //! successfully to continue.
 |  | ||||||
| //!
 |  | ||||||
| //! ## Example
 |  | ||||||
| //!
 |  | ||||||
| //! ```
 |  | ||||||
| //! # use std::sync::Arc;
 |  | ||||||
| //! # use bitcoin::*;
 |  | ||||||
| //! # use bdk::address_validator::*;
 |  | ||||||
| //! # use bdk::database::*;
 |  | ||||||
| //! # use bdk::*;
 |  | ||||||
| //! # use bdk::wallet::AddressIndex::New;
 |  | ||||||
| //! #[derive(Debug)]
 |  | ||||||
| //! struct PrintAddressAndContinue;
 |  | ||||||
| //!
 |  | ||||||
| //! impl AddressValidator for PrintAddressAndContinue {
 |  | ||||||
| //!     fn validate(
 |  | ||||||
| //!         &self,
 |  | ||||||
| //!         keychain: KeychainKind,
 |  | ||||||
| //!         hd_keypaths: &HdKeyPaths,
 |  | ||||||
| //!         script: &Script
 |  | ||||||
| //!     ) -> Result<(), AddressValidatorError> {
 |  | ||||||
| //!         let address = Address::from_script(script, Network::Testnet)
 |  | ||||||
| //!             .as_ref()
 |  | ||||||
| //!             .map(Address::to_string)
 |  | ||||||
| //!             .unwrap_or(script.to_string());
 |  | ||||||
| //!         println!("New address of type {:?}: {}", keychain, address);
 |  | ||||||
| //!         println!("HD keypaths: {:#?}", hd_keypaths);
 |  | ||||||
| //!
 |  | ||||||
| //!         Ok(())
 |  | ||||||
| //!     }
 |  | ||||||
| //! }
 |  | ||||||
| //!
 |  | ||||||
| //! let descriptor = "wpkh(tpubD6NzVbkrYhZ4Xferm7Pz4VnjdcDPFyjVu5K4iZXQ4pVN8Cks4pHVowTBXBKRhX64pkRyJZJN5xAKj4UDNnLPb5p2sSKXhewoYx5GbTdUFWq/*)";
 |  | ||||||
| //! let mut wallet = Wallet::new(descriptor, None, Network::Testnet, MemoryDatabase::default())?;
 |  | ||||||
| //! wallet.add_address_validator(Arc::new(PrintAddressAndContinue));
 |  | ||||||
| //!
 |  | ||||||
| //! let address = wallet.get_address(New)?;
 |  | ||||||
| //! println!("Address: {}", address);
 |  | ||||||
| //! # Ok::<(), bdk::Error>(())
 |  | ||||||
| //! ```
 |  | ||||||
| 
 |  | ||||||
| use std::fmt; |  | ||||||
| 
 |  | ||||||
| use bitcoin::Script; |  | ||||||
| 
 |  | ||||||
| use crate::descriptor::HdKeyPaths; |  | ||||||
| use crate::types::KeychainKind; |  | ||||||
| 
 |  | ||||||
| /// Errors that can be returned to fail the validation of an address
 |  | ||||||
| #[derive(Debug, Clone, PartialEq, Eq)] |  | ||||||
| pub enum AddressValidatorError { |  | ||||||
|     /// User rejected the address
 |  | ||||||
|     UserRejected, |  | ||||||
|     /// Network connection error
 |  | ||||||
|     ConnectionError, |  | ||||||
|     /// Network request timeout error
 |  | ||||||
|     TimeoutError, |  | ||||||
|     /// Invalid script
 |  | ||||||
|     InvalidScript, |  | ||||||
|     /// A custom error message
 |  | ||||||
|     Message(String), |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl fmt::Display for AddressValidatorError { |  | ||||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |  | ||||||
|         write!(f, "{:?}", self) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl std::error::Error for AddressValidatorError {} |  | ||||||
| 
 |  | ||||||
| /// Trait to build address validators
 |  | ||||||
| ///
 |  | ||||||
| /// All the address validators attached to a wallet with [`Wallet::add_address_validator`](super::Wallet::add_address_validator) will be polled
 |  | ||||||
| /// every time an address (external or internal) is generated by the wallet. Errors returned in the
 |  | ||||||
| /// validator will be propagated up to the original caller that triggered the address generation.
 |  | ||||||
| ///
 |  | ||||||
| /// For a usage example see [this module](crate::address_validator)'s documentation.
 |  | ||||||
| #[deprecated = "AddressValidator was rarely used. Address validation can occur outside of BDK"] |  | ||||||
| pub trait AddressValidator: Send + Sync + fmt::Debug { |  | ||||||
|     /// Validate or inspect an address
 |  | ||||||
|     fn validate( |  | ||||||
|         &self, |  | ||||||
|         keychain: KeychainKind, |  | ||||||
|         hd_keypaths: &HdKeyPaths, |  | ||||||
|         script: &Script, |  | ||||||
|     ) -> Result<(), AddressValidatorError>; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[cfg(test)] |  | ||||||
| mod test { |  | ||||||
|     use std::sync::Arc; |  | ||||||
| 
 |  | ||||||
|     use super::*; |  | ||||||
|     use crate::wallet::AddressIndex::New; |  | ||||||
|     use crate::wallet::{get_funded_wallet, test::get_test_wpkh}; |  | ||||||
| 
 |  | ||||||
|     #[derive(Debug)] |  | ||||||
|     struct TestValidator; |  | ||||||
|     #[allow(deprecated)] |  | ||||||
|     impl AddressValidator for TestValidator { |  | ||||||
|         fn validate( |  | ||||||
|             &self, |  | ||||||
|             _keychain: KeychainKind, |  | ||||||
|             _hd_keypaths: &HdKeyPaths, |  | ||||||
|             _script: &bitcoin::Script, |  | ||||||
|         ) -> Result<(), AddressValidatorError> { |  | ||||||
|             Err(AddressValidatorError::InvalidScript) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     #[should_panic(expected = "InvalidScript")] |  | ||||||
|     fn test_address_validator_external() { |  | ||||||
|         let (mut wallet, _, _) = get_funded_wallet(get_test_wpkh()); |  | ||||||
|         #[allow(deprecated)] |  | ||||||
|         wallet.add_address_validator(Arc::new(TestValidator)); |  | ||||||
| 
 |  | ||||||
|         wallet.get_address(New).unwrap(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     #[should_panic(expected = "InvalidScript")] |  | ||||||
|     fn test_address_validator_internal() { |  | ||||||
|         let (mut wallet, descriptors, _) = get_funded_wallet(get_test_wpkh()); |  | ||||||
|         #[allow(deprecated)] |  | ||||||
|         wallet.add_address_validator(Arc::new(TestValidator)); |  | ||||||
| 
 |  | ||||||
|         let addr = crate::testutils!(@external descriptors, 10); |  | ||||||
|         let mut builder = wallet.build_tx(); |  | ||||||
|         builder.add_recipient(addr.script_pubkey(), 25_000); |  | ||||||
|         builder.finish().unwrap(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @ -310,7 +310,7 @@ pub fn decide_change(remaining_amount: u64, fee_rate: FeeRate, drain_script: &Sc | |||||||
|     let drain_val = remaining_amount.saturating_sub(change_fee); |     let drain_val = remaining_amount.saturating_sub(change_fee); | ||||||
| 
 | 
 | ||||||
|     if drain_val.is_dust(drain_script) { |     if drain_val.is_dust(drain_script) { | ||||||
|         let dust_threshold = drain_script.dust_value().as_sat(); |         let dust_threshold = drain_script.dust_value().to_sat(); | ||||||
|         Excess::NoChange { |         Excess::NoChange { | ||||||
|             dust_threshold, |             dust_threshold, | ||||||
|             change_fee, |             change_fee, | ||||||
| @ -835,7 +835,7 @@ mod test { | |||||||
|                     ) |                     ) | ||||||
|                     .unwrap(), |                     .unwrap(), | ||||||
|                     txout: TxOut { |                     txout: TxOut { | ||||||
|                         value: rng.gen_range(0, 200000000), |                         value: rng.gen_range(0..200000000), | ||||||
|                         script_pubkey: Script::new(), |                         script_pubkey: Script::new(), | ||||||
|                     }, |                     }, | ||||||
|                     keychain: KeychainKind::External, |                     keychain: KeychainKind::External, | ||||||
| @ -866,7 +866,7 @@ mod test { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn sum_random_utxos(mut rng: &mut StdRng, utxos: &mut Vec<WeightedUtxo>) -> u64 { |     fn sum_random_utxos(mut rng: &mut StdRng, utxos: &mut Vec<WeightedUtxo>) -> u64 { | ||||||
|         let utxos_picked_len = rng.gen_range(2, utxos.len() / 2); |         let utxos_picked_len = rng.gen_range(2..utxos.len() / 2); | ||||||
|         utxos.shuffle(&mut rng); |         utxos.shuffle(&mut rng); | ||||||
|         utxos[..utxos_picked_len] |         utxos[..utxos_picked_len] | ||||||
|             .iter() |             .iter() | ||||||
| @ -1226,6 +1226,7 @@ mod test { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|  |     #[ignore] | ||||||
|     fn test_bnb_coin_selection_required_not_enough() { |     fn test_bnb_coin_selection_required_not_enough() { | ||||||
|         let utxos = get_test_utxos(); |         let utxos = get_test_utxos(); | ||||||
|         let database = MemoryDatabase::default(); |         let database = MemoryDatabase::default(); | ||||||
|  | |||||||
| @ -24,20 +24,17 @@ use std::sync::Arc; | |||||||
| use bitcoin::secp256k1::Secp256k1; | use bitcoin::secp256k1::Secp256k1; | ||||||
| 
 | 
 | ||||||
| use bitcoin::consensus::encode::serialize; | use bitcoin::consensus::encode::serialize; | ||||||
| use bitcoin::util::{psbt, taproot}; | use bitcoin::util::psbt; | ||||||
| use bitcoin::{ | use bitcoin::{ | ||||||
|     Address, EcdsaSighashType, Network, OutPoint, SchnorrSighashType, Script, Transaction, TxOut, |     Address, EcdsaSighashType, LockTime, Network, OutPoint, SchnorrSighashType, Script, Sequence, | ||||||
|     Txid, Witness, |     Transaction, TxOut, Txid, Witness, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| use miniscript::descriptor::DescriptorTrait; | use miniscript::psbt::{PsbtExt, PsbtInputExt, PsbtInputSatisfier}; | ||||||
| use miniscript::psbt::PsbtInputSatisfier; |  | ||||||
| use miniscript::ToPublicKey; |  | ||||||
| 
 | 
 | ||||||
| #[allow(unused_imports)] | #[allow(unused_imports)] | ||||||
| use log::{debug, error, info, trace}; | use log::{debug, error, info, trace}; | ||||||
| 
 | 
 | ||||||
| pub mod address_validator; |  | ||||||
| pub mod coin_selection; | pub mod coin_selection; | ||||||
| pub mod export; | pub mod export; | ||||||
| pub mod signer; | pub mod signer; | ||||||
| @ -54,24 +51,20 @@ pub mod hardwaresigner; | |||||||
| 
 | 
 | ||||||
| pub use utils::IsDust; | pub use utils::IsDust; | ||||||
| 
 | 
 | ||||||
| #[allow(deprecated)] |  | ||||||
| use address_validator::AddressValidator; |  | ||||||
| use coin_selection::DefaultCoinSelectionAlgorithm; | use coin_selection::DefaultCoinSelectionAlgorithm; | ||||||
| use signer::{SignOptions, SignerOrdering, SignersContainer, TransactionSigner}; | use signer::{SignOptions, SignerOrdering, SignersContainer, TransactionSigner}; | ||||||
| use tx_builder::{BumpFee, CreateTx, FeePolicy, TxBuilder, TxParams}; | use tx_builder::{BumpFee, CreateTx, FeePolicy, TxBuilder, TxParams}; | ||||||
| use utils::{check_nlocktime, check_nsequence_rbf, After, Older, SecpCtx}; | use utils::{check_nsequence_rbf, After, Older, SecpCtx}; | ||||||
| 
 | 
 | ||||||
| use crate::blockchain::{GetHeight, NoopProgress, Progress, WalletSync}; | use crate::blockchain::{GetHeight, NoopProgress, Progress, WalletSync}; | ||||||
| use crate::database::memory::MemoryDatabase; | use crate::database::memory::MemoryDatabase; | ||||||
| use crate::database::{AnyDatabase, BatchDatabase, BatchOperations, DatabaseUtils, SyncTime}; | use crate::database::{AnyDatabase, BatchDatabase, BatchOperations, DatabaseUtils, SyncTime}; | ||||||
| use crate::descriptor::derived::AsDerived; |  | ||||||
| use crate::descriptor::policy::BuildSatisfaction; | use crate::descriptor::policy::BuildSatisfaction; | ||||||
| use crate::descriptor::{ | use crate::descriptor::{ | ||||||
|     get_checksum, into_wallet_descriptor_checked, DerivedDescriptor, DerivedDescriptorMeta, |     get_checksum, into_wallet_descriptor_checked, DerivedDescriptor, DescriptorMeta, | ||||||
|     DescriptorMeta, DescriptorScripts, ExtendedDescriptor, ExtractPolicy, IntoWalletDescriptor, |     ExtendedDescriptor, ExtractPolicy, IntoWalletDescriptor, Policy, XKeyUtils, | ||||||
|     Policy, XKeyUtils, |  | ||||||
| }; | }; | ||||||
| use crate::error::Error; | use crate::error::{Error, MiniscriptPsbtError}; | ||||||
| use crate::psbt::PsbtUtils; | use crate::psbt::PsbtUtils; | ||||||
| use crate::signer::SignerError; | use crate::signer::SignerError; | ||||||
| use crate::testutils; | use crate::testutils; | ||||||
| @ -100,9 +93,6 @@ pub struct Wallet<D> { | |||||||
|     signers: Arc<SignersContainer>, |     signers: Arc<SignersContainer>, | ||||||
|     change_signers: Arc<SignersContainer>, |     change_signers: Arc<SignersContainer>, | ||||||
| 
 | 
 | ||||||
|     #[allow(deprecated)] |  | ||||||
|     address_validators: Vec<Arc<dyn AddressValidator>>, |  | ||||||
| 
 |  | ||||||
|     network: Network, |     network: Network, | ||||||
| 
 | 
 | ||||||
|     database: RefCell<D>, |     database: RefCell<D>, | ||||||
| @ -143,7 +133,7 @@ pub enum AddressIndex { | |||||||
| 
 | 
 | ||||||
| /// A derived address and the index it was found at
 | /// A derived address and the index it was found at
 | ||||||
| /// For convenience this automatically derefs to `Address`
 | /// For convenience this automatically derefs to `Address`
 | ||||||
| #[derive(Debug, PartialEq)] | #[derive(Debug, PartialEq, Eq)] | ||||||
| pub struct AddressInfo { | pub struct AddressInfo { | ||||||
|     /// Child index of this address
 |     /// Child index of this address
 | ||||||
|     pub index: u32, |     pub index: u32, | ||||||
| @ -236,7 +226,6 @@ where | |||||||
|             change_descriptor, |             change_descriptor, | ||||||
|             signers, |             signers, | ||||||
|             change_signers, |             change_signers, | ||||||
|             address_validators: Vec::new(), |  | ||||||
|             network, |             network, | ||||||
|             database: RefCell::new(database), |             database: RefCell::new(database), | ||||||
|             secp, |             secp, | ||||||
| @ -254,7 +243,7 @@ where | |||||||
| 
 | 
 | ||||||
|         let address_result = self |         let address_result = self | ||||||
|             .get_descriptor_for_keychain(keychain) |             .get_descriptor_for_keychain(keychain) | ||||||
|             .as_derived(incremented_index, &self.secp) |             .at_derivation_index(incremented_index) | ||||||
|             .address(self.network); |             .address(self.network); | ||||||
| 
 | 
 | ||||||
|         address_result |         address_result | ||||||
| @ -273,7 +262,7 @@ where | |||||||
| 
 | 
 | ||||||
|         let derived_key = self |         let derived_key = self | ||||||
|             .get_descriptor_for_keychain(keychain) |             .get_descriptor_for_keychain(keychain) | ||||||
|             .as_derived(current_index, &self.secp); |             .at_derivation_index(current_index); | ||||||
| 
 | 
 | ||||||
|         let script_pubkey = derived_key.script_pubkey(); |         let script_pubkey = derived_key.script_pubkey(); | ||||||
| 
 | 
 | ||||||
| @ -301,7 +290,7 @@ where | |||||||
|     // Return derived address for the descriptor of given [`KeychainKind`] at a specific index
 |     // Return derived address for the descriptor of given [`KeychainKind`] at a specific index
 | ||||||
|     fn peek_address(&self, index: u32, keychain: KeychainKind) -> Result<AddressInfo, Error> { |     fn peek_address(&self, index: u32, keychain: KeychainKind) -> Result<AddressInfo, Error> { | ||||||
|         self.get_descriptor_for_keychain(keychain) |         self.get_descriptor_for_keychain(keychain) | ||||||
|             .as_derived(index, &self.secp) |             .at_derivation_index(index) | ||||||
|             .address(self.network) |             .address(self.network) | ||||||
|             .map(|address| AddressInfo { |             .map(|address| AddressInfo { | ||||||
|                 index, |                 index, | ||||||
| @ -317,7 +306,7 @@ where | |||||||
|         self.set_index(keychain, index)?; |         self.set_index(keychain, index)?; | ||||||
| 
 | 
 | ||||||
|         self.get_descriptor_for_keychain(keychain) |         self.get_descriptor_for_keychain(keychain) | ||||||
|             .as_derived(index, &self.secp) |             .at_derivation_index(index) | ||||||
|             .address(self.network) |             .address(self.network) | ||||||
|             .map(|address| AddressInfo { |             .map(|address| AddressInfo { | ||||||
|                 index, |                 index, | ||||||
| @ -366,7 +355,7 @@ where | |||||||
|     /// transaction output scripts.
 |     /// transaction output scripts.
 | ||||||
|     pub fn ensure_addresses_cached(&self, max_addresses: u32) -> Result<bool, Error> { |     pub fn ensure_addresses_cached(&self, max_addresses: u32) -> Result<bool, Error> { | ||||||
|         let mut new_addresses_cached = false; |         let mut new_addresses_cached = false; | ||||||
|         let max_address = match self.descriptor.is_deriveable() { |         let max_address = match self.descriptor.has_wildcard() { | ||||||
|             false => 0, |             false => 0, | ||||||
|             true => max_addresses, |             true => max_addresses, | ||||||
|         }; |         }; | ||||||
| @ -383,7 +372,7 @@ where | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if let Some(change_descriptor) = &self.change_descriptor { |         if let Some(change_descriptor) = &self.change_descriptor { | ||||||
|             let max_address = match change_descriptor.is_deriveable() { |             let max_address = match change_descriptor.has_wildcard() { | ||||||
|                 false => 0, |                 false => 0, | ||||||
|                 true => max_addresses, |                 true => max_addresses, | ||||||
|             }; |             }; | ||||||
| @ -552,24 +541,6 @@ where | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// Add an address validator
 |  | ||||||
|     ///
 |  | ||||||
|     /// See [the `address_validator` module](address_validator) for an example.
 |  | ||||||
|     #[deprecated] |  | ||||||
|     #[allow(deprecated)] |  | ||||||
|     pub fn add_address_validator(&mut self, validator: Arc<dyn AddressValidator>) { |  | ||||||
|         self.address_validators.push(validator); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// Get the address validators
 |  | ||||||
|     ///
 |  | ||||||
|     /// See [the `address_validator` module](address_validator).
 |  | ||||||
|     #[deprecated] |  | ||||||
|     #[allow(deprecated)] |  | ||||||
|     pub fn get_address_validators(&self) -> &[Arc<dyn AddressValidator>] { |  | ||||||
|         &self.address_validators |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// Start building a transaction.
 |     /// Start building a transaction.
 | ||||||
|     ///
 |     ///
 | ||||||
|     /// This returns a blank [`TxBuilder`] from which you can specify the parameters for the transaction.
 |     /// This returns a blank [`TxBuilder`] from which you can specify the parameters for the transaction.
 | ||||||
| @ -684,10 +655,9 @@ where | |||||||
|         // We use a match here instead of a map_or_else as it's way more readable :)
 |         // We use a match here instead of a map_or_else as it's way more readable :)
 | ||||||
|         let current_height = match params.current_height { |         let current_height = match params.current_height { | ||||||
|             // If they didn't tell us the current height, we assume it's the latest sync height.
 |             // If they didn't tell us the current height, we assume it's the latest sync height.
 | ||||||
|             None => self |             None => self.database().get_sync_time()?.map(|sync_time| { | ||||||
|                 .database() |                 LockTime::from_height(sync_time.block_time.height).expect("Invalid height") | ||||||
|                 .get_sync_time()? |             }), | ||||||
|                 .map(|sync_time| sync_time.block_time.height), |  | ||||||
|             h => h, |             h => h, | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
| @ -697,24 +667,33 @@ where | |||||||
|                 // Fee sniping can be partially prevented by setting the timelock
 |                 // Fee sniping can be partially prevented by setting the timelock
 | ||||||
|                 // to current_height. If we don't know the current_height,
 |                 // to current_height. If we don't know the current_height,
 | ||||||
|                 // we default to 0.
 |                 // we default to 0.
 | ||||||
|                 let fee_sniping_height = current_height.unwrap_or(0); |                 let fee_sniping_height = current_height.unwrap_or(LockTime::ZERO); | ||||||
|  | 
 | ||||||
|                 // We choose the biggest between the required nlocktime and the fee sniping
 |                 // We choose the biggest between the required nlocktime and the fee sniping
 | ||||||
|                 // height
 |                 // height
 | ||||||
|                 std::cmp::max(requirements.timelock.unwrap_or(0), fee_sniping_height) |                 match requirements.timelock { | ||||||
|  |                     // No requirement, just use the fee_sniping_height
 | ||||||
|  |                     None => fee_sniping_height, | ||||||
|  |                     // There's a block-based requirement, but the value is lower than the fee_sniping_height
 | ||||||
|  |                     Some(value @ LockTime::Blocks(_)) if value < fee_sniping_height => fee_sniping_height, | ||||||
|  |                     // There's a time-based requirement or a block-based requirement greater
 | ||||||
|  |                     // than the fee_sniping_height use that value
 | ||||||
|  |                     Some(value) => value, | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             // Specific nLockTime required and we have no constraints, so just set to that value
 |             // Specific nLockTime required and we have no constraints, so just set to that value
 | ||||||
|             Some(x) if requirements.timelock.is_none() => x, |             Some(x) if requirements.timelock.is_none() => x, | ||||||
|             // Specific nLockTime required and it's compatible with the constraints
 |             // Specific nLockTime required and it's compatible with the constraints
 | ||||||
|             Some(x) if check_nlocktime(x, requirements.timelock.unwrap()) => x, |             Some(x) if requirements.timelock.unwrap().is_same_unit(x) && x >= requirements.timelock.unwrap() => x, | ||||||
|             // Invalid nLockTime required
 |             // Invalid nLockTime required
 | ||||||
|             Some(x) => return Err(Error::Generic(format!("TxBuilder requested timelock of `{}`, but at least `{}` is required to spend from this script", x, requirements.timelock.unwrap()))) |             Some(x) => return Err(Error::Generic(format!("TxBuilder requested timelock of `{:?}`, but at least `{:?}` is required to spend from this script", x, requirements.timelock.unwrap()))) | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         let n_sequence = match (params.rbf, requirements.csv) { |         let n_sequence = match (params.rbf, requirements.csv) { | ||||||
|             // No RBF or CSV but there's an nLockTime, so the nSequence cannot be final
 |             // No RBF or CSV but there's an nLockTime, so the nSequence cannot be final
 | ||||||
|             (None, None) if lock_time != 0 => 0xFFFFFFFE, |             (None, None) if lock_time != LockTime::ZERO => Sequence::ENABLE_LOCKTIME_NO_RBF, | ||||||
|             // No RBF, CSV or nLockTime, make the transaction final
 |             // No RBF, CSV or nLockTime, make the transaction final
 | ||||||
|             (None, None) => 0xFFFFFFFF, |             (None, None) => Sequence::MAX, | ||||||
| 
 | 
 | ||||||
|             // No RBF requested, use the value from CSV. Note that this value is by definition
 |             // No RBF requested, use the value from CSV. Note that this value is by definition
 | ||||||
|             // non-final, so even if a timelock is enabled this nSequence is fine, hence why we
 |             // non-final, so even if a timelock is enabled this nSequence is fine, hence why we
 | ||||||
| @ -722,7 +701,7 @@ where | |||||||
|             (None, Some(csv)) => csv, |             (None, Some(csv)) => csv, | ||||||
| 
 | 
 | ||||||
|             // RBF with a specific value but that value is too high
 |             // RBF with a specific value but that value is too high
 | ||||||
|             (Some(tx_builder::RbfValue::Value(rbf)), _) if rbf >= 0xFFFFFFFE => { |             (Some(tx_builder::RbfValue::Value(rbf)), _) if !rbf.is_rbf() => { | ||||||
|                 return Err(Error::Generic( |                 return Err(Error::Generic( | ||||||
|                     "Cannot enable RBF with a nSequence >= 0xFFFFFFFE".into(), |                     "Cannot enable RBF with a nSequence >= 0xFFFFFFFE".into(), | ||||||
|                 )) |                 )) | ||||||
| @ -732,7 +711,7 @@ where | |||||||
|                 if !check_nsequence_rbf(rbf, csv) => |                 if !check_nsequence_rbf(rbf, csv) => | ||||||
|             { |             { | ||||||
|                 return Err(Error::Generic(format!( |                 return Err(Error::Generic(format!( | ||||||
|                     "Cannot enable RBF with nSequence `{}` given a required OP_CSV of `{}`", |                     "Cannot enable RBF with nSequence `{:?}` given a required OP_CSV of `{:?}`", | ||||||
|                     rbf, csv |                     rbf, csv | ||||||
|                 ))) |                 ))) | ||||||
|             } |             } | ||||||
| @ -775,7 +754,7 @@ where | |||||||
| 
 | 
 | ||||||
|         let mut tx = Transaction { |         let mut tx = Transaction { | ||||||
|             version, |             version, | ||||||
|             lock_time, |             lock_time: lock_time.into(), | ||||||
|             input: vec![], |             input: vec![], | ||||||
|             output: vec![], |             output: vec![], | ||||||
|         }; |         }; | ||||||
| @ -840,7 +819,7 @@ where | |||||||
|             params.drain_wallet, |             params.drain_wallet, | ||||||
|             params.manually_selected_only, |             params.manually_selected_only, | ||||||
|             params.bumping_fee.is_some(), // we mandate confirmed transactions if we're bumping the fee
 |             params.bumping_fee.is_some(), // we mandate confirmed transactions if we're bumping the fee
 | ||||||
|             current_height, |             current_height.map(LockTime::to_consensus_u32), | ||||||
|         )?; |         )?; | ||||||
| 
 | 
 | ||||||
|         // get drain script
 |         // get drain script
 | ||||||
| @ -992,7 +971,11 @@ where | |||||||
|             Some(tx) => tx, |             Some(tx) => tx, | ||||||
|         }; |         }; | ||||||
|         let mut tx = details.transaction.take().unwrap(); |         let mut tx = details.transaction.take().unwrap(); | ||||||
|         if !tx.input.iter().any(|txin| txin.sequence <= 0xFFFFFFFD) { |         if !tx | ||||||
|  |             .input | ||||||
|  |             .iter() | ||||||
|  |             .any(|txin| txin.sequence.to_consensus_u32() <= 0xFFFFFFFD) | ||||||
|  |         { | ||||||
|             return Err(Error::IrreplaceableTransaction); |             return Err(Error::IrreplaceableTransaction); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @ -1118,8 +1101,9 @@ where | |||||||
|         psbt: &mut psbt::PartiallySignedTransaction, |         psbt: &mut psbt::PartiallySignedTransaction, | ||||||
|         sign_options: SignOptions, |         sign_options: SignOptions, | ||||||
|     ) -> Result<bool, Error> { |     ) -> Result<bool, Error> { | ||||||
|         // this helps us doing our job later
 |         // This adds all the PSBT metadata for the inputs, which will help us later figure out how
 | ||||||
|         self.add_input_hd_keypaths(psbt)?; |         // to derive our keys
 | ||||||
|  |         self.update_psbt_with_descriptor(psbt)?; | ||||||
| 
 | 
 | ||||||
|         // If we aren't allowed to use `witness_utxo`, ensure that every input (except p2tr and finalized ones)
 |         // If we aren't allowed to use `witness_utxo`, ensure that every input (except p2tr and finalized ones)
 | ||||||
|         // has the `non_witness_utxo`
 |         // has the `non_witness_utxo`
 | ||||||
| @ -1320,21 +1304,18 @@ where | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn get_descriptor_for_txout( |     fn get_descriptor_for_txout(&self, txout: &TxOut) -> Result<Option<DerivedDescriptor>, Error> { | ||||||
|         &self, |  | ||||||
|         txout: &TxOut, |  | ||||||
|     ) -> Result<Option<DerivedDescriptor<'_>>, Error> { |  | ||||||
|         Ok(self |         Ok(self | ||||||
|             .database |             .database | ||||||
|             .borrow() |             .borrow() | ||||||
|             .get_path_from_script_pubkey(&txout.script_pubkey)? |             .get_path_from_script_pubkey(&txout.script_pubkey)? | ||||||
|             .map(|(keychain, child)| (self.get_descriptor_for_keychain(keychain), child)) |             .map(|(keychain, child)| (self.get_descriptor_for_keychain(keychain), child)) | ||||||
|             .map(|(desc, child)| desc.as_derived(child, &self.secp))) |             .map(|(desc, child)| desc.at_derivation_index(child))) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn fetch_and_increment_index(&self, keychain: KeychainKind) -> Result<u32, Error> { |     fn fetch_and_increment_index(&self, keychain: KeychainKind) -> Result<u32, Error> { | ||||||
|         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); |         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); | ||||||
|         let index = match descriptor.is_deriveable() { |         let index = match descriptor.has_wildcard() { | ||||||
|             false => 0, |             false => 0, | ||||||
|             true => self.database.borrow_mut().increment_last_index(keychain)?, |             true => self.database.borrow_mut().increment_last_index(keychain)?, | ||||||
|         }; |         }; | ||||||
| @ -1348,22 +1329,12 @@ where | |||||||
|             self.cache_addresses(keychain, index, CACHE_ADDR_BATCH_SIZE)?; |             self.cache_addresses(keychain, index, CACHE_ADDR_BATCH_SIZE)?; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         let derived_descriptor = descriptor.as_derived(index, &self.secp); |  | ||||||
| 
 |  | ||||||
|         let hd_keypaths = derived_descriptor.get_hd_keypaths(&self.secp); |  | ||||||
|         let script = derived_descriptor.script_pubkey(); |  | ||||||
| 
 |  | ||||||
|         for validator in &self.address_validators { |  | ||||||
|             #[allow(deprecated)] |  | ||||||
|             validator.validate(keychain, &hd_keypaths, &script)?; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         Ok(index) |         Ok(index) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn fetch_index(&self, keychain: KeychainKind) -> Result<u32, Error> { |     fn fetch_index(&self, keychain: KeychainKind) -> Result<u32, Error> { | ||||||
|         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); |         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); | ||||||
|         let index = match descriptor.is_deriveable() { |         let index = match descriptor.has_wildcard() { | ||||||
|             false => Some(0), |             false => Some(0), | ||||||
|             true => self.database.borrow_mut().get_last_index(keychain)?, |             true => self.database.borrow_mut().get_last_index(keychain)?, | ||||||
|         }; |         }; | ||||||
| @ -1387,7 +1358,7 @@ where | |||||||
|         mut count: u32, |         mut count: u32, | ||||||
|     ) -> Result<(), Error> { |     ) -> Result<(), Error> { | ||||||
|         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); |         let (descriptor, keychain) = self._get_descriptor_for_keychain(keychain); | ||||||
|         if !descriptor.is_deriveable() { |         if !descriptor.has_wildcard() { | ||||||
|             if from > 0 { |             if from > 0 { | ||||||
|                 return Ok(()); |                 return Ok(()); | ||||||
|             } |             } | ||||||
| @ -1400,7 +1371,7 @@ where | |||||||
|         let start_time = time::Instant::new(); |         let start_time = time::Instant::new(); | ||||||
|         for i in from..(from + count) { |         for i in from..(from + count) { | ||||||
|             address_batch.set_script_pubkey( |             address_batch.set_script_pubkey( | ||||||
|                 &descriptor.as_derived(i, &self.secp).script_pubkey(), |                 &descriptor.at_derivation_index(i).script_pubkey(), | ||||||
|                 keychain, |                 keychain, | ||||||
|                 i, |                 i, | ||||||
|             )?; |             )?; | ||||||
| @ -1604,52 +1575,7 @@ where | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // probably redundant but it doesn't hurt...
 |         self.update_psbt_with_descriptor(&mut psbt)?; | ||||||
|         self.add_input_hd_keypaths(&mut psbt)?; |  | ||||||
| 
 |  | ||||||
|         // add metadata for the outputs
 |  | ||||||
|         for (psbt_output, tx_output) in psbt.outputs.iter_mut().zip(psbt.unsigned_tx.output.iter()) |  | ||||||
|         { |  | ||||||
|             if let Some((keychain, child)) = self |  | ||||||
|                 .database |  | ||||||
|                 .borrow() |  | ||||||
|                 .get_path_from_script_pubkey(&tx_output.script_pubkey)? |  | ||||||
|             { |  | ||||||
|                 let (desc, _) = self._get_descriptor_for_keychain(keychain); |  | ||||||
|                 let derived_descriptor = desc.as_derived(child, &self.secp); |  | ||||||
| 
 |  | ||||||
|                 if let miniscript::Descriptor::Tr(tr) = &derived_descriptor { |  | ||||||
|                     let tap_tree = if tr.taptree().is_some() { |  | ||||||
|                         let mut builder = taproot::TaprootBuilder::new(); |  | ||||||
|                         for (depth, ms) in tr.iter_scripts() { |  | ||||||
|                             let script = ms.encode(); |  | ||||||
|                             builder = builder.add_leaf(depth, script).expect( |  | ||||||
|                                 "Computing spend data on a valid Tree should always succeed", |  | ||||||
|                             ); |  | ||||||
|                         } |  | ||||||
|                         Some( |  | ||||||
|                             psbt::TapTree::from_builder(builder) |  | ||||||
|                                 .expect("The tree should always be valid"), |  | ||||||
|                         ) |  | ||||||
|                     } else { |  | ||||||
|                         None |  | ||||||
|                     }; |  | ||||||
|                     psbt_output.tap_tree = tap_tree; |  | ||||||
|                     psbt_output |  | ||||||
|                         .tap_key_origins |  | ||||||
|                         .append(&mut derived_descriptor.get_tap_key_origins(&self.secp)); |  | ||||||
|                     psbt_output.tap_internal_key = Some(tr.internal_key().to_x_only_pubkey()); |  | ||||||
|                 } else { |  | ||||||
|                     psbt_output |  | ||||||
|                         .bip32_derivation |  | ||||||
|                         .append(&mut derived_descriptor.get_hd_keypaths(&self.secp)); |  | ||||||
|                 } |  | ||||||
|                 if params.include_output_redeem_witness_script { |  | ||||||
|                     psbt_output.witness_script = derived_descriptor.psbt_witness_script(); |  | ||||||
|                     psbt_output.redeem_script = derived_descriptor.psbt_redeem_script(); |  | ||||||
|                 }; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         Ok(psbt) |         Ok(psbt) | ||||||
|     } |     } | ||||||
| @ -1675,29 +1601,11 @@ where | |||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         let desc = self.get_descriptor_for_keychain(keychain); |         let desc = self.get_descriptor_for_keychain(keychain); | ||||||
|         let derived_descriptor = desc.as_derived(child, &self.secp); |         let derived_descriptor = desc.at_derivation_index(child); | ||||||
| 
 | 
 | ||||||
|         if let miniscript::Descriptor::Tr(tr) = &derived_descriptor { |         psbt_input | ||||||
|             psbt_input.tap_key_origins = derived_descriptor.get_tap_key_origins(&self.secp); |             .update_with_descriptor_unchecked(&derived_descriptor) | ||||||
|             psbt_input.tap_internal_key = Some(tr.internal_key().to_x_only_pubkey()); |             .map_err(MiniscriptPsbtError::Conversion)?; | ||||||
| 
 |  | ||||||
|             let spend_info = tr.spend_info(); |  | ||||||
|             psbt_input.tap_merkle_root = spend_info.merkle_root(); |  | ||||||
|             psbt_input.tap_scripts = spend_info |  | ||||||
|                 .as_script_map() |  | ||||||
|                 .keys() |  | ||||||
|                 .filter_map(|script_ver| { |  | ||||||
|                     spend_info |  | ||||||
|                         .control_block(script_ver) |  | ||||||
|                         .map(|cb| (cb, script_ver.clone())) |  | ||||||
|                 }) |  | ||||||
|                 .collect(); |  | ||||||
|         } else { |  | ||||||
|             psbt_input.bip32_derivation = derived_descriptor.get_hd_keypaths(&self.secp); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         psbt_input.redeem_script = derived_descriptor.psbt_redeem_script(); |  | ||||||
|         psbt_input.witness_script = derived_descriptor.psbt_witness_script(); |  | ||||||
| 
 | 
 | ||||||
|         let prev_output = utxo.outpoint; |         let prev_output = utxo.outpoint; | ||||||
|         if let Some(prev_tx) = self.database.borrow().get_raw_tx(&prev_output.txid)? { |         if let Some(prev_tx) = self.database.borrow().get_raw_tx(&prev_output.txid)? { | ||||||
| @ -1711,38 +1619,47 @@ where | |||||||
|         Ok(psbt_input) |         Ok(psbt_input) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn add_input_hd_keypaths( |     fn update_psbt_with_descriptor( | ||||||
|         &self, |         &self, | ||||||
|         psbt: &mut psbt::PartiallySignedTransaction, |         psbt: &mut psbt::PartiallySignedTransaction, | ||||||
|     ) -> Result<(), Error> { |     ) -> Result<(), Error> { | ||||||
|         let mut input_utxos = Vec::with_capacity(psbt.inputs.len()); |         // We need to borrow `psbt` mutably within the loops, so we have to allocate a vec for all
 | ||||||
|         for n in 0..psbt.inputs.len() { |         // the input utxos and outputs
 | ||||||
|             input_utxos.push(psbt.get_utxo_for(n).clone()); |         //
 | ||||||
|         } |         // Clippy complains that the collect is not required, but that's wrong
 | ||||||
|  |         #[allow(clippy::needless_collect)] | ||||||
|  |         let utxos = (0..psbt.inputs.len()) | ||||||
|  |             .filter_map(|i| psbt.get_utxo_for(i).map(|utxo| (true, i, utxo))) | ||||||
|  |             .chain( | ||||||
|  |                 psbt.unsigned_tx | ||||||
|  |                     .output | ||||||
|  |                     .iter() | ||||||
|  |                     .enumerate() | ||||||
|  |                     .map(|(i, out)| (false, i, out.clone())), | ||||||
|  |             ) | ||||||
|  |             .collect::<Vec<_>>(); | ||||||
| 
 | 
 | ||||||
|         // try to add hd_keypaths if we've already seen the output
 |         // Try to figure out the keychain and derivation for every input and output
 | ||||||
|         for (psbt_input, out) in psbt.inputs.iter_mut().zip(input_utxos.iter()) { |         for (is_input, index, out) in utxos.into_iter() { | ||||||
|             if let Some(out) = out { |  | ||||||
|             if let Some((keychain, child)) = self |             if let Some((keychain, child)) = self | ||||||
|                 .database |                 .database | ||||||
|                 .borrow() |                 .borrow() | ||||||
|                 .get_path_from_script_pubkey(&out.script_pubkey)? |                 .get_path_from_script_pubkey(&out.script_pubkey)? | ||||||
|             { |             { | ||||||
|                     debug!("Found descriptor {:?}/{}", keychain, child); |                 debug!( | ||||||
|  |                     "Found descriptor for input #{} {:?}/{}", | ||||||
|  |                     index, keychain, child | ||||||
|  |                 ); | ||||||
| 
 | 
 | ||||||
|                     // merge hd_keypaths or tap_key_origins
 |  | ||||||
|                 let desc = self.get_descriptor_for_keychain(keychain); |                 let desc = self.get_descriptor_for_keychain(keychain); | ||||||
|                     if desc.is_taproot() { |                 let desc = desc.at_derivation_index(child); | ||||||
|                         let mut tap_key_origins = desc | 
 | ||||||
|                             .as_derived(child, &self.secp) |                 if is_input { | ||||||
|                             .get_tap_key_origins(&self.secp); |                     psbt.update_input_with_descriptor(index, &desc) | ||||||
|                         psbt_input.tap_key_origins.append(&mut tap_key_origins); |                         .map_err(MiniscriptPsbtError::UtxoUpdate)?; | ||||||
|                 } else { |                 } else { | ||||||
|                         let mut hd_keypaths = desc |                     psbt.update_output_with_descriptor(index, &desc) | ||||||
|                             .as_derived(child, &self.secp) |                         .map_err(MiniscriptPsbtError::OutputUpdate)?; | ||||||
|                             .get_hd_keypaths(&self.secp); |  | ||||||
|                         psbt_input.bip32_derivation.append(&mut hd_keypaths); |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @ -1781,12 +1698,12 @@ where | |||||||
| 
 | 
 | ||||||
|         // We need to ensure descriptor is derivable to fullfil "missing cache", otherwise we will
 |         // We need to ensure descriptor is derivable to fullfil "missing cache", otherwise we will
 | ||||||
|         // end up with an infinite loop
 |         // end up with an infinite loop
 | ||||||
|         let is_deriveable = self.descriptor.is_deriveable() |         let has_wildcard = self.descriptor.has_wildcard() | ||||||
|             && (self.change_descriptor.is_none() |             && (self.change_descriptor.is_none() | ||||||
|                 || self.change_descriptor.as_ref().unwrap().is_deriveable()); |                 || self.change_descriptor.as_ref().unwrap().has_wildcard()); | ||||||
| 
 | 
 | ||||||
|         // Restrict max rounds in case of faulty "missing cache" implementation by blockchain
 |         // Restrict max rounds in case of faulty "missing cache" implementation by blockchain
 | ||||||
|         let max_rounds = if is_deriveable { 100 } else { 1 }; |         let max_rounds = if has_wildcard { 100 } else { 1 }; | ||||||
| 
 | 
 | ||||||
|         for _ in 0..max_rounds { |         for _ in 0..max_rounds { | ||||||
|             let sync_res = |             let sync_res = | ||||||
| @ -1921,7 +1838,7 @@ pub fn get_funded_wallet( | |||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| pub(crate) mod test { | pub(crate) mod test { | ||||||
|     use bitcoin::{util::psbt, Network}; |     use bitcoin::{util::psbt, Network, PackedLockTime, Sequence}; | ||||||
| 
 | 
 | ||||||
|     use crate::database::Database; |     use crate::database::Database; | ||||||
|     use crate::types::KeychainKind; |     use crate::types::KeychainKind; | ||||||
| @ -2234,7 +2151,7 @@ pub(crate) mod test { | |||||||
| 
 | 
 | ||||||
|         // Since we never synced the wallet we don't have a last_sync_height
 |         // Since we never synced the wallet we don't have a last_sync_height
 | ||||||
|         // we could use to try to prevent fee sniping. We default to 0.
 |         // we could use to try to prevent fee sniping. We default to 0.
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, 0); |         assert_eq!(psbt.unsigned_tx.lock_time, PackedLockTime(0)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2259,7 +2176,7 @@ pub(crate) mod test { | |||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         // current_height will override the last sync height
 |         // current_height will override the last sync height
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, current_height); |         assert_eq!(psbt.unsigned_tx.lock_time, PackedLockTime(current_height)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2282,7 +2199,10 @@ pub(crate) mod test { | |||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         // If there's no current_height we're left with using the last sync height
 |         // If there's no current_height we're left with using the last sync height
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, sync_time.block_time.height); |         assert_eq!( | ||||||
|  |             psbt.unsigned_tx.lock_time, | ||||||
|  |             PackedLockTime(sync_time.block_time.height) | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2293,7 +2213,7 @@ pub(crate) mod test { | |||||||
|         builder.add_recipient(addr.script_pubkey(), 25_000); |         builder.add_recipient(addr.script_pubkey(), 25_000); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, 100_000); |         assert_eq!(psbt.unsigned_tx.lock_time, PackedLockTime(100_000)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2304,13 +2224,13 @@ pub(crate) mod test { | |||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .current_height(630_001) |             .current_height(630_001) | ||||||
|             .nlocktime(630_000); |             .nlocktime(LockTime::from_height(630_000).unwrap()); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         // When we explicitly specify a nlocktime
 |         // When we explicitly specify a nlocktime
 | ||||||
|         // we don't try any fee sniping prevention trick
 |         // we don't try any fee sniping prevention trick
 | ||||||
|         // (we ignore the current_height)
 |         // (we ignore the current_height)
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, 630_000); |         assert_eq!(psbt.unsigned_tx.lock_time, PackedLockTime(630_000)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2320,15 +2240,15 @@ pub(crate) mod test { | |||||||
|         let mut builder = wallet.build_tx(); |         let mut builder = wallet.build_tx(); | ||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .nlocktime(630_000); |             .nlocktime(LockTime::from_height(630_000).unwrap()); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.lock_time, 630_000); |         assert_eq!(psbt.unsigned_tx.lock_time, PackedLockTime(630_000)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     #[should_panic(
 |     #[should_panic(
 | ||||||
|         expected = "TxBuilder requested timelock of `50000`, but at least `100000` is required to spend from this script" |         expected = "TxBuilder requested timelock of `Blocks(Height(50000))`, but at least `Blocks(Height(100000))` is required to spend from this script" | ||||||
|     )] |     )] | ||||||
|     fn test_create_tx_custom_locktime_incompatible_with_cltv() { |     fn test_create_tx_custom_locktime_incompatible_with_cltv() { | ||||||
|         let (wallet, _, _) = get_funded_wallet(get_test_single_sig_cltv()); |         let (wallet, _, _) = get_funded_wallet(get_test_single_sig_cltv()); | ||||||
| @ -2336,7 +2256,7 @@ pub(crate) mod test { | |||||||
|         let mut builder = wallet.build_tx(); |         let mut builder = wallet.build_tx(); | ||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .nlocktime(50000); |             .nlocktime(LockTime::from_height(50000).unwrap()); | ||||||
|         builder.finish().unwrap(); |         builder.finish().unwrap(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -2348,7 +2268,7 @@ pub(crate) mod test { | |||||||
|         builder.add_recipient(addr.script_pubkey(), 25_000); |         builder.add_recipient(addr.script_pubkey(), 25_000); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 6); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(6)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2362,12 +2282,12 @@ pub(crate) mod test { | |||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
|         // When CSV is enabled it takes precedence over the rbf value (unless forced by the user).
 |         // When CSV is enabled it takes precedence over the rbf value (unless forced by the user).
 | ||||||
|         // It will be set to the OP_CSV value, in this case 6
 |         // It will be set to the OP_CSV value, in this case 6
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 6); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(6)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     #[should_panic(
 |     #[should_panic(
 | ||||||
|         expected = "Cannot enable RBF with nSequence `3` given a required OP_CSV of `6`" |         expected = "Cannot enable RBF with nSequence `Sequence(3)` given a required OP_CSV of `Sequence(6)`" | ||||||
|     )] |     )] | ||||||
|     fn test_create_tx_with_custom_rbf_csv() { |     fn test_create_tx_with_custom_rbf_csv() { | ||||||
|         let (wallet, _, _) = get_funded_wallet(get_test_single_sig_csv()); |         let (wallet, _, _) = get_funded_wallet(get_test_single_sig_csv()); | ||||||
| @ -2375,7 +2295,7 @@ pub(crate) mod test { | |||||||
|         let mut builder = wallet.build_tx(); |         let mut builder = wallet.build_tx(); | ||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .enable_rbf_with_sequence(3); |             .enable_rbf_with_sequence(Sequence(3)); | ||||||
|         builder.finish().unwrap(); |         builder.finish().unwrap(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -2387,7 +2307,7 @@ pub(crate) mod test { | |||||||
|         builder.add_recipient(addr.script_pubkey(), 25_000); |         builder.add_recipient(addr.script_pubkey(), 25_000); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 0xFFFFFFFE); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(0xFFFFFFFE)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2398,7 +2318,7 @@ pub(crate) mod test { | |||||||
|         let mut builder = wallet.build_tx(); |         let mut builder = wallet.build_tx(); | ||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .enable_rbf_with_sequence(0xFFFFFFFE); |             .enable_rbf_with_sequence(Sequence(0xFFFFFFFE)); | ||||||
|         builder.finish().unwrap(); |         builder.finish().unwrap(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -2409,10 +2329,10 @@ pub(crate) mod test { | |||||||
|         let mut builder = wallet.build_tx(); |         let mut builder = wallet.build_tx(); | ||||||
|         builder |         builder | ||||||
|             .add_recipient(addr.script_pubkey(), 25_000) |             .add_recipient(addr.script_pubkey(), 25_000) | ||||||
|             .enable_rbf_with_sequence(0xDEADBEEF); |             .enable_rbf_with_sequence(Sequence(0xDEADBEEF)); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 0xDEADBEEF); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(0xDEADBEEF)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2439,7 +2359,7 @@ pub(crate) mod test { | |||||||
|         builder.add_recipient(addr.script_pubkey(), 25_000); |         builder.add_recipient(addr.script_pubkey(), 25_000); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 0xFFFFFFFF); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(0xFFFFFFFF)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2960,7 +2880,7 @@ pub(crate) mod test { | |||||||
|             .policy_path(path, KeychainKind::External); |             .policy_path(path, KeychainKind::External); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 0xFFFFFFFF); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(0xFFFFFFFF)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -2979,7 +2899,7 @@ pub(crate) mod test { | |||||||
|             .policy_path(path, KeychainKind::External); |             .policy_path(path, KeychainKind::External); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!(psbt.unsigned_tx.input[0].sequence, 144); |         assert_eq!(psbt.unsigned_tx.input[0].sequence, Sequence(144)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -4832,7 +4752,7 @@ pub(crate) mod test { | |||||||
|         let (wallet, _, _) = get_funded_wallet(get_test_tr_repeated_key()); |         let (wallet, _, _) = get_funded_wallet(get_test_tr_repeated_key()); | ||||||
|         let addr = wallet.get_address(AddressIndex::New).unwrap(); |         let addr = wallet.get_address(AddressIndex::New).unwrap(); | ||||||
| 
 | 
 | ||||||
|         let path = vec![("rn4nre9c".to_string(), vec![0])] |         let path = vec![("e5mmg3xh".to_string(), vec![0])] | ||||||
|             .into_iter() |             .into_iter() | ||||||
|             .collect(); |             .collect(); | ||||||
| 
 | 
 | ||||||
| @ -4842,13 +4762,24 @@ pub(crate) mod test { | |||||||
|             .policy_path(path, KeychainKind::External); |             .policy_path(path, KeychainKind::External); | ||||||
|         let (psbt, _) = builder.finish().unwrap(); |         let (psbt, _) = builder.finish().unwrap(); | ||||||
| 
 | 
 | ||||||
|         assert_eq!( |         let mut input_key_origins = psbt.inputs[0] | ||||||
|             psbt.inputs[0] |  | ||||||
|             .tap_key_origins |             .tap_key_origins | ||||||
|             .clone() |             .clone() | ||||||
|             .into_iter() |             .into_iter() | ||||||
|                 .collect::<Vec<_>>(), |             .collect::<Vec<_>>(); | ||||||
|             vec![( |         input_key_origins.sort(); | ||||||
|  | 
 | ||||||
|  |         assert_eq!( | ||||||
|  |             input_key_origins, | ||||||
|  |             vec![ | ||||||
|  |                 ( | ||||||
|  |                     from_str!("b511bd5771e47ee27558b1765e87b541668304ec567721c7b880edc0a010da55"), | ||||||
|  |                     ( | ||||||
|  |                         vec![], | ||||||
|  |                         (FromStr::from_str("871fd295").unwrap(), vec![].into()) | ||||||
|  |                     ) | ||||||
|  |                 ), | ||||||
|  |                 ( | ||||||
|                     from_str!("2b0558078bec38694a84933d659303e2575dae7e91685911454115bfd64487e3"), |                     from_str!("2b0558078bec38694a84933d659303e2575dae7e91685911454115bfd64487e3"), | ||||||
|                     ( |                     ( | ||||||
|                         vec![ |                         vec![ | ||||||
| @ -4857,33 +4788,24 @@ pub(crate) mod test { | |||||||
|                             ), |                             ), | ||||||
|                             from_str!( |                             from_str!( | ||||||
|                                 "f6e927ad4492c051fe325894a4f5f14538333b55a35f099876be42009ec8f903" |                                 "f6e927ad4492c051fe325894a4f5f14538333b55a35f099876be42009ec8f903" | ||||||
|  |                             ), | ||||||
|  |                         ], | ||||||
|  |                         (FromStr::from_str("ece52657").unwrap(), vec![].into()) | ||||||
|  |                     ) | ||||||
|                 ) |                 ) | ||||||
|             ], |             ], | ||||||
|                     (Default::default(), Default::default()) |  | ||||||
|                 ) |  | ||||||
|             )], |  | ||||||
|             "Wrong input tap_key_origins" |             "Wrong input tap_key_origins" | ||||||
|         ); |         ); | ||||||
|         assert_eq!( | 
 | ||||||
|             psbt.outputs[0] |         let mut output_key_origins = psbt.outputs[0] | ||||||
|             .tap_key_origins |             .tap_key_origins | ||||||
|             .clone() |             .clone() | ||||||
|             .into_iter() |             .into_iter() | ||||||
|                 .collect::<Vec<_>>(), |             .collect::<Vec<_>>(); | ||||||
|             vec![( |         output_key_origins.sort(); | ||||||
|                 from_str!("2b0558078bec38694a84933d659303e2575dae7e91685911454115bfd64487e3"), | 
 | ||||||
|                 ( |         assert_eq!( | ||||||
|                     vec![ |             input_key_origins, output_key_origins, | ||||||
|                         from_str!( |  | ||||||
|                             "858ad7a7d7f270e2c490c4d6ba00c499e46b18fdd59ea3c2c47d20347110271e" |  | ||||||
|                         ), |  | ||||||
|                         from_str!( |  | ||||||
|                             "f6e927ad4492c051fe325894a4f5f14538333b55a35f099876be42009ec8f903" |  | ||||||
|                         ) |  | ||||||
|                     ], |  | ||||||
|                     (Default::default(), Default::default()) |  | ||||||
|                 ) |  | ||||||
|             )], |  | ||||||
|             "Wrong output tap_key_origins" |             "Wrong output tap_key_origins" | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| @ -5135,7 +5057,7 @@ pub(crate) mod test { | |||||||
|     #[test] |     #[test] | ||||||
|     fn test_taproot_script_spend_sign_include_some_leaves() { |     fn test_taproot_script_spend_sign_include_some_leaves() { | ||||||
|         use crate::signer::TapLeavesOptions; |         use crate::signer::TapLeavesOptions; | ||||||
|         use crate::wallet::taproot::TapLeafHash; |         use bitcoin::util::taproot::TapLeafHash; | ||||||
| 
 | 
 | ||||||
|         let (wallet, _, _) = get_funded_wallet(get_test_tr_with_taptree_both_priv()); |         let (wallet, _, _) = get_funded_wallet(get_test_tr_with_taptree_both_priv()); | ||||||
|         let addr = wallet.get_address(AddressIndex::New).unwrap(); |         let addr = wallet.get_address(AddressIndex::New).unwrap(); | ||||||
| @ -5177,7 +5099,7 @@ pub(crate) mod test { | |||||||
|     #[test] |     #[test] | ||||||
|     fn test_taproot_script_spend_sign_exclude_some_leaves() { |     fn test_taproot_script_spend_sign_exclude_some_leaves() { | ||||||
|         use crate::signer::TapLeavesOptions; |         use crate::signer::TapLeavesOptions; | ||||||
|         use crate::wallet::taproot::TapLeafHash; |         use bitcoin::util::taproot::TapLeafHash; | ||||||
| 
 | 
 | ||||||
|         let (wallet, _, _) = get_funded_wallet(get_test_tr_with_taptree_both_priv()); |         let (wallet, _, _) = get_funded_wallet(get_test_tr_with_taptree_both_priv()); | ||||||
|         let addr = wallet.get_address(AddressIndex::New).unwrap(); |         let addr = wallet.get_address(AddressIndex::New).unwrap(); | ||||||
|  | |||||||
| @ -96,10 +96,10 @@ use bitcoin::{secp256k1, XOnlyPublicKey}; | |||||||
| use bitcoin::{EcdsaSighashType, PrivateKey, PublicKey, SchnorrSighashType, Script}; | use bitcoin::{EcdsaSighashType, PrivateKey, PublicKey, SchnorrSighashType, Script}; | ||||||
| 
 | 
 | ||||||
| use miniscript::descriptor::{ | use miniscript::descriptor::{ | ||||||
|     Descriptor, DescriptorPublicKey, DescriptorSecretKey, DescriptorSinglePriv, DescriptorXKey, |     Descriptor, DescriptorPublicKey, DescriptorSecretKey, DescriptorXKey, KeyMap, SinglePriv, | ||||||
|     KeyMap, SinglePubKey, |     SinglePubKey, | ||||||
| }; | }; | ||||||
| use miniscript::{Legacy, MiniscriptKey, Segwitv0, Tap}; | use miniscript::{Legacy, Segwitv0, SigType, Tap, ToPublicKey}; | ||||||
| 
 | 
 | ||||||
| use super::utils::SecpCtx; | use super::utils::SecpCtx; | ||||||
| use crate::descriptor::{DescriptorMeta, XKeyUtils}; | use crate::descriptor::{DescriptorMeta, XKeyUtils}; | ||||||
| @ -369,11 +369,11 @@ impl InputSigner for SignerWrapper<DescriptorXKey<ExtendedPrivKey>> { | |||||||
| 
 | 
 | ||||||
| impl SignerCommon for SignerWrapper<PrivateKey> { | impl SignerCommon for SignerWrapper<PrivateKey> { | ||||||
|     fn id(&self, secp: &SecpCtx) -> SignerId { |     fn id(&self, secp: &SecpCtx) -> SignerId { | ||||||
|         SignerId::from(self.public_key(secp).to_pubkeyhash()) |         SignerId::from(self.public_key(secp).to_pubkeyhash(SigType::Ecdsa)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn descriptor_secret_key(&self) -> Option<DescriptorSecretKey> { |     fn descriptor_secret_key(&self) -> Option<DescriptorSecretKey> { | ||||||
|         Some(DescriptorSecretKey::SinglePriv(DescriptorSinglePriv { |         Some(DescriptorSecretKey::Single(SinglePriv { | ||||||
|             key: self.signer, |             key: self.signer, | ||||||
|             origin: None, |             origin: None, | ||||||
|         })) |         })) | ||||||
| @ -517,13 +517,13 @@ fn sign_psbt_schnorr( | |||||||
|     let keypair = match leaf_hash { |     let keypair = match leaf_hash { | ||||||
|         None => keypair |         None => keypair | ||||||
|             .tap_tweak(secp, psbt_input.tap_merkle_root) |             .tap_tweak(secp, psbt_input.tap_merkle_root) | ||||||
|             .into_inner(), |             .to_inner(), | ||||||
|         Some(_) => keypair, // no tweak for script spend
 |         Some(_) => keypair, // no tweak for script spend
 | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     let msg = &Message::from_slice(&hash.into_inner()[..]).unwrap(); |     let msg = &Message::from_slice(&hash.into_inner()[..]).unwrap(); | ||||||
|     let sig = secp.sign_schnorr(msg, &keypair); |     let sig = secp.sign_schnorr(msg, &keypair); | ||||||
|     secp.verify_schnorr(&sig, msg, &XOnlyPublicKey::from_keypair(&keypair)) |     secp.verify_schnorr(&sig, msg, &XOnlyPublicKey::from_keypair(&keypair).0) | ||||||
|         .expect("invalid or corrupted schnorr signature"); |         .expect("invalid or corrupted schnorr signature"); | ||||||
| 
 | 
 | ||||||
|     let final_signature = schnorr::SchnorrSig { sig, hash_ty }; |     let final_signature = schnorr::SchnorrSig { sig, hash_ty }; | ||||||
| @ -576,7 +576,7 @@ impl SignersContainer { | |||||||
|         self.0 |         self.0 | ||||||
|             .values() |             .values() | ||||||
|             .filter_map(|signer| signer.descriptor_secret_key()) |             .filter_map(|signer| signer.descriptor_secret_key()) | ||||||
|             .filter_map(|secret| secret.as_public(secp).ok().map(|public| (public, secret))) |             .filter_map(|secret| secret.to_public(secp).ok().map(|public| (public, secret))) | ||||||
|             .collect() |             .collect() | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -601,8 +601,13 @@ impl SignersContainer { | |||||||
|             }; |             }; | ||||||
| 
 | 
 | ||||||
|             match secret { |             match secret { | ||||||
|                 DescriptorSecretKey::SinglePriv(private_key) => container.add_external( |                 DescriptorSecretKey::Single(private_key) => container.add_external( | ||||||
|                     SignerId::from(private_key.key.public_key(secp).to_pubkeyhash()), |                     SignerId::from( | ||||||
|  |                         private_key | ||||||
|  |                             .key | ||||||
|  |                             .public_key(secp) | ||||||
|  |                             .to_pubkeyhash(SigType::Ecdsa), | ||||||
|  |                     ), | ||||||
|                     SignerOrdering::default(), |                     SignerOrdering::default(), | ||||||
|                     Arc::new(SignerWrapper::new(private_key.key, ctx)), |                     Arc::new(SignerWrapper::new(private_key.key, ctx)), | ||||||
|                 ), |                 ), | ||||||
| @ -732,7 +737,7 @@ pub struct SignOptions { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Customize which taproot script-path leaves the signer should sign.
 | /// Customize which taproot script-path leaves the signer should sign.
 | ||||||
| #[derive(Debug, Clone, PartialEq)] | #[derive(Debug, Clone, PartialEq, Eq)] | ||||||
| pub enum TapLeavesOptions { | pub enum TapLeavesOptions { | ||||||
|     /// The signer will sign all the leaves it has a key for.
 |     /// The signer will sign all the leaves it has a key for.
 | ||||||
|     All, |     All, | ||||||
|  | |||||||
| @ -42,9 +42,7 @@ use std::default::Default; | |||||||
| use std::marker::PhantomData; | use std::marker::PhantomData; | ||||||
| 
 | 
 | ||||||
| use bitcoin::util::psbt::{self, PartiallySignedTransaction as Psbt}; | use bitcoin::util::psbt::{self, PartiallySignedTransaction as Psbt}; | ||||||
| use bitcoin::{OutPoint, Script, Transaction}; | use bitcoin::{LockTime, OutPoint, Script, Sequence, Transaction}; | ||||||
| 
 |  | ||||||
| use miniscript::descriptor::DescriptorTrait; |  | ||||||
| 
 | 
 | ||||||
| use super::coin_selection::{CoinSelectionAlgorithm, DefaultCoinSelectionAlgorithm}; | use super::coin_selection::{CoinSelectionAlgorithm, DefaultCoinSelectionAlgorithm}; | ||||||
| use crate::{database::BatchDatabase, Error, Utxo, Wallet}; | use crate::{database::BatchDatabase, Error, Utxo, Wallet}; | ||||||
| @ -139,7 +137,7 @@ pub(crate) struct TxParams { | |||||||
|     pub(crate) manually_selected_only: bool, |     pub(crate) manually_selected_only: bool, | ||||||
|     pub(crate) sighash: Option<psbt::PsbtSighashType>, |     pub(crate) sighash: Option<psbt::PsbtSighashType>, | ||||||
|     pub(crate) ordering: TxOrdering, |     pub(crate) ordering: TxOrdering, | ||||||
|     pub(crate) locktime: Option<u32>, |     pub(crate) locktime: Option<LockTime>, | ||||||
|     pub(crate) rbf: Option<RbfValue>, |     pub(crate) rbf: Option<RbfValue>, | ||||||
|     pub(crate) version: Option<Version>, |     pub(crate) version: Option<Version>, | ||||||
|     pub(crate) change_policy: ChangeSpendPolicy, |     pub(crate) change_policy: ChangeSpendPolicy, | ||||||
| @ -147,7 +145,7 @@ pub(crate) struct TxParams { | |||||||
|     pub(crate) add_global_xpubs: bool, |     pub(crate) add_global_xpubs: bool, | ||||||
|     pub(crate) include_output_redeem_witness_script: bool, |     pub(crate) include_output_redeem_witness_script: bool, | ||||||
|     pub(crate) bumping_fee: Option<PreviousFee>, |     pub(crate) bumping_fee: Option<PreviousFee>, | ||||||
|     pub(crate) current_height: Option<u32>, |     pub(crate) current_height: Option<LockTime>, | ||||||
|     pub(crate) allow_dust: bool, |     pub(crate) allow_dust: bool, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -426,7 +424,7 @@ impl<'a, D: BatchDatabase, Cs: CoinSelectionAlgorithm<D>, Ctx: TxBuilderContext> | |||||||
|     /// Use a specific nLockTime while creating the transaction
 |     /// Use a specific nLockTime while creating the transaction
 | ||||||
|     ///
 |     ///
 | ||||||
|     /// This can cause conflicts if the wallet's descriptors contain an "after" (OP_CLTV) operator.
 |     /// This can cause conflicts if the wallet's descriptors contain an "after" (OP_CLTV) operator.
 | ||||||
|     pub fn nlocktime(&mut self, locktime: u32) -> &mut Self { |     pub fn nlocktime(&mut self, locktime: LockTime) -> &mut Self { | ||||||
|         self.params.locktime = Some(locktime); |         self.params.locktime = Some(locktime); | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
| @ -541,7 +539,7 @@ impl<'a, D: BatchDatabase, Cs: CoinSelectionAlgorithm<D>, Ctx: TxBuilderContext> | |||||||
|     ///
 |     ///
 | ||||||
|     /// If the `nsequence` is higher than `0xFFFFFFFD` an error will be thrown, since it would not
 |     /// If the `nsequence` is higher than `0xFFFFFFFD` an error will be thrown, since it would not
 | ||||||
|     /// be a valid nSequence to signal RBF.
 |     /// be a valid nSequence to signal RBF.
 | ||||||
|     pub fn enable_rbf_with_sequence(&mut self, nsequence: u32) -> &mut Self { |     pub fn enable_rbf_with_sequence(&mut self, nsequence: Sequence) -> &mut Self { | ||||||
|         self.params.rbf = Some(RbfValue::Value(nsequence)); |         self.params.rbf = Some(RbfValue::Value(nsequence)); | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
| @ -558,7 +556,7 @@ impl<'a, D: BatchDatabase, Cs: CoinSelectionAlgorithm<D>, Ctx: TxBuilderContext> | |||||||
|     ///
 |     ///
 | ||||||
|     /// In both cases, if you don't provide a current height, we use the last sync height.
 |     /// In both cases, if you don't provide a current height, we use the last sync height.
 | ||||||
|     pub fn current_height(&mut self, height: u32) -> &mut Self { |     pub fn current_height(&mut self, height: u32) -> &mut Self { | ||||||
|         self.params.current_height = Some(height); |         self.params.current_height = Some(LockTime::from_height(height).expect("Invalid height")); | ||||||
|         self |         self | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -703,7 +701,7 @@ impl TxOrdering { | |||||||
|                 #[cfg(not(test))] |                 #[cfg(not(test))] | ||||||
|                 let mut rng = rand::thread_rng(); |                 let mut rng = rand::thread_rng(); | ||||||
|                 #[cfg(test)] |                 #[cfg(test)] | ||||||
|                 let mut rng = rand::rngs::StdRng::seed_from_u64(0); |                 let mut rng = rand::rngs::StdRng::seed_from_u64(12345); | ||||||
| 
 | 
 | ||||||
|                 tx.output.shuffle(&mut rng); |                 tx.output.shuffle(&mut rng); | ||||||
|             } |             } | ||||||
| @ -736,13 +734,13 @@ impl Default for Version { | |||||||
| #[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] | #[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Hash, Clone, Copy)] | ||||||
| pub(crate) enum RbfValue { | pub(crate) enum RbfValue { | ||||||
|     Default, |     Default, | ||||||
|     Value(u32), |     Value(Sequence), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl RbfValue { | impl RbfValue { | ||||||
|     pub(crate) fn get_value(&self) -> u32 { |     pub(crate) fn get_value(&self) -> Sequence { | ||||||
|         match self { |         match self { | ||||||
|             RbfValue::Default => 0xFFFFFFFD, |             RbfValue::Default => Sequence::ENABLE_RBF_NO_LOCKTIME, | ||||||
|             RbfValue::Value(v) => *v, |             RbfValue::Value(v) => *v, | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -858,10 +856,12 @@ mod test { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn get_test_utxos() -> Vec<LocalUtxo> { |     fn get_test_utxos() -> Vec<LocalUtxo> { | ||||||
|  |         use bitcoin::hashes::Hash; | ||||||
|  | 
 | ||||||
|         vec![ |         vec![ | ||||||
|             LocalUtxo { |             LocalUtxo { | ||||||
|                 outpoint: OutPoint { |                 outpoint: OutPoint { | ||||||
|                     txid: Default::default(), |                     txid: bitcoin::Txid::from_inner([0; 32]), | ||||||
|                     vout: 0, |                     vout: 0, | ||||||
|                 }, |                 }, | ||||||
|                 txout: Default::default(), |                 txout: Default::default(), | ||||||
| @ -870,7 +870,7 @@ mod test { | |||||||
|             }, |             }, | ||||||
|             LocalUtxo { |             LocalUtxo { | ||||||
|                 outpoint: OutPoint { |                 outpoint: OutPoint { | ||||||
|                     txid: Default::default(), |                     txid: bitcoin::Txid::from_inner([0; 32]), | ||||||
|                     vout: 1, |                     vout: 1, | ||||||
|                 }, |                 }, | ||||||
|                 txout: Default::default(), |                 txout: Default::default(), | ||||||
|  | |||||||
| @ -9,23 +9,11 @@ | |||||||
| // You may not use this file except in accordance with one or both of these
 | // You may not use this file except in accordance with one or both of these
 | ||||||
| // licenses.
 | // licenses.
 | ||||||
| 
 | 
 | ||||||
| use bitcoin::blockdata::script::Script; |  | ||||||
| use bitcoin::secp256k1::{All, Secp256k1}; | use bitcoin::secp256k1::{All, Secp256k1}; | ||||||
|  | use bitcoin::{LockTime, Script, Sequence}; | ||||||
| 
 | 
 | ||||||
| use miniscript::{MiniscriptKey, Satisfier, ToPublicKey}; | use miniscript::{MiniscriptKey, Satisfier, ToPublicKey}; | ||||||
| 
 | 
 | ||||||
| // MSB of the nSequence. If set there's no consensus-constraint, so it must be disabled when
 |  | ||||||
| // spending using CSV in order to enforce CSV rules
 |  | ||||||
| pub(crate) const SEQUENCE_LOCKTIME_DISABLE_FLAG: u32 = 1 << 31; |  | ||||||
| // When nSequence is lower than this flag the timelock is interpreted as block-height-based,
 |  | ||||||
| // otherwise it's time-based
 |  | ||||||
| pub(crate) const SEQUENCE_LOCKTIME_TYPE_FLAG: u32 = 1 << 22; |  | ||||||
| // Mask for the bits used to express the timelock
 |  | ||||||
| pub(crate) const SEQUENCE_LOCKTIME_MASK: u32 = 0x0000FFFF; |  | ||||||
| 
 |  | ||||||
| // Threshold for nLockTime to be considered a block-height-based timelock rather than time-based
 |  | ||||||
| pub(crate) const BLOCKS_TIMELOCK_THRESHOLD: u32 = 500000000; |  | ||||||
| 
 |  | ||||||
| /// Trait to check if a value is below the dust limit.
 | /// Trait to check if a value is below the dust limit.
 | ||||||
| /// We are performing dust value calculation for a given script public key using rust-bitcoin to
 | /// We are performing dust value calculation for a given script public key using rust-bitcoin to
 | ||||||
| /// keep it compatible with network dust rate
 | /// keep it compatible with network dust rate
 | ||||||
| @ -38,7 +26,7 @@ pub trait IsDust { | |||||||
| 
 | 
 | ||||||
| impl IsDust for u64 { | impl IsDust for u64 { | ||||||
|     fn is_dust(&self, script: &Script) -> bool { |     fn is_dust(&self, script: &Script) -> bool { | ||||||
|         *self < script.dust_value().as_sat() |         *self < script.dust_value().to_sat() | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -56,19 +44,15 @@ impl After { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub(crate) fn check_nsequence_rbf(rbf: u32, csv: u32) -> bool { | pub(crate) fn check_nsequence_rbf(rbf: Sequence, csv: Sequence) -> bool { | ||||||
|     // This flag cannot be set in the nSequence when spending using OP_CSV
 |     // The RBF value must enable relative timelocks
 | ||||||
|     if rbf & SEQUENCE_LOCKTIME_DISABLE_FLAG != 0 { |     if !rbf.is_relative_lock_time() { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let mask = SEQUENCE_LOCKTIME_TYPE_FLAG | SEQUENCE_LOCKTIME_MASK; |  | ||||||
|     let rbf = rbf & mask; |  | ||||||
|     let csv = csv & mask; |  | ||||||
| 
 |  | ||||||
|     // Both values should be represented in the same unit (either time-based or
 |     // Both values should be represented in the same unit (either time-based or
 | ||||||
|     // block-height based)
 |     // block-height based)
 | ||||||
|     if (rbf < SEQUENCE_LOCKTIME_TYPE_FLAG) != (csv < SEQUENCE_LOCKTIME_TYPE_FLAG) { |     if rbf.is_time_locked() != csv.is_time_locked() { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -80,24 +64,10 @@ pub(crate) fn check_nsequence_rbf(rbf: u32, csv: u32) -> bool { | |||||||
|     true |     true | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| pub(crate) fn check_nlocktime(nlocktime: u32, required: u32) -> bool { |  | ||||||
|     // Both values should be expressed in the same unit
 |  | ||||||
|     if (nlocktime < BLOCKS_TIMELOCK_THRESHOLD) != (required < BLOCKS_TIMELOCK_THRESHOLD) { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // The value should be at least `required`
 |  | ||||||
|     if nlocktime < required { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     true |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<Pk: MiniscriptKey + ToPublicKey> Satisfier<Pk> for After { | impl<Pk: MiniscriptKey + ToPublicKey> Satisfier<Pk> for After { | ||||||
|     fn check_after(&self, n: u32) -> bool { |     fn check_after(&self, n: LockTime) -> bool { | ||||||
|         if let Some(current_height) = self.current_height { |         if let Some(current_height) = self.current_height { | ||||||
|             current_height >= n |             current_height >= n.to_consensus_u32() | ||||||
|         } else { |         } else { | ||||||
|             self.assume_height_reached |             self.assume_height_reached | ||||||
|         } |         } | ||||||
| @ -125,10 +95,15 @@ impl Older { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<Pk: MiniscriptKey + ToPublicKey> Satisfier<Pk> for Older { | impl<Pk: MiniscriptKey + ToPublicKey> Satisfier<Pk> for Older { | ||||||
|     fn check_older(&self, n: u32) -> bool { |     fn check_older(&self, n: Sequence) -> bool { | ||||||
|         if let Some(current_height) = self.current_height { |         if let Some(current_height) = self.current_height { | ||||||
|             // TODO: test >= / >
 |             // TODO: test >= / >
 | ||||||
|             current_height as u64 >= self.create_height.unwrap_or(0) as u64 + n as u64 |             current_height | ||||||
|  |                 >= self | ||||||
|  |                     .create_height | ||||||
|  |                     .unwrap_or(0) | ||||||
|  |                     .checked_add(n.to_consensus_u32()) | ||||||
|  |                     .expect("Overflowing addition") | ||||||
|         } else { |         } else { | ||||||
|             self.assume_height_reached |             self.assume_height_reached | ||||||
|         } |         } | ||||||
| @ -139,11 +114,12 @@ pub(crate) type SecpCtx = Secp256k1<All>; | |||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod test { | mod test { | ||||||
|     use super::{ |     // When nSequence is lower than this flag the timelock is interpreted as block-height-based,
 | ||||||
|         check_nlocktime, check_nsequence_rbf, IsDust, BLOCKS_TIMELOCK_THRESHOLD, |     // otherwise it's time-based
 | ||||||
|         SEQUENCE_LOCKTIME_TYPE_FLAG, |     pub(crate) const SEQUENCE_LOCKTIME_TYPE_FLAG: u32 = 1 << 22; | ||||||
|     }; | 
 | ||||||
|     use crate::bitcoin::Address; |     use super::{check_nsequence_rbf, IsDust}; | ||||||
|  |     use crate::bitcoin::{Address, Sequence}; | ||||||
|     use std::str::FromStr; |     use std::str::FromStr; | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
| @ -165,66 +141,40 @@ mod test { | |||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_msb_set() { |     fn test_check_nsequence_rbf_msb_set() { | ||||||
|         let result = check_nsequence_rbf(0x80000000, 5000); |         let result = check_nsequence_rbf(Sequence(0x80000000), Sequence(5000)); | ||||||
|         assert!(!result); |         assert!(!result); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_lt_csv() { |     fn test_check_nsequence_rbf_lt_csv() { | ||||||
|         let result = check_nsequence_rbf(4000, 5000); |         let result = check_nsequence_rbf(Sequence(4000), Sequence(5000)); | ||||||
|         assert!(!result); |         assert!(!result); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_different_unit() { |     fn test_check_nsequence_rbf_different_unit() { | ||||||
|         let result = check_nsequence_rbf(SEQUENCE_LOCKTIME_TYPE_FLAG + 5000, 5000); |         let result = | ||||||
|  |             check_nsequence_rbf(Sequence(SEQUENCE_LOCKTIME_TYPE_FLAG + 5000), Sequence(5000)); | ||||||
|         assert!(!result); |         assert!(!result); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_mask() { |     fn test_check_nsequence_rbf_mask() { | ||||||
|         let result = check_nsequence_rbf(0x3f + 10_000, 5000); |         let result = check_nsequence_rbf(Sequence(0x3f + 10_000), Sequence(5000)); | ||||||
|         assert!(result); |         assert!(result); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_same_unit_blocks() { |     fn test_check_nsequence_rbf_same_unit_blocks() { | ||||||
|         let result = check_nsequence_rbf(10_000, 5000); |         let result = check_nsequence_rbf(Sequence(10_000), Sequence(5000)); | ||||||
|         assert!(result); |         assert!(result); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[test] |     #[test] | ||||||
|     fn test_check_nsequence_rbf_same_unit_time() { |     fn test_check_nsequence_rbf_same_unit_time() { | ||||||
|         let result = check_nsequence_rbf( |         let result = check_nsequence_rbf( | ||||||
|             SEQUENCE_LOCKTIME_TYPE_FLAG + 10_000, |             Sequence(SEQUENCE_LOCKTIME_TYPE_FLAG + 10_000), | ||||||
|             SEQUENCE_LOCKTIME_TYPE_FLAG + 5000, |             Sequence(SEQUENCE_LOCKTIME_TYPE_FLAG + 5000), | ||||||
|         ); |  | ||||||
|         assert!(result); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     fn test_check_nlocktime_lt_cltv() { |  | ||||||
|         let result = check_nlocktime(4000, 5000); |  | ||||||
|         assert!(!result); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     fn test_check_nlocktime_different_unit() { |  | ||||||
|         let result = check_nlocktime(BLOCKS_TIMELOCK_THRESHOLD + 5000, 5000); |  | ||||||
|         assert!(!result); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     fn test_check_nlocktime_same_unit_blocks() { |  | ||||||
|         let result = check_nlocktime(10_000, 5000); |  | ||||||
|         assert!(result); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     #[test] |  | ||||||
|     fn test_check_nlocktime_same_unit_time() { |  | ||||||
|         let result = check_nlocktime( |  | ||||||
|             BLOCKS_TIMELOCK_THRESHOLD + 10_000, |  | ||||||
|             BLOCKS_TIMELOCK_THRESHOLD + 5000, |  | ||||||
|         ); |         ); | ||||||
|         assert!(result); |         assert!(result); | ||||||
|     } |     } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user