#[cfg(feature = "ledger")]
pub mod ledger;
#[cfg(feature = "specter")]
pub mod specter;
use async_trait::async_trait;
use bitcoin::util::{
bip32::{DerivationPath, ExtendedPubKey, Fingerprint},
psbt::PartiallySignedTransaction as Psbt,
};
use std::fmt::Debug;
#[derive(Debug, Clone)]
pub enum Error {
UnsupportedVersion,
UnsupportedInput,
UnimplementedMethod,
DeviceDisconnected,
DeviceNotFound,
DeviceDidNotSign,
Device(String),
}
impl std::fmt::Display for Error {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
Error::UnsupportedVersion => write!(f, "Unsupported version"),
Error::UnsupportedInput => write!(f, "Unsupported input"),
Error::UnimplementedMethod => write!(f, "Unimplemented method"),
Error::DeviceDisconnected => write!(f, "Device disconnected"),
Error::DeviceNotFound => write!(f, "Device not found"),
Error::DeviceDidNotSign => write!(f, "Device did not sign"),
Error::Device(e) => write!(f, "{}", e),
}
}
}
#[async_trait]
pub trait HWI: Debug {
fn device_kind(&self) -> DeviceKind;
async fn get_version(&self) -> Result<Version, Error>;
async fn is_connected(&self) -> Result<(), Error>;
async fn get_master_fingerprint(&self) -> Result<Fingerprint, Error>;
async fn get_extended_pubkey(
&self,
path: &DerivationPath,
display: bool,
) -> Result<ExtendedPubKey, Error>;
async fn register_wallet(&self, name: &str, policy: &str) -> Result<Option<[u8; 32]>, Error>;
async fn sign_tx(&self, tx: &mut Psbt) -> Result<(), Error>;
}
#[derive(PartialEq, Eq, Debug, Clone, Default)]
pub struct Version {
pub major: u32,
pub minor: u32,
pub patch: u32,
pub prerelease: Option<String>,
}
impl std::fmt::Display for Version {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
if let Some(prerelease) = &self.prerelease {
write!(
f,
"{}.{}.{}-{}",
self.major, self.minor, self.patch, prerelease
)
} else {
write!(f, "{}.{}.{}", self.major, self.minor, self.patch)
}
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum DeviceKind {
Specter,
SpecterSimulator,
Ledger,
LedgerSimulator,
}
impl std::fmt::Display for DeviceKind {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
DeviceKind::Specter => write!(f, "specter"),
DeviceKind::SpecterSimulator => write!(f, "specter-simulator"),
DeviceKind::Ledger => write!(f, "ledger"),
DeviceKind::LedgerSimulator => write!(f, "ledger-simulator"),
}
}
}
impl std::str::FromStr for DeviceKind {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"specter" => Ok(DeviceKind::Specter),
"specter-simulator" => Ok(DeviceKind::SpecterSimulator),
"ledger" => Ok(DeviceKind::Ledger),
"ledger-simulator" => Ok(DeviceKind::LedgerSimulator),
_ => Err(()),
}
}
}