363 lines
11 KiB
Rust
363 lines
11 KiB
Rust
use bdk::bitcoin::hashes::hex::ToHex;
|
|
use bdk::bitcoin::secp256k1::Secp256k1;
|
|
use bdk::bitcoin::util::psbt::PartiallySignedTransaction;
|
|
use bdk::bitcoin::{Address, Network};
|
|
use bdk::blockchain::any::{AnyBlockchain, AnyBlockchainConfig};
|
|
use bdk::blockchain::Progress;
|
|
use bdk::blockchain::{
|
|
electrum::ElectrumBlockchainConfig, esplora::EsploraBlockchainConfig, ConfigurableBlockchain,
|
|
};
|
|
use bdk::database::any::{AnyDatabase, SledDbConfiguration, SqliteDbConfiguration};
|
|
use bdk::database::{AnyDatabaseConfig, ConfigurableDatabase};
|
|
use bdk::keys::bip39::{Language, Mnemonic, WordCount};
|
|
use bdk::keys::{DerivableKey, ExtendedKey, GeneratableKey, GeneratedKey};
|
|
use bdk::miniscript::BareCtx;
|
|
use bdk::wallet::AddressIndex;
|
|
use bdk::{BlockTime, Error, FeeRate, SignOptions, Wallet as BdkWallet};
|
|
use std::convert::TryFrom;
|
|
use std::str::FromStr;
|
|
use std::sync::{Arc, Mutex, MutexGuard};
|
|
|
|
uniffi_macros::include_scaffolding!("bdk");
|
|
|
|
type BdkError = Error;
|
|
|
|
pub enum DatabaseConfig {
|
|
Memory,
|
|
Sled { config: SledDbConfiguration },
|
|
Sqlite { config: SqliteDbConfiguration },
|
|
}
|
|
|
|
pub struct ElectrumConfig {
|
|
pub url: String,
|
|
pub socks5: Option<String>,
|
|
pub retry: u8,
|
|
pub timeout: Option<u8>,
|
|
pub stop_gap: u64,
|
|
}
|
|
|
|
pub struct EsploraConfig {
|
|
pub base_url: String,
|
|
pub proxy: Option<String>,
|
|
pub timeout_read: u64,
|
|
pub timeout_write: u64,
|
|
pub stop_gap: u64,
|
|
}
|
|
|
|
pub enum BlockchainConfig {
|
|
Electrum { config: ElectrumConfig },
|
|
Esplora { config: EsploraConfig },
|
|
}
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Default)]
|
|
pub struct TransactionDetails {
|
|
pub fees: Option<u64>,
|
|
pub received: u64,
|
|
pub sent: u64,
|
|
pub txid: String,
|
|
}
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
pub enum Transaction {
|
|
Unconfirmed {
|
|
details: TransactionDetails,
|
|
},
|
|
Confirmed {
|
|
details: TransactionDetails,
|
|
confirmation: BlockTime,
|
|
},
|
|
}
|
|
|
|
impl From<&bdk::TransactionDetails> for TransactionDetails {
|
|
fn from(x: &bdk::TransactionDetails) -> TransactionDetails {
|
|
TransactionDetails {
|
|
fees: x.fee,
|
|
txid: x.txid.to_string(),
|
|
received: x.received,
|
|
sent: x.sent,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<&bdk::TransactionDetails> for Transaction {
|
|
fn from(x: &bdk::TransactionDetails) -> Transaction {
|
|
match x.confirmation_time.clone() {
|
|
Some(block_time) => Transaction::Confirmed {
|
|
details: TransactionDetails::from(x),
|
|
confirmation: block_time,
|
|
},
|
|
None => Transaction::Unconfirmed {
|
|
details: TransactionDetails::from(x),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
struct Wallet {
|
|
wallet_mutex: Mutex<BdkWallet<AnyBlockchain, AnyDatabase>>,
|
|
}
|
|
|
|
pub trait BdkProgress: Send + Sync {
|
|
fn update(&self, progress: f32, message: Option<String>);
|
|
}
|
|
|
|
struct BdkProgressHolder {
|
|
progress_update: Box<dyn BdkProgress>,
|
|
}
|
|
|
|
impl Progress for BdkProgressHolder {
|
|
fn update(&self, progress: f32, message: Option<String>) -> Result<(), Error> {
|
|
self.progress_update.update(progress, message);
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct PartiallySignedBitcoinTransaction {
|
|
internal: Mutex<PartiallySignedTransaction>,
|
|
}
|
|
|
|
impl PartiallySignedBitcoinTransaction {
|
|
fn new(psbt_base64: String) -> Result<Self, Error> {
|
|
let psbt: PartiallySignedTransaction = PartiallySignedTransaction::from_str(&psbt_base64)?;
|
|
Ok(PartiallySignedBitcoinTransaction {
|
|
internal: Mutex::new(psbt),
|
|
})
|
|
}
|
|
|
|
fn serialize(&self) -> String {
|
|
let psbt = self.internal.lock().unwrap().clone();
|
|
psbt.to_string()
|
|
}
|
|
}
|
|
|
|
impl Wallet {
|
|
fn new(
|
|
descriptor: String,
|
|
change_descriptor: Option<String>,
|
|
network: Network,
|
|
database_config: DatabaseConfig,
|
|
blockchain_config: BlockchainConfig,
|
|
) -> Result<Self, BdkError> {
|
|
let any_database_config = match database_config {
|
|
DatabaseConfig::Memory => AnyDatabaseConfig::Memory(()),
|
|
DatabaseConfig::Sled { config } => AnyDatabaseConfig::Sled(config),
|
|
DatabaseConfig::Sqlite { config } => AnyDatabaseConfig::Sqlite(config),
|
|
};
|
|
let any_blockchain_config = match blockchain_config {
|
|
BlockchainConfig::Electrum { config } => {
|
|
AnyBlockchainConfig::Electrum(ElectrumBlockchainConfig {
|
|
retry: config.retry,
|
|
socks5: config.socks5,
|
|
timeout: config.timeout,
|
|
url: config.url,
|
|
stop_gap: usize::try_from(config.stop_gap).unwrap(),
|
|
})
|
|
}
|
|
BlockchainConfig::Esplora { config } => {
|
|
AnyBlockchainConfig::Esplora(EsploraBlockchainConfig {
|
|
base_url: config.base_url,
|
|
proxy: config.proxy,
|
|
timeout_read: config.timeout_read,
|
|
timeout_write: config.timeout_write,
|
|
stop_gap: usize::try_from(config.stop_gap).unwrap(),
|
|
})
|
|
}
|
|
};
|
|
let database = AnyDatabase::from_config(&any_database_config)?;
|
|
let blockchain = AnyBlockchain::from_config(&any_blockchain_config)?;
|
|
let wallet_mutex = Mutex::new(BdkWallet::new(
|
|
&descriptor,
|
|
change_descriptor.as_ref(),
|
|
network,
|
|
database,
|
|
blockchain,
|
|
)?);
|
|
Ok(Wallet { wallet_mutex })
|
|
}
|
|
|
|
fn get_wallet(&self) -> MutexGuard<BdkWallet<AnyBlockchain, AnyDatabase>> {
|
|
self.wallet_mutex.lock().expect("wallet")
|
|
}
|
|
|
|
fn get_network(&self) -> Network {
|
|
self.get_wallet().network()
|
|
}
|
|
|
|
fn sync(
|
|
&self,
|
|
progress_update: Box<dyn BdkProgress>,
|
|
max_address_param: Option<u32>,
|
|
) -> Result<(), BdkError> {
|
|
self.get_wallet()
|
|
.sync(BdkProgressHolder { progress_update }, max_address_param)
|
|
}
|
|
|
|
fn get_new_address(&self) -> String {
|
|
self.get_wallet()
|
|
.get_address(AddressIndex::New)
|
|
.unwrap()
|
|
.address
|
|
.to_string()
|
|
}
|
|
|
|
fn get_last_unused_address(&self) -> String {
|
|
self.get_wallet()
|
|
.get_address(AddressIndex::LastUnused)
|
|
.unwrap()
|
|
.address
|
|
.to_string()
|
|
}
|
|
|
|
fn get_balance(&self) -> Result<u64, Error> {
|
|
self.get_wallet().get_balance()
|
|
}
|
|
|
|
fn sign(&self, psbt: &PartiallySignedBitcoinTransaction) -> Result<(), Error> {
|
|
let mut psbt = psbt.internal.lock().unwrap();
|
|
let finalized = self.get_wallet().sign(&mut psbt, SignOptions::default())?;
|
|
match finalized {
|
|
true => Ok(()),
|
|
false => Err(BdkError::Generic(format!(
|
|
"transaction signing not finalized {:?}",
|
|
psbt
|
|
))),
|
|
}
|
|
}
|
|
|
|
fn get_transactions(&self) -> Result<Vec<Transaction>, Error> {
|
|
let transactions = self.get_wallet().list_transactions(true)?;
|
|
Ok(transactions.iter().map(Transaction::from).collect())
|
|
}
|
|
|
|
fn broadcast(&self, psbt: &PartiallySignedBitcoinTransaction) -> Result<String, Error> {
|
|
let tx = psbt.internal.lock().unwrap().clone().extract_tx();
|
|
let txid = self.get_wallet().broadcast(&tx)?;
|
|
Ok(txid.to_hex())
|
|
}
|
|
}
|
|
|
|
pub struct ExtendedKeyInfo {
|
|
mnemonic: String,
|
|
xprv: String,
|
|
fingerprint: String,
|
|
}
|
|
|
|
fn generate_extended_key(
|
|
network: Network,
|
|
word_count: WordCount,
|
|
password: Option<String>,
|
|
) -> Result<ExtendedKeyInfo, Error> {
|
|
let mnemonic: GeneratedKey<_, BareCtx> =
|
|
Mnemonic::generate((word_count, Language::English)).unwrap();
|
|
let mnemonic = mnemonic.into_key();
|
|
let xkey: ExtendedKey = (mnemonic.clone(), password).into_extended_key()?;
|
|
let xprv = xkey.into_xprv(network).unwrap();
|
|
let fingerprint = xprv.fingerprint(&Secp256k1::new());
|
|
Ok(ExtendedKeyInfo {
|
|
mnemonic: mnemonic.to_string(),
|
|
xprv: xprv.to_string(),
|
|
fingerprint: fingerprint.to_string(),
|
|
})
|
|
}
|
|
|
|
fn restore_extended_key(
|
|
network: Network,
|
|
mnemonic: String,
|
|
password: Option<String>,
|
|
) -> Result<ExtendedKeyInfo, Error> {
|
|
let mnemonic = Mnemonic::parse_in(Language::English, mnemonic).unwrap();
|
|
let xkey: ExtendedKey = (mnemonic.clone(), password).into_extended_key()?;
|
|
let xprv = xkey.into_xprv(network).unwrap();
|
|
let fingerprint = xprv.fingerprint(&Secp256k1::new());
|
|
Ok(ExtendedKeyInfo {
|
|
mnemonic: mnemonic.to_string(),
|
|
xprv: xprv.to_string(),
|
|
fingerprint: fingerprint.to_string(),
|
|
})
|
|
}
|
|
|
|
struct TxBuilder {
|
|
recipients: Vec<(String, u64)>,
|
|
fee_rate: Option<f32>,
|
|
drain_wallet: bool,
|
|
drain_to: Option<String>,
|
|
}
|
|
|
|
impl TxBuilder {
|
|
fn new() -> Self {
|
|
TxBuilder {
|
|
recipients: Vec::new(),
|
|
fee_rate: None,
|
|
drain_wallet: false,
|
|
drain_to: None,
|
|
}
|
|
}
|
|
|
|
fn add_recipient(&self, recipient: String, amount: u64) -> Arc<Self> {
|
|
let mut recipients = self.recipients.to_vec();
|
|
recipients.append(&mut vec![(recipient, amount)]);
|
|
Arc::new(TxBuilder {
|
|
recipients,
|
|
fee_rate: self.fee_rate,
|
|
drain_wallet: self.drain_wallet,
|
|
drain_to: self.drain_to.clone(),
|
|
})
|
|
}
|
|
|
|
fn fee_rate(&self, sat_per_vb: f32) -> Arc<Self> {
|
|
Arc::new(TxBuilder {
|
|
recipients: self.recipients.to_vec(),
|
|
fee_rate: Some(sat_per_vb),
|
|
drain_wallet: self.drain_wallet,
|
|
drain_to: self.drain_to.clone(),
|
|
})
|
|
}
|
|
|
|
fn drain_wallet(&self) -> Arc<Self> {
|
|
Arc::new(TxBuilder {
|
|
recipients: self.recipients.to_vec(),
|
|
fee_rate: self.fee_rate,
|
|
drain_wallet: true,
|
|
drain_to: self.drain_to.clone(),
|
|
})
|
|
}
|
|
|
|
fn drain_to(&self, address: String) -> Arc<Self> {
|
|
Arc::new(TxBuilder {
|
|
recipients: self.recipients.to_vec(),
|
|
fee_rate: self.fee_rate,
|
|
drain_wallet: self.drain_wallet,
|
|
drain_to: Some(address),
|
|
})
|
|
}
|
|
|
|
fn build(&self, wallet: &Wallet) -> Result<Arc<PartiallySignedBitcoinTransaction>, Error> {
|
|
let wallet = wallet.get_wallet();
|
|
let mut tx_builder = wallet.build_tx();
|
|
for (address, amount) in &self.recipients {
|
|
let address = Address::from_str(address).map_err(|e| BdkError::Generic(e.to_string()))?;
|
|
tx_builder.add_recipient(address.script_pubkey(), *amount);
|
|
}
|
|
if let Some(sat_per_vb) = self.fee_rate {
|
|
tx_builder.fee_rate(FeeRate::from_sat_per_vb(sat_per_vb));
|
|
}
|
|
if self.drain_wallet {
|
|
tx_builder.drain_wallet();
|
|
}
|
|
if let Some(address) = &self.drain_to {
|
|
let drain_to =
|
|
Address::from_str(address).map_err(|e| BdkError::Generic(e.to_string()))?;
|
|
tx_builder.drain_to(drain_to.script_pubkey());
|
|
}
|
|
tx_builder
|
|
.finish()
|
|
.map(|(psbt, _)| PartiallySignedBitcoinTransaction {
|
|
internal: Mutex::new(psbt),
|
|
})
|
|
.map(Arc::new)
|
|
}
|
|
}
|
|
|
|
uniffi::deps::static_assertions::assert_impl_all!(Wallet: Sync, Send);
|