2023-03-03 17:53:44 +01:00
|
|
|
use bdk::{
|
|
|
|
bitcoin::{Address, Network},
|
|
|
|
wallet::AddressIndex,
|
|
|
|
SignOptions, Wallet,
|
|
|
|
};
|
|
|
|
use bdk_esplora::esplora_client;
|
|
|
|
use bdk_esplora::EsploraExt;
|
|
|
|
use bdk_file_store::KeychainStore;
|
2023-03-08 15:28:56 +13:00
|
|
|
use std::{io::Write, str::FromStr};
|
2023-03-03 17:53:44 +01:00
|
|
|
|
2023-03-07 10:47:13 +13:00
|
|
|
const SEND_AMOUNT: u64 = 5000;
|
2023-03-03 17:53:44 +01:00
|
|
|
const STOP_GAP: usize = 50;
|
|
|
|
const PARALLEL_REQUESTS: usize = 5;
|
|
|
|
|
|
|
|
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
2023-03-07 10:59:13 +13:00
|
|
|
let db_path = std::env::temp_dir().join("bdk-esplora-example");
|
|
|
|
let db = KeychainStore::new_from_path(db_path)?;
|
2023-03-03 17:53:44 +01:00
|
|
|
let external_descriptor = "wpkh(tprv8ZgxMBicQKsPdy6LMhUtFHAgpocR8GC6QmwMSFpZs7h6Eziw3SpThFfczTDh5rW2krkqffa11UpX3XkeTTB2FvzZKWXqPY54Y6Rq4AQ5R8L/84'/0'/0'/0/*)";
|
|
|
|
let internal_descriptor = "wpkh(tprv8ZgxMBicQKsPdy6LMhUtFHAgpocR8GC6QmwMSFpZs7h6Eziw3SpThFfczTDh5rW2krkqffa11UpX3XkeTTB2FvzZKWXqPY54Y6Rq4AQ5R8L/84'/0'/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());
|
|
|
|
|
2023-03-08 15:28:56 +13:00
|
|
|
print!("Syncing...");
|
2023-03-03 17:53:44 +01:00
|
|
|
// Scanning the chain...
|
|
|
|
let esplora_url = "https://mempool.space/testnet/api";
|
|
|
|
let client = esplora_client::Builder::new(esplora_url).build_blocking()?;
|
|
|
|
let checkpoints = wallet.checkpoints();
|
2023-03-08 15:28:56 +13:00
|
|
|
let spks = wallet
|
|
|
|
.spks_of_all_keychains()
|
|
|
|
.into_iter()
|
|
|
|
.map(|(k, spks)| {
|
|
|
|
let mut first = true;
|
|
|
|
(
|
|
|
|
k,
|
|
|
|
spks.inspect(move |(spk_i, _)| {
|
|
|
|
if first {
|
|
|
|
first = false;
|
|
|
|
print!("\nScanning keychain [{:?}]:", k);
|
|
|
|
}
|
|
|
|
print!(" {}", spk_i);
|
|
|
|
let _ = std::io::stdout().flush();
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.collect();
|
2023-03-03 17:53:44 +01:00
|
|
|
let update = client.scan(
|
|
|
|
checkpoints,
|
|
|
|
spks,
|
|
|
|
core::iter::empty(),
|
|
|
|
core::iter::empty(),
|
|
|
|
STOP_GAP,
|
|
|
|
PARALLEL_REQUESTS,
|
|
|
|
)?;
|
2023-03-08 15:28:56 +13:00
|
|
|
println!();
|
2023-03-03 17:53:44 +01:00
|
|
|
wallet.apply_update(update)?;
|
|
|
|
wallet.commit()?;
|
|
|
|
|
|
|
|
let balance = wallet.get_balance();
|
|
|
|
println!("Wallet balance after syncing: {} sats", balance.total());
|
|
|
|
|
2023-03-08 15:20:36 +01:00
|
|
|
if balance.total() < SEND_AMOUNT {
|
|
|
|
println!(
|
|
|
|
"Please send at least {} sats to the receiving address",
|
|
|
|
SEND_AMOUNT
|
|
|
|
);
|
2023-03-03 17:53:44 +01:00
|
|
|
std::process::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
let faucet_address = Address::from_str("mkHS9ne12qx9pS9VojpwU5xtRd4T7X7ZUt")?;
|
|
|
|
|
|
|
|
let mut tx_builder = wallet.build_tx();
|
|
|
|
tx_builder
|
2023-03-07 10:47:13 +13:00
|
|
|
.add_recipient(faucet_address.script_pubkey(), SEND_AMOUNT)
|
2023-03-03 17:53:44 +01:00
|
|
|
.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(())
|
2023-02-21 16:23:08 +11:00
|
|
|
}
|