#[macro_use] extern crate lazy_static;
#[macro_use] extern crate serde as serde_crate;
pub extern crate bitcoin;
pub mod cpfp;
pub mod fee;
#[cfg(feature = "bdk")]
pub mod bdk;
#[cfg(feature = "rpc")]
pub mod rpc;
pub mod serde;
pub use mbitcoin::{
AmountExt, FeeRateExt, TaprootSpendInfoExt, KeypairExt, TransactionExt, TxOutExt,
};
#[path = "bitcoin.rs"]
mod mbitcoin;
use std::{fmt, str::FromStr};
use bitcoin::{Amount, BlockHash};
use serde_crate::ser::SerializeStruct;
pub const DEEPLY_CONFIRMED: BlockHeight = 100;
pub const P2TR_DUST_VB: u64 = 110;
pub const P2TR_DUST_SAT: u64 = P2TR_DUST_VB * 3;
pub const P2TR_DUST: Amount = Amount::from_sat(P2TR_DUST_SAT);
pub const P2WPKH_DUST_VB: u64 = 90;
pub const P2WPKH_DUST_SAT: u64 = P2WPKH_DUST_VB * 3;
pub const P2WPKH_DUST: Amount = Amount::from_sat(P2WPKH_DUST_SAT);
pub const P2PKH_DUST_VB: u64 = 182;
pub const P2PKH_DUST_SAT: u64 = P2PKH_DUST_VB * 3;
pub const P2PKH_DUST: Amount = Amount::from_sat(P2PKH_DUST_SAT);
pub const P2SH_DUST_VB: u64 = 180;
pub const P2SH_DUST_SAT: u64 = P2SH_DUST_VB * 3;
pub const P2SH_DUST: Amount = Amount::from_sat(P2SH_DUST_SAT);
pub const P2WSH_DUST_VB: u64 = 110;
pub const P2WSH_DUST_SAT: u64 = P2WSH_DUST_VB * 3;
pub const P2WSH_DUST: Amount = Amount::from_sat(P2WSH_DUST_SAT);
pub const TAPROOT_KEYSPEND_WEIGHT: usize = 66;
pub type BlockHeight = u32;
pub type BlockDelta = u16;
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct BlockRef {
pub height: BlockHeight,
pub hash: BlockHash,
}
impl fmt::Display for BlockRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}:{}", self.height, self.hash)
}
}
impl fmt::Debug for BlockRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Display::fmt(self, f)
}
}
impl FromStr for BlockRef {
type Err = &'static str;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let mut parts = s.splitn(2, ':');
Ok(BlockRef {
height: parts.next().expect("always one part")
.parse().map_err(|_| "invalid height")?,
hash: parts.next().ok_or("should be <height>:<hash> string")?
.parse().map_err(|_| "invalid hash")?,
})
}
}
impl serde_crate::Serialize for BlockRef {
fn serialize<S: serde_crate::Serializer>(&self, s: S) -> Result<S::Ok, S::Error> {
let mut state = s.serialize_struct("BlockRef", 2)?;
state.serialize_field("height", &self.height)?;
state.serialize_field("hash", &self.hash)?;
state.end()
}
}
impl<'de> serde_crate::Deserialize<'de> for BlockRef {
fn deserialize<D: serde_crate::Deserializer<'de>>(d: D) -> Result<Self, D::Error> {
struct Visitor;
impl<'de> serde_crate::de::Visitor<'de> for Visitor {
type Value = BlockRef;
fn expecting(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "a BlockRef (struct/string)")
}
fn visit_str<E: serde_crate::de::Error>(self, v: &str) -> Result<Self::Value, E> {
BlockRef::from_str(v).map_err(serde_crate::de::Error::custom)
}
fn visit_map<A: serde_crate::de::MapAccess<'de>>(self, mut map: A) -> Result<Self::Value, A::Error> {
let mut height = None;
let mut hash = None;
while let Some(key) = map.next_key::<&str>()? {
match key {
"height" => height = Some(map.next_value()?),
"hash" => hash = Some(map.next_value()?),
_ => {
let _ = map.next_value::<serde_crate::de::IgnoredAny>()?;
}
}
}
Ok(BlockRef {
height: height.ok_or_else(|| serde_crate::de::Error::missing_field("height"))?,
hash: hash.ok_or_else(|| serde_crate::de::Error::missing_field("hash"))?,
})
}
}
d.deserialize_any(Visitor)
}
}
#[cfg(feature = "bdk")]
impl From<bdk_wallet::chain::BlockId> for BlockRef {
fn from(id: bdk_wallet::chain::BlockId) -> Self {
Self {
height: id.height,
hash: id.hash,
}
}
}
#[derive(Copy, Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
pub enum TxStatus {
Confirmed(BlockRef),
Mempool,
NotFound,
}
impl TxStatus {
pub fn is_confirmed(&self) -> bool {
match self {
TxStatus::Confirmed(_) => true,
_ => false,
}
}
pub fn confirmed_height(&self) -> Option<BlockHeight> {
match self {
TxStatus::Confirmed(block_ref) => Some(block_ref.height),
_ => None,
}
}
pub fn confirmed_in(&self) -> Option<BlockRef> {
match self {
TxStatus::Confirmed(block_ref) => Some(*block_ref),
_ => None,
}
}
}