Skip to content
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .github/workflows/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ jobs:
matrix:
rust:
- version: 1.61.0 # STABLE
- version: 1.41.1 # MSRV
- version: 1.56.1 # MSRV
emulator:
- name: trezor
- name: ledger
Expand Down
1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ bitcoin = { version = "0.28", features = ["use-serde", "base64"] }
serde = { version = "^1.0", features = ["derive"] }
serde_json = { version = "^1.0" }
pyo3 = { version = "0.15.1", features = ["auto-initialize"]}
bdk = "0.20.0"

[dev-dependencies]
serial_test = "0.6.0"
2 changes: 2 additions & 0 deletions src/interface.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ macro_rules! deserialize_obj {
}

/// Convenience class containing required Python objects
#[derive(Debug)]
struct HWILib {
commands: Py<PyModule>,
json_dumps: Py<PyAny>,
Expand All @@ -44,6 +45,7 @@ impl HWILib {
}
}

#[derive(Debug)]
pub struct HWIClient {
hwilib: HWILib,
hw_client: PyObject,
Expand Down
50 changes: 50 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,17 +31,27 @@ pub use interface::HWIClient;

pub mod error;
pub mod interface;
pub mod signer;
pub mod types;

#[cfg(test)]
mod tests {
use crate::signer::HWISigner;
use crate::types;
use crate::HWIClient;
use std::collections::BTreeMap;
use std::str::FromStr;
use std::sync::Arc;

use bdk::database::MemoryDatabase;
use bdk::signer::SignerOrdering;
use bdk::KeychainKind;
use bdk::SignOptions;
use bdk::Wallet;
use bitcoin::psbt::{Input, Output};
use bitcoin::util::bip32::{DerivationPath, KeySource};
use bitcoin::Address;
use bitcoin::Network;
use bitcoin::{secp256k1, Transaction};
use bitcoin::{TxIn, TxOut};

Expand Down Expand Up @@ -320,4 +330,44 @@ mod tests {
client.wipe_device().unwrap();
}
}
#[test]
#[serial]
fn test_create_signer() {
let descriptors = get_first_device().get_descriptors(None).unwrap();
let devices = HWIClient::enumerate().unwrap();
let custom_signer =
HWISigner::from_device(devices.first().unwrap(), types::HWIChain::Test).unwrap();

let mut wallet = Wallet::new(
&descriptors.internal[0],
Some(&descriptors.receive[0]),
Network::Testnet,
MemoryDatabase::default(),
)
.unwrap();
wallet.add_signer(
KeychainKind::External,
SignerOrdering(200),
Arc::new(custom_signer),
);

let client =
bdk::electrum_client::Client::new("ssl://electrum.blockstream.info:60002").unwrap();
let blockchain = bdk::blockchain::ElectrumBlockchain::from(client);
wallet
.sync(&blockchain, bdk::SyncOptions::default())
.unwrap();

let balance = wallet.get_balance().unwrap();
let faucet_address = Address::from_str("mv4rnyY3Su5gjcDNzbMLKBQkBicCtHUtFB").unwrap();

let mut tx_builder = wallet.build_tx();
tx_builder
.add_recipient(faucet_address.script_pubkey(), balance / 2)
.enable_rbf();
let (mut psbt, _tx_details) = tx_builder.finish().unwrap();
let finalized = wallet.sign(&mut psbt, SignOptions::default()).unwrap();

assert!(finalized, "Tx has not been finalized");
}
}
51 changes: 51 additions & 0 deletions src/signer.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
use bdk::signer::{SignerCommon, SignerError, SignerId, TransactionSigner};
use bitcoin::{
psbt::PartiallySignedTransaction,
secp256k1::{All, Secp256k1},
util::bip32::Fingerprint,
};

use crate::{
error::Error,
types::{HWIChain, HWIDevice},
HWIClient,
};

#[derive(Debug)]
pub struct HWISigner {
fingerprint: Fingerprint,
client: HWIClient,
}

impl HWISigner {
pub fn from_device(device: &HWIDevice, chain: HWIChain) -> Result<HWISigner, Error> {
let client = HWIClient::get_client(device, false, chain)?;
Ok(HWISigner {
fingerprint: device.fingerprint,
client,
})
}
}

impl SignerCommon for HWISigner {
fn id(&self, _secp: &Secp256k1<All>) -> SignerId {
SignerId::Fingerprint(self.fingerprint)
}
}

impl TransactionSigner for HWISigner {
fn sign_transaction(
&self,
psbt: &mut PartiallySignedTransaction,
_secp: &Secp256k1<All>,
) -> Result<(), SignerError> {
psbt.combine(
self.client
.sign_tx(psbt)
.expect("Hardware Wallet couldn't sign transaction")
.psbt,
)
.expect("Failed to combine HW signed psbt with passed PSBT");
Ok(())
}
}