forked from bitcoindevkit/bdk
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(example): add RPC wallet example
- Loading branch information
1 parent
bc8dfe6
commit 8d31b33
Showing
5 changed files
with
189 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
# Example RPC CLI | ||
|
||
### Simple Regtest Test | ||
|
||
1. Start local regtest bitcoind. | ||
``` | ||
mkdir -p /tmp/regtest/bitcoind | ||
bitcoind -regtest -server -fallbackfee=0.0002 -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> -datadir=/tmp/regtest/bitcoind -daemon | ||
``` | ||
2. Create a test bitcoind wallet and set bitcoind env. | ||
``` | ||
bitcoin-cli -datadir=/tmp/regtest/bitcoind -regtest -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> -named createwallet wallet_name="test" | ||
export RPC_URL=127.0.0.1:18443 | ||
export RPC_USER=<your-rpc-username> | ||
export RPC_PASS=<your-rpc-password> | ||
``` | ||
3. Get test bitcoind wallet info. | ||
``` | ||
bitcoin-cli -rpcwallet="test" -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> -datadir=/tmp/regtest/bitcoind -regtest getwalletinfo | ||
``` | ||
4. Get new test bitcoind wallet address. | ||
``` | ||
BITCOIND_ADDRESS=$(bitcoin-cli -rpcwallet="test" -datadir=/tmp/regtest/bitcoind -regtest -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> getnewaddress) | ||
echo $BITCOIND_ADDRESS | ||
``` | ||
5. Generate 101 blocks with reward to test bitcoind wallet address. | ||
``` | ||
bitcoin-cli -datadir=/tmp/regtest/bitcoind -regtest -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> generatetoaddress 101 $BITCOIND_ADDRESS | ||
``` | ||
6. Verify test bitcoind wallet balance. | ||
``` | ||
bitcoin-cli -rpcwallet="test" -datadir=/tmp/regtest/bitcoind -regtest -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> getbalances | ||
``` | ||
7. Set descriptor env and get address from RPC CLI wallet. | ||
``` | ||
export DESCRIPTOR="wpkh(tprv8ZgxMBicQKsPfK9BTf82oQkHhawtZv19CorqQKPFeaHDMA4dXYX6eWsJGNJ7VTQXWmoHdrfjCYuDijcRmNFwSKcVhswzqs4fugE8turndGc/1/*)" | ||
cargo run -- --network regtest address next | ||
``` | ||
8. Send 5 test bitcoin to RPC CLI wallet. | ||
``` | ||
bitcoin-cli -rpcwallet="test" -datadir=/tmp/regtest/bitcoind -regtest -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> sendtoaddress <address> 5 | ||
``` | ||
9. Sync blockchain with RPC CLI wallet. | ||
``` | ||
cargo run -- --network regtest sync | ||
<CNTRL-C to stop syncing> | ||
``` | ||
10. Get RPC CLI wallet unconfirmed balances. | ||
``` | ||
cargo run -- --network regtest balance | ||
``` | ||
11. Generate 1 block with reward to test bitcoind wallet address. | ||
``` | ||
bitcoin-cli -datadir=/tmp/regtest/bitcoind -rpcuser=<your-rpc-username> -rpcpassword=<your-rpc-password> -regtest generatetoaddress 10 $BITCOIND_ADDRESS | ||
``` | ||
12. Sync the blockchain with RPC CLI wallet. | ||
``` | ||
cargo run -- --network regtest sync | ||
<CNTRL-C to stop syncing> | ||
``` | ||
13. Get RPC CLI wallet confirmed balances. | ||
``` | ||
cargo run -- --network regtest balance | ||
``` | ||
14. Get RPC CLI wallet transactions. | ||
``` | ||
cargo run -- --network regtest txout list | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
[package] | ||
name = "wallet_rpc" | ||
version = "0.1.0" | ||
edition = "2021" | ||
|
||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||
|
||
[dependencies] | ||
bdk = { path = "../../crates/bdk" } | ||
bdk_file_store = { path = "../../crates/file_store" } | ||
bdk_bitcoind_rpc = { path = "../../crates/bitcoind_rpc" } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
# Wallet RPC Example | ||
|
||
# To run the wallet example, execute the following code (replace arguments with values that match your setup) | ||
|
||
``` | ||
cargo run -- <RPC_URL> <RPC_USER> <RPC_PASS> <LOOKAHEAD> <FALLBACK_HEIGHT> | ||
``` | ||
|
||
Here is the command we used during testing | ||
|
||
``` | ||
cargo run -- 127.0.0.1:18332 bitcoin password 20 2532323 | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,96 @@ | ||
use bdk::{ | ||
bitcoin::{Address, Network}, | ||
wallet::{AddressIndex, Wallet}, | ||
SignOptions, | ||
}; | ||
use bdk_bitcoind_rpc::{ | ||
bitcoincore_rpc::{Auth, Client, RpcApi}, | ||
Emitter, | ||
}; | ||
use bdk_file_store::Store; | ||
use std::str::FromStr; | ||
|
||
const DB_MAGIC: &str = "bdk-rpc-wallet-example"; | ||
const SEND_AMOUNT: u64 = 5000; | ||
|
||
fn main() -> Result<(), Box<dyn std::error::Error>> { | ||
let args = std::env::args().collect::<Vec<_>>(); | ||
let db_path = std::env::temp_dir().join("bdk-rpc-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/*)"; | ||
|
||
if args.len() < 6 { | ||
println!("Usage: wallet_rpc <RPC_URL> <RPC_USER> <RPC_PASS> <LOOKAHEAD> <FALLBACK_HEIGHT>"); | ||
std::process::exit(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()); | ||
|
||
let rpc_client = Client::new(&args[1], Auth::UserPass(args[2].clone(), args[3].clone()))?; | ||
|
||
println!( | ||
"Connected to Bitcoin Core RPC at {:?}", | ||
rpc_client.get_blockchain_info().unwrap() | ||
); | ||
|
||
wallet.set_lookahead_for_all(args[4].parse::<u32>()?)?; | ||
|
||
let chain_tip = wallet.latest_checkpoint(); | ||
let mut emitter = match chain_tip { | ||
Some(cp) => Emitter::from_checkpoint(&rpc_client, cp), | ||
None => Emitter::from_height(&rpc_client, args[5].parse::<u32>()?), | ||
}; | ||
|
||
while let Some((height, block)) = emitter.next_block()? { | ||
println!("Applying block {} at height {}", block.block_hash(), height); | ||
wallet.apply_block_relevant(block, height)?; | ||
wallet.commit()?; | ||
} | ||
|
||
let unconfirmed_txs = emitter.mempool()?; | ||
println!("Applying unconfirmed transactions: ..."); | ||
wallet.batch_insert_relevant_unconfirmed(unconfirmed_txs.iter().map(|(tx, time)| (tx, *time))); | ||
wallet.commit()?; | ||
|
||
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(1); | ||
} | ||
|
||
let faucet_address = Address::from_str("tb1qw2c3lxufxqe2x9s4rdzh65tpf4d7fssjgh8nv6")? | ||
.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(); | ||
rpc_client.send_raw_transaction(&tx)?; | ||
println!("Tx broadcasted! Txid: {}", tx.txid()); | ||
|
||
Ok(()) | ||
} |