use std::io::{Error, Write};
use std::str::FromStr;
use anyhow::format_err;
use bitcoin::address::NetworkUnchecked;
use bitcoin::hashes::Hash as BitcoinHash;
use hex::{FromHex, ToHex};
use miniscript::{Descriptor, MiniscriptKey};
use crate::bitcoin_migration::{
bitcoin29_to_bitcoin30_network_magic, bitcoin29_to_bitcoin30_psbt,
bitcoin30_to_bitcoin29_network_magic, bitcoin30_to_bitcoin29_psbt,
checked_address_to_unchecked_address,
};
use crate::encoding::{Decodable, DecodeError, Encodable};
use crate::module::registry::ModuleDecoderRegistry;
macro_rules! impl_encode_decode_bridge {
($btc_type:ty) => {
impl crate::encoding::Encodable for $btc_type {
fn consensus_encode<W: std::io::Write>(
&self,
writer: &mut W,
) -> Result<usize, std::io::Error> {
bitcoin::consensus::Encodable::consensus_encode(self, writer)
}
}
impl crate::encoding::Decodable for $btc_type {
fn consensus_decode_from_finite_reader<D: std::io::Read>(
d: &mut D,
_modules: &$crate::module::registry::ModuleDecoderRegistry,
) -> Result<Self, crate::encoding::DecodeError> {
bitcoin::consensus::Decodable::consensus_decode_from_finite_reader(d)
.map_err(crate::encoding::DecodeError::from_err)
}
}
};
}
impl_encode_decode_bridge!(bitcoin::block::Header);
impl_encode_decode_bridge!(bitcoin::BlockHash);
impl_encode_decode_bridge!(bitcoin::OutPoint);
impl_encode_decode_bridge!(bitcoin::ScriptBuf);
impl_encode_decode_bridge!(bitcoin::Transaction);
impl_encode_decode_bridge!(bitcoin::merkle_tree::PartialMerkleTree);
impl crate::encoding::Encodable for bitcoin::psbt::PartiallySignedTransaction {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
bitcoin29::consensus::Encodable::consensus_encode(
&bitcoin30_to_bitcoin29_psbt(self),
writer,
)
}
}
impl crate::encoding::Decodable for bitcoin::psbt::PartiallySignedTransaction {
fn consensus_decode_from_finite_reader<D: std::io::Read>(
d: &mut D,
_modules: &ModuleDecoderRegistry,
) -> Result<Self, crate::encoding::DecodeError> {
Ok(bitcoin29_to_bitcoin30_psbt(
&bitcoin29::consensus::Decodable::consensus_decode_from_finite_reader(d)
.map_err(crate::encoding::DecodeError::from_err)?,
))
}
}
impl crate::encoding::Encodable for bitcoin::Txid {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
bitcoin::consensus::Encodable::consensus_encode(self, writer)
}
fn consensus_encode_to_hex(&self) -> String {
let mut bytes = vec![];
self.consensus_encode(&mut bytes)
.expect("encoding to bytes can't fail for io reasons");
bytes.reverse();
bytes.encode_hex()
}
}
impl crate::encoding::Decodable for bitcoin::Txid {
fn consensus_decode_from_finite_reader<D: std::io::Read>(
d: &mut D,
_modules: &::fedimint_core::module::registry::ModuleDecoderRegistry,
) -> Result<Self, crate::encoding::DecodeError> {
bitcoin::consensus::Decodable::consensus_decode_from_finite_reader(d)
.map_err(crate::encoding::DecodeError::from_err)
}
fn consensus_decode_hex(
hex: &str,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let mut bytes = Vec::<u8>::from_hex(hex)
.map_err(anyhow::Error::from)
.map_err(DecodeError::new_custom)?;
bytes.reverse();
let mut reader = std::io::Cursor::new(bytes);
Decodable::consensus_decode(&mut reader, modules)
}
}
impl<K> Encodable for miniscript::Descriptor<K>
where
K: miniscript::MiniscriptKey,
{
fn consensus_encode<W: Write>(&self, writer: &mut W) -> Result<usize, Error> {
let descriptor_str = self.to_string();
descriptor_str.consensus_encode(writer)
}
}
impl<K> Decodable for miniscript::Descriptor<K>
where
Self: FromStr,
<Self as FromStr>::Err: ToString + std::error::Error + Send + Sync + 'static,
K: MiniscriptKey,
{
fn consensus_decode_from_finite_reader<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let descriptor_str = String::consensus_decode_from_finite_reader(d, modules)?;
Descriptor::<K>::from_str(&descriptor_str).map_err(DecodeError::from_err)
}
}
impl Encodable for bitcoin::network::Magic {
fn consensus_encode<W: Write>(&self, writer: &mut W) -> Result<usize, Error> {
let num = bitcoin30_to_bitcoin29_network_magic(self);
num.consensus_encode(writer)
}
}
impl Decodable for bitcoin::network::Magic {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let num = u32::consensus_decode(d, modules)?;
Ok(bitcoin29_to_bitcoin30_network_magic(num))
}
}
impl Encodable for bitcoin::Network {
fn consensus_encode<W: Write>(&self, writer: &mut W) -> Result<usize, Error> {
self.magic().consensus_encode(writer)
}
}
impl Decodable for bitcoin::Network {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let magic = bitcoin::network::Magic::consensus_decode(d, modules)?;
bitcoin::Network::from_magic(magic).ok_or_else(|| {
DecodeError::new_custom(format_err!("Unknown network magic: {:x}", magic))
})
}
}
impl Encodable for bitcoin::Amount {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, std::io::Error> {
self.to_sat().consensus_encode(writer)
}
}
impl Decodable for bitcoin::Amount {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
Ok(bitcoin::Amount::from_sat(u64::consensus_decode(
d, modules,
)?))
}
}
impl Encodable for bitcoin::Address {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, Error> {
let mut len = 0;
len += self.network.magic().consensus_encode(writer)?;
len += self.script_pubkey().consensus_encode(writer)?;
Ok(len)
}
}
impl Encodable for bitcoin::Address<NetworkUnchecked> {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, Error> {
self.clone().assume_checked().consensus_encode(writer)
}
}
impl Decodable for bitcoin::Address<NetworkUnchecked> {
fn consensus_decode<D: std::io::Read>(
mut d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
let network = bitcoin::Network::from_magic(bitcoin::network::Magic::consensus_decode(
&mut d, modules,
)?)
.ok_or_else(|| DecodeError::from_str("Unknown network"))?;
let script_pk = bitcoin::ScriptBuf::consensus_decode(&mut d, modules)?;
let address = bitcoin::Address::from_script(&script_pk, network)
.map_err(|e| DecodeError::new_custom(e.into()))?;
Ok(checked_address_to_unchecked_address(&address))
}
}
impl Encodable for bitcoin::hashes::sha256::Hash {
fn consensus_encode<W: std::io::Write>(&self, writer: &mut W) -> Result<usize, Error> {
self.to_byte_array().consensus_encode(writer)
}
}
impl Decodable for bitcoin::hashes::sha256::Hash {
fn consensus_decode<D: std::io::Read>(
d: &mut D,
modules: &ModuleDecoderRegistry,
) -> Result<Self, DecodeError> {
Ok(bitcoin::hashes::sha256::Hash::from_byte_array(
Decodable::consensus_decode(d, modules)?,
))
}
}
#[cfg(test)]
mod tests {
use std::io::Cursor;
use std::str::FromStr;
use bitcoin::hashes::Hash as BitcoinHash;
use crate::encoding::{Decodable, Encodable};
use crate::ModuleDecoderRegistry;
#[test_log::test]
fn sha256_roundtrip() {
let hash = bitcoin::hashes::sha256::Hash::hash(b"Hello world!");
let mut encoded = Vec::new();
hash.consensus_encode(&mut encoded).unwrap();
let hash_decoded = bitcoin::hashes::sha256::Hash::consensus_decode(
&mut Cursor::new(encoded),
&ModuleDecoderRegistry::default(),
)
.unwrap();
assert_eq!(hash, hash_decoded);
}
#[test_log::test]
fn address_roundtrip() {
let addresses = [
"bc1p2wsldez5mud2yam29q22wgfh9439spgduvct83k3pm50fcxa5dps59h4z5",
"mxMYaq5yWinZ9AKjCDcBEbiEwPJD9n2uLU",
"1FK8o7mUxyd6QWJAUw7J4vW7eRxuyjj6Ne",
"3JSrSU7z7R1Yhh26pt1zzRjQz44qjcrXwb",
"tb1qunn0thpt8uk3yk2938ypjccn3urxprt78z9ccq",
"2MvUMRv2DRHZi3VshkP7RMEU84mVTfR9xjq",
];
for address_str in addresses {
let address =
bitcoin::Address::from_str(address_str).expect("All tested addresses are valid");
let mut encoding = vec![];
address
.consensus_encode(&mut encoding)
.expect("Encoding to vec can't fail");
let mut cursor = Cursor::new(encoding);
let parsed_address =
bitcoin::Address::consensus_decode(&mut cursor, &ModuleDecoderRegistry::default())
.expect("Decoding address failed");
assert_eq!(address, parsed_address);
}
}
}