1ff806c67f4da9ba58b7c7689fde0fe41a34a6f5 fix(chain)!: rm weird `From` impl (志宇) d43ae0231fa4670b98780cad84466c14ae087292 refactor: improve docs, cleanup unnecessary types and improve code (Vladimir Fomene) 41042069809e3eeb4a8cc8a5a8db1af2c57c4a11 feat: impl Append for lots of tuples (LLFourn) c56728ff1315e0deaf256af07fd1ff5e18fced8a refactor: Remove `scan` and `scan_txout` from SpkTxoutIndex and KeychainTxoutIndex (Vladimir Fomene) 32c40ac939bc514ac7f1d3f1d7cd1080011e20ba feat(electrum)!: change signature of `ElectrumExt` (志宇) a28748c33976312b9e6671636ab7e305323efb03 refactor: Implement Default for WalletUpdate (Vladimir Fomene) f42f8b8ff19c2e67888b476487e4e5c9edb0d0ff refactor: Allow for no chain update (Vladimir Fomene) 68572bfd2e32efdeefaa46618e8e248d3a87f143 refactor: move WalletChangeset to wallet module (Vladimir Fomene) 2392e50fd9793902d480556caa4ec225085c82d6 refactor: Move WalletUpdate to wallet module (Vladimir Fomene) 7c12dc994242bf2d7e35c2723f6e7000de97a388 refactor: Remove ForEachTxout trait (Vladimir Fomene) 6bcbb93233824ec391689191b8ca1f5459cec930 refactor: Edit ElectrumExt not to use WalletUpdate (Vladimir Fomene) Pull request description: ### Description Fixes #1061 ### Changelog notice - Move WalletUpdate to the wallet module - Remove ForEachTxout trait completely - Refactor ElectrumExt to not use WalletUpdate. ### 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: evanlinjin: ACK 1ff806c67f4da9ba58b7c7689fde0fe41a34a6f5 Tree-SHA512: 05349713af9d2efa14a522ceaabb7513bb437d786adf2f93055765589a67e4eb68bda36ff415aeba07816c4d30988d4d55bac018e7697019270a219105ed65a2
99 lines
3.3 KiB
Rust
99 lines
3.3 KiB
Rust
const DB_MAGIC: &str = "bdk_wallet_esplora_example";
|
|
const SEND_AMOUNT: u64 = 1000;
|
|
const STOP_GAP: usize = 5;
|
|
const PARALLEL_REQUESTS: usize = 1;
|
|
|
|
use std::{io::Write, str::FromStr};
|
|
|
|
use bdk::{
|
|
bitcoin::{Address, Network},
|
|
wallet::{AddressIndex, Update},
|
|
SignOptions, Wallet,
|
|
};
|
|
use bdk_esplora::{esplora_client, EsploraExt};
|
|
use bdk_file_store::Store;
|
|
|
|
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
|
let db_path = std::env::temp_dir().join("bdk-esplora-example");
|
|
let db = Store::<bdk::wallet::ChangeSet>::new_from_path(DB_MAGIC.as_bytes(), db_path)?;
|
|
let external_descriptor = "wpkh(tprv8ZgxMBicQKsPdy6LMhUtFHAgpocR8GC6QmwMSFpZs7h6Eziw3SpThFfczTDh5rW2krkqffa11UpX3XkeTTB2FvzZKWXqPY54Y6Rq4AQ5R8L/84'/1'/0'/0/*)";
|
|
let internal_descriptor = "wpkh(tprv8ZgxMBicQKsPdy6LMhUtFHAgpocR8GC6QmwMSFpZs7h6Eziw3SpThFfczTDh5rW2krkqffa11UpX3XkeTTB2FvzZKWXqPY54Y6Rq4AQ5R8L/84'/1'/0'/1/*)";
|
|
|
|
let mut wallet = Wallet::new(
|
|
external_descriptor,
|
|
Some(internal_descriptor),
|
|
db,
|
|
Network::Testnet,
|
|
)?;
|
|
|
|
let address = wallet.get_address(AddressIndex::New);
|
|
println!("Generated Address: {}", address);
|
|
|
|
let balance = wallet.get_balance();
|
|
println!("Wallet balance before syncing: {} sats", balance.total());
|
|
|
|
print!("Syncing...");
|
|
let client =
|
|
esplora_client::Builder::new("https://blockstream.info/testnet/api").build_blocking()?;
|
|
|
|
let prev_tip = wallet.latest_checkpoint();
|
|
let keychain_spks = wallet
|
|
.spks_of_all_keychains()
|
|
.into_iter()
|
|
.map(|(k, k_spks)| {
|
|
let mut once = Some(());
|
|
let mut stdout = std::io::stdout();
|
|
let k_spks = k_spks
|
|
.inspect(move |(spk_i, _)| match once.take() {
|
|
Some(_) => print!("\nScanning keychain [{:?}]", k),
|
|
None => print!(" {:<3}", spk_i),
|
|
})
|
|
.inspect(move |_| stdout.flush().expect("must flush"));
|
|
(k, k_spks)
|
|
})
|
|
.collect();
|
|
|
|
let (update_graph, last_active_indices) =
|
|
client.scan_txs_with_keychains(keychain_spks, None, None, STOP_GAP, PARALLEL_REQUESTS)?;
|
|
let missing_heights = wallet.tx_graph().missing_heights(wallet.local_chain());
|
|
let chain_update = client.update_local_chain(prev_tip, missing_heights)?;
|
|
let update = Update {
|
|
last_active_indices,
|
|
graph: update_graph,
|
|
chain: Some(chain_update),
|
|
};
|
|
|
|
wallet.apply_update(update)?;
|
|
wallet.commit()?;
|
|
println!();
|
|
|
|
let balance = wallet.get_balance();
|
|
println!("Wallet balance after syncing: {} sats", balance.total());
|
|
|
|
if balance.total() < SEND_AMOUNT {
|
|
println!(
|
|
"Please send at least {} sats to the receiving address",
|
|
SEND_AMOUNT
|
|
);
|
|
std::process::exit(0);
|
|
}
|
|
|
|
let faucet_address = Address::from_str("mkHS9ne12qx9pS9VojpwU5xtRd4T7X7ZUt")?
|
|
.require_network(Network::Testnet)?;
|
|
|
|
let mut tx_builder = wallet.build_tx();
|
|
tx_builder
|
|
.add_recipient(faucet_address.script_pubkey(), SEND_AMOUNT)
|
|
.enable_rbf();
|
|
|
|
let mut psbt = tx_builder.finish()?;
|
|
let finalized = wallet.sign(&mut psbt, SignOptions::default())?;
|
|
assert!(finalized);
|
|
|
|
let tx = psbt.extract_tx();
|
|
client.broadcast(&tx)?;
|
|
println!("Tx broadcasted! Txid: {}", tx.txid());
|
|
|
|
Ok(())
|
|
}
|