Use marker type to enforce validation of `Address`'s network
Parsing addresses from strings required a subsequent validation of network of the parsed address. However, this validation was not enforced by compiler, one had to remember to perform it. This change adds a marker type to `Address` that will assist the compiler in enforcing this validation.
This commit is contained in:
parent
ac6340943c
commit
bef7c6e687
bitcoin
examples
fuzz/fuzz_targets
src
|
@ -177,7 +177,8 @@ impl WatchOnly {
|
||||||
|
|
||||||
/// Creates the PSBT, in BIP174 parlance this is the 'Creater'.
|
/// Creates the PSBT, in BIP174 parlance this is the 'Creater'.
|
||||||
fn create_psbt<C: Verification>(&self, secp: &Secp256k1<C>) -> Result<Psbt> {
|
fn create_psbt<C: Verification>(&self, secp: &Secp256k1<C>) -> Result<Psbt> {
|
||||||
let to_address = Address::from_str(RECEIVE_ADDRESS)?;
|
let to_address = Address::from_str(RECEIVE_ADDRESS)?
|
||||||
|
.require_network(Network::Regtest)?;
|
||||||
let to_amount = Amount::from_str(OUTPUT_AMOUNT_BTC)?;
|
let to_amount = Amount::from_str(OUTPUT_AMOUNT_BTC)?;
|
||||||
|
|
||||||
let (_, change_address, _) = self.change_address(secp)?;
|
let (_, change_address, _) = self.change_address(secp)?;
|
||||||
|
|
|
@ -93,7 +93,7 @@ use bitcoin::taproot::{
|
||||||
LeafVersion, TapLeafHash, TapSighashHash, TaprootBuilder, TaprootSpendInfo,
|
LeafVersion, TapLeafHash, TapSighashHash, TaprootBuilder, TaprootSpendInfo,
|
||||||
};
|
};
|
||||||
use bitcoin::{
|
use bitcoin::{
|
||||||
absolute, script, Address, Amount, OutPoint, ScriptBuf, Transaction, TxIn, TxOut, Txid, Witness,
|
absolute, script, Address, Amount, Network, OutPoint, ScriptBuf, Transaction, TxIn, TxOut, Txid, Witness,
|
||||||
};
|
};
|
||||||
|
|
||||||
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
@ -104,9 +104,11 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
|
||||||
// Just some addresses for outputs from our wallets. Not really important.
|
// Just some addresses for outputs from our wallets. Not really important.
|
||||||
let to_address =
|
let to_address =
|
||||||
Address::from_str("bcrt1p0p3rvwww0v9znrclp00uneq8ytre9kj922v8fxhnezm3mgsmn9usdxaefc")?;
|
Address::from_str("bcrt1p0p3rvwww0v9znrclp00uneq8ytre9kj922v8fxhnezm3mgsmn9usdxaefc")?
|
||||||
|
.require_network(Network::Regtest)?;
|
||||||
let change_address =
|
let change_address =
|
||||||
Address::from_str("bcrt1pz449kexzydh2kaypatup5ultru3ej284t6eguhnkn6wkhswt0l7q3a7j76")?;
|
Address::from_str("bcrt1pz449kexzydh2kaypatup5ultru3ej284t6eguhnkn6wkhswt0l7q3a7j76")?
|
||||||
|
.require_network(Network::Regtest)?;
|
||||||
let amount_to_send_in_sats = COIN_VALUE;
|
let amount_to_send_in_sats = COIN_VALUE;
|
||||||
let change_amount = UTXO_1
|
let change_amount = UTXO_1
|
||||||
.amount_in_sats
|
.amount_in_sats
|
||||||
|
|
|
@ -3,7 +3,7 @@ use std::str::FromStr;
|
||||||
fn do_test(data: &[u8]) {
|
fn do_test(data: &[u8]) {
|
||||||
let data_str = String::from_utf8_lossy(data);
|
let data_str = String::from_utf8_lossy(data);
|
||||||
let addr = match bitcoin::address::Address::from_str(&data_str) {
|
let addr = match bitcoin::address::Address::from_str(&data_str) {
|
||||||
Ok(addr) => addr,
|
Ok(addr) => addr.assume_checked(),
|
||||||
Err(_) => return,
|
Err(_) => return,
|
||||||
};
|
};
|
||||||
assert_eq!(addr.to_string(), data_str);
|
assert_eq!(addr.to_string(), data_str);
|
||||||
|
|
|
@ -29,6 +29,7 @@
|
||||||
|
|
||||||
use core::convert::TryFrom;
|
use core::convert::TryFrom;
|
||||||
use core::fmt;
|
use core::fmt;
|
||||||
|
use core::marker::PhantomData;
|
||||||
use core::str::FromStr;
|
use core::str::FromStr;
|
||||||
|
|
||||||
use bech32;
|
use bech32;
|
||||||
|
@ -87,6 +88,13 @@ pub enum Error {
|
||||||
UnrecognizedScript,
|
UnrecognizedScript,
|
||||||
/// Address type is either invalid or not supported in rust-bitcoin.
|
/// Address type is either invalid or not supported in rust-bitcoin.
|
||||||
UnknownAddressType(String),
|
UnknownAddressType(String),
|
||||||
|
/// Address's network differs from required one.
|
||||||
|
NetworkValidation {
|
||||||
|
/// Network that was required.
|
||||||
|
required: Network,
|
||||||
|
/// Network on which the address was found to be valid.
|
||||||
|
found: Network
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
impl fmt::Display for Error {
|
||||||
|
@ -105,6 +113,7 @@ impl fmt::Display for Error {
|
||||||
Error::ExcessiveScriptSize => write!(f, "script size exceed 520 bytes"),
|
Error::ExcessiveScriptSize => write!(f, "script size exceed 520 bytes"),
|
||||||
Error::UnrecognizedScript => write!(f, "script is not a p2pkh, p2sh or witness program"),
|
Error::UnrecognizedScript => write!(f, "script is not a p2pkh, p2sh or witness program"),
|
||||||
Error::UnknownAddressType(ref s) => write!(f, "unknown address type: '{}' is either invalid or not supported in rust-bitcoin", s),
|
Error::UnknownAddressType(ref s) => write!(f, "unknown address type: '{}' is either invalid or not supported in rust-bitcoin", s),
|
||||||
|
Error::NetworkValidation { required, found } => write!(f, "address's network {} is different from required {}", found, required),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -128,7 +137,8 @@ impl std::error::Error for Error {
|
||||||
| UncompressedPubkey
|
| UncompressedPubkey
|
||||||
| ExcessiveScriptSize
|
| ExcessiveScriptSize
|
||||||
| UnrecognizedScript
|
| UnrecognizedScript
|
||||||
| UnknownAddressType(_) => None,
|
| UnknownAddressType(_)
|
||||||
|
| NetworkValidation { .. } => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -553,8 +563,108 @@ impl<'a> fmt::Display for AddressEncoding<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mod sealed {
|
||||||
|
pub trait NetworkValidation {}
|
||||||
|
impl NetworkValidation for super::NetworkChecked {}
|
||||||
|
impl NetworkValidation for super::NetworkUnchecked {}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Marker of status of address's network validation. See section [*Parsing addresses*](Address#parsing-addresses)
|
||||||
|
/// on [`Address`] for details.
|
||||||
|
pub trait NetworkValidation: sealed::NetworkValidation {}
|
||||||
|
|
||||||
|
/// Marker that address's network has been successfully validated. See section [*Parsing addresses*](Address#parsing-addresses)
|
||||||
|
/// on [`Address`] for details.
|
||||||
|
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
|
pub enum NetworkChecked {}
|
||||||
|
|
||||||
|
/// Marker that address's network has not yet been validated. See section [*Parsing addresses*](Address#parsing-addresses)
|
||||||
|
/// on [`Address`] for details.
|
||||||
|
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
|
pub enum NetworkUnchecked {}
|
||||||
|
|
||||||
|
impl NetworkValidation for NetworkChecked {}
|
||||||
|
impl NetworkValidation for NetworkUnchecked {}
|
||||||
|
|
||||||
/// A Bitcoin address.
|
/// A Bitcoin address.
|
||||||
///
|
///
|
||||||
|
/// ### Parsing addresses
|
||||||
|
///
|
||||||
|
/// When parsing string as an address, one has to pay attention to the network, on which the parsed
|
||||||
|
/// address is supposed to be valid. For the purpose of this validation, `Address` has
|
||||||
|
/// [`is_valid_for_network`](Address<NetworkUnchecked>::is_valid_for_network) method. In order to provide more safety,
|
||||||
|
/// enforced by compiler, `Address` also contains a special marker type, which indicates whether network of the parsed
|
||||||
|
/// address has been checked. This marker type will prevent from calling certain functions unless the network
|
||||||
|
/// verification has been successfully completed.
|
||||||
|
///
|
||||||
|
/// The result of parsing an address is `Address<NetworkUnchecked>` suggesting that network of the parsed address
|
||||||
|
/// has not yet been verified. To perform this verification, method [`require_network`](Address<NetworkUnchecked>::require_network)
|
||||||
|
/// can be called, providing network on which the address is supposed to be valid. If the verification succeeds,
|
||||||
|
/// `Address<NetworkChecked>` is returned.
|
||||||
|
///
|
||||||
|
/// The types `Address` and `Address<NetworkChecked>` are synonymous, i. e. they can be used interchangeably.
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// use std::str::FromStr;
|
||||||
|
/// use bitcoin::{Address, Network};
|
||||||
|
/// use bitcoin::address::{NetworkUnchecked, NetworkChecked};
|
||||||
|
///
|
||||||
|
/// // variant 1
|
||||||
|
/// let address: Address<NetworkUnchecked> = "32iVBEu4dxkUQk9dJbZUiBiQdmypcEyJRf".parse().unwrap();
|
||||||
|
/// let address: Address<NetworkChecked> = address.require_network(Network::Bitcoin).unwrap();
|
||||||
|
///
|
||||||
|
/// // variant 2
|
||||||
|
/// let address: Address = Address::from_str("32iVBEu4dxkUQk9dJbZUiBiQdmypcEyJRf").unwrap()
|
||||||
|
/// .require_network(Network::Bitcoin).unwrap();
|
||||||
|
///
|
||||||
|
/// // variant 3
|
||||||
|
/// let address: Address<NetworkChecked> = "32iVBEu4dxkUQk9dJbZUiBiQdmypcEyJRf".parse::<Address<_>>()
|
||||||
|
/// .unwrap().require_network(Network::Bitcoin).unwrap();
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// ### Formatting addresses
|
||||||
|
///
|
||||||
|
/// To format address into its textual representation, both `Debug` (for usage in programmer-facing,
|
||||||
|
/// debugging context) and `Display` (for user-facing output) can be used, with the following caveats:
|
||||||
|
///
|
||||||
|
/// 1. `Display` is implemented only for `Address<NetworkChecked>`:
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # use std::str::FromStr;
|
||||||
|
/// # use bitcoin::address::{Address, NetworkChecked};
|
||||||
|
/// let address: Address<NetworkChecked> = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM")
|
||||||
|
/// .unwrap().assume_checked();
|
||||||
|
/// assert_eq!(address.to_string(), "132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM");
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// ```compile_fail
|
||||||
|
/// # use std::str::FromStr;
|
||||||
|
/// # use bitcoin::address::{Address, NetworkChecked};
|
||||||
|
/// let address: Address<NetworkUnchecked> = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM")
|
||||||
|
/// .unwrap();
|
||||||
|
/// let s = address.to_string(); // does not compile
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// 2. `Debug` on `Address<NetworkUnchecked>` does not produce clean address but address wrapped by
|
||||||
|
/// an indicator that its network has not been checked. This is to encourage programmer to properly
|
||||||
|
/// check the network and use `Display` in user-facing context.
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # use std::str::FromStr;
|
||||||
|
/// # use bitcoin::address::{Address, NetworkUnchecked};
|
||||||
|
/// let address: Address<NetworkUnchecked> = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM")
|
||||||
|
/// .unwrap();
|
||||||
|
/// assert_eq!(format!("{:?}", address), "Address<NetworkUnchecked>(132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM)");
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # use std::str::FromStr;
|
||||||
|
/// # use bitcoin::address::{Address, NetworkChecked};
|
||||||
|
/// let address: Address<NetworkChecked> = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM")
|
||||||
|
/// .unwrap().assume_checked();
|
||||||
|
/// assert_eq!(format!("{:?}", address), "132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM");
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
/// ### Relevant BIPs
|
/// ### Relevant BIPs
|
||||||
///
|
///
|
||||||
/// * [BIP13 - Address Format for pay-to-script-hash](https://github.com/bitcoin/bips/blob/master/bip-0013.mediawiki)
|
/// * [BIP13 - Address Format for pay-to-script-hash](https://github.com/bitcoin/bips/blob/master/bip-0013.mediawiki)
|
||||||
|
@ -564,87 +674,54 @@ impl<'a> fmt::Display for AddressEncoding<'a> {
|
||||||
/// * [BIP341 - Taproot: SegWit version 1 spending rules](https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki)
|
/// * [BIP341 - Taproot: SegWit version 1 spending rules](https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki)
|
||||||
/// * [BIP350 - Bech32m format for v1+ witness addresses](https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki)
|
/// * [BIP350 - Bech32m format for v1+ witness addresses](https://github.com/bitcoin/bips/blob/master/bip-0350.mediawiki)
|
||||||
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
pub struct Address {
|
pub struct Address<V = NetworkChecked>
|
||||||
|
where
|
||||||
|
V: NetworkValidation,
|
||||||
|
{
|
||||||
/// The type of the address.
|
/// The type of the address.
|
||||||
pub payload: Payload,
|
pub payload: Payload,
|
||||||
/// The network on which this address is usable.
|
/// The network on which this address is usable.
|
||||||
pub network: Network,
|
pub network: Network,
|
||||||
|
/// Marker of the status of network validation.
|
||||||
|
_validation: PhantomData<V>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "serde")]
|
#[cfg(feature = "serde")]
|
||||||
crate::serde_utils::serde_string_impl!(Address, "a Bitcoin address");
|
struct DisplayUnchecked<'a>(&'a Address<NetworkUnchecked>);
|
||||||
|
|
||||||
impl Address {
|
#[cfg(feature = "serde")]
|
||||||
/// Creates a pay to (compressed) public key hash address from a public key.
|
impl fmt::Display for DisplayUnchecked<'_> {
|
||||||
///
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { self.0.fmt_internal(fmt) }
|
||||||
/// This is the preferred non-witness type address.
|
}
|
||||||
#[inline]
|
|
||||||
pub fn p2pkh(pk: &PublicKey, network: Network) -> Address {
|
#[cfg(feature = "serde")]
|
||||||
Address { network, payload: Payload::p2pkh(pk) }
|
crate::serde_utils::serde_string_serialize_impl!(Address, "a Bitcoin address");
|
||||||
}
|
|
||||||
|
#[cfg(feature = "serde")]
|
||||||
/// Creates a pay to script hash P2SH address from a script.
|
crate::serde_utils::serde_string_deserialize_impl!(Address<NetworkUnchecked>, "a Bitcoin address");
|
||||||
///
|
|
||||||
/// This address type was introduced with BIP16 and is the popular type to implement multi-sig
|
#[cfg(feature = "serde")]
|
||||||
/// these days.
|
impl serde::Serialize for Address<NetworkUnchecked> {
|
||||||
#[inline]
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
pub fn p2sh(script: &script::Script, network: Network) -> Result<Address, Error> {
|
where
|
||||||
Ok(Address { network, payload: Payload::p2sh(script)? })
|
S: serde::Serializer,
|
||||||
}
|
{
|
||||||
|
serializer.collect_str(&DisplayUnchecked(self))
|
||||||
/// Creates a witness pay to public key address from a public key.
|
|
||||||
///
|
|
||||||
/// This is the native segwit address type for an output redeemable with a single signature.
|
|
||||||
///
|
|
||||||
/// # Errors
|
|
||||||
/// Will only return an error if an uncompressed public key is provided.
|
|
||||||
pub fn p2wpkh(pk: &PublicKey, network: Network) -> Result<Address, Error> {
|
|
||||||
Ok(Address { network, payload: Payload::p2wpkh(pk)? })
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a pay to script address that embeds a witness pay to public key.
|
|
||||||
///
|
|
||||||
/// This is a segwit address type that looks familiar (as p2sh) to legacy clients.
|
|
||||||
///
|
|
||||||
/// # Errors
|
|
||||||
/// Will only return an Error if an uncompressed public key is provided.
|
|
||||||
pub fn p2shwpkh(pk: &PublicKey, network: Network) -> Result<Address, Error> {
|
|
||||||
Ok(Address { network, payload: Payload::p2shwpkh(pk)? })
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a witness pay to script hash address.
|
|
||||||
pub fn p2wsh(script: &script::Script, network: Network) -> Address {
|
|
||||||
Address { network, payload: Payload::p2wsh(script) }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a pay to script address that embeds a witness pay to script hash address.
|
|
||||||
///
|
|
||||||
/// This is a segwit address type that looks familiar (as p2sh) to legacy clients.
|
|
||||||
pub fn p2shwsh(script: &script::Script, network: Network) -> Address {
|
|
||||||
Address { network, payload: Payload::p2shwsh(script) }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a pay to taproot address from an untweaked key.
|
|
||||||
pub fn p2tr<C: Verification>(
|
|
||||||
secp: &Secp256k1<C>,
|
|
||||||
internal_key: UntweakedPublicKey,
|
|
||||||
merkle_root: Option<TapNodeHash>,
|
|
||||||
network: Network,
|
|
||||||
) -> Address {
|
|
||||||
Address { network, payload: Payload::p2tr(secp, internal_key, merkle_root) }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates a pay to taproot address from a pre-tweaked output key.
|
|
||||||
///
|
|
||||||
/// This method is not recommended for use, [`Address::p2tr()`] should be used where possible.
|
|
||||||
pub fn p2tr_tweaked(output_key: TweakedPublicKey, network: Network) -> Address {
|
|
||||||
Address { network, payload: Payload::p2tr_tweaked(output_key) }
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Methods on [`Address`] that can be called on both `Address<NetworkChecked>` and
|
||||||
|
/// `Address<NetworkUnchecked>`.
|
||||||
|
impl<V: NetworkValidation> Address<V> {
|
||||||
/// Gets the address type of the address.
|
/// Gets the address type of the address.
|
||||||
///
|
///
|
||||||
|
/// This method is publicly available as [`address_type`](Address<NetworkChecked>::address_type)
|
||||||
|
/// on `Address<NetworkChecked>` but internally can be called on `Address<NetworkUnchecked>` as
|
||||||
|
/// `address_type_internal`.
|
||||||
|
///
|
||||||
/// # Returns
|
/// # Returns
|
||||||
/// None if unknown, non-standard or related to the future witness version.
|
/// None if unknown, non-standard or related to the future witness version.
|
||||||
pub fn address_type(&self) -> Option<AddressType> {
|
fn address_type_internal(&self) -> Option<AddressType> {
|
||||||
match self.payload {
|
match self.payload {
|
||||||
Payload::PubkeyHash(_) => Some(AddressType::P2pkh),
|
Payload::PubkeyHash(_) => Some(AddressType::P2pkh),
|
||||||
Payload::ScriptHash(_) => Some(AddressType::P2sh),
|
Payload::ScriptHash(_) => Some(AddressType::P2sh),
|
||||||
|
@ -663,6 +740,114 @@ impl Address {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Format the address for the usage by `Debug` and `Display` implementations.
|
||||||
|
fn fmt_internal(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
let p2pkh_prefix = match self.network {
|
||||||
|
Network::Bitcoin => PUBKEY_ADDRESS_PREFIX_MAIN,
|
||||||
|
Network::Testnet | Network::Signet | Network::Regtest => PUBKEY_ADDRESS_PREFIX_TEST,
|
||||||
|
};
|
||||||
|
let p2sh_prefix = match self.network {
|
||||||
|
Network::Bitcoin => SCRIPT_ADDRESS_PREFIX_MAIN,
|
||||||
|
Network::Testnet | Network::Signet | Network::Regtest => SCRIPT_ADDRESS_PREFIX_TEST,
|
||||||
|
};
|
||||||
|
let bech32_hrp = match self.network {
|
||||||
|
Network::Bitcoin => "bc",
|
||||||
|
Network::Testnet | Network::Signet => "tb",
|
||||||
|
Network::Regtest => "bcrt",
|
||||||
|
};
|
||||||
|
let encoding =
|
||||||
|
AddressEncoding { payload: &self.payload, p2pkh_prefix, p2sh_prefix, bech32_hrp };
|
||||||
|
|
||||||
|
use fmt::Display;
|
||||||
|
|
||||||
|
encoding.fmt(fmt)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create new address from given components, infering the network validation
|
||||||
|
/// marker type of the address.
|
||||||
|
#[inline]
|
||||||
|
pub fn new(network: Network, payload: Payload) -> Address<V> {
|
||||||
|
Address { network, payload, _validation: PhantomData }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Methods and functions that can be called only on `Address<NetworkChecked>`.
|
||||||
|
impl Address {
|
||||||
|
/// Creates a pay to (compressed) public key hash address from a public key.
|
||||||
|
///
|
||||||
|
/// This is the preferred non-witness type address.
|
||||||
|
#[inline]
|
||||||
|
pub fn p2pkh(pk: &PublicKey, network: Network) -> Address {
|
||||||
|
Address::new(network, Payload::p2pkh(pk))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a pay to script hash P2SH address from a script.
|
||||||
|
///
|
||||||
|
/// This address type was introduced with BIP16 and is the popular type to implement multi-sig
|
||||||
|
/// these days.
|
||||||
|
#[inline]
|
||||||
|
pub fn p2sh(script: &script::Script, network: Network) -> Result<Address, Error> {
|
||||||
|
Ok(Address::new(network, Payload::p2sh(script)?))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a witness pay to public key address from a public key.
|
||||||
|
///
|
||||||
|
/// This is the native segwit address type for an output redeemable with a single signature.
|
||||||
|
///
|
||||||
|
/// # Errors
|
||||||
|
/// Will only return an error if an uncompressed public key is provided.
|
||||||
|
pub fn p2wpkh(pk: &PublicKey, network: Network) -> Result<Address, Error> {
|
||||||
|
Ok(Address::new(network, Payload::p2wpkh(pk)?))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a pay to script address that embeds a witness pay to public key.
|
||||||
|
///
|
||||||
|
/// This is a segwit address type that looks familiar (as p2sh) to legacy clients.
|
||||||
|
///
|
||||||
|
/// # Errors
|
||||||
|
/// Will only return an Error if an uncompressed public key is provided.
|
||||||
|
pub fn p2shwpkh(pk: &PublicKey, network: Network) -> Result<Address, Error> {
|
||||||
|
Ok(Address::new(network, Payload::p2shwpkh(pk)?))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a witness pay to script hash address.
|
||||||
|
pub fn p2wsh(script: &script::Script, network: Network) -> Address {
|
||||||
|
Address::new(network, Payload::p2wsh(script))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a pay to script address that embeds a witness pay to script hash address.
|
||||||
|
///
|
||||||
|
/// This is a segwit address type that looks familiar (as p2sh) to legacy clients.
|
||||||
|
pub fn p2shwsh(script: &script::Script, network: Network) -> Address {
|
||||||
|
Address::new(network, Payload::p2shwsh(script))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a pay to taproot address from an untweaked key.
|
||||||
|
pub fn p2tr<C: Verification>(
|
||||||
|
secp: &Secp256k1<C>,
|
||||||
|
internal_key: UntweakedPublicKey,
|
||||||
|
merkle_root: Option<TapNodeHash>,
|
||||||
|
network: Network,
|
||||||
|
) -> Address {
|
||||||
|
Address::new(network, Payload::p2tr(secp, internal_key, merkle_root))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Creates a pay to taproot address from a pre-tweaked output key.
|
||||||
|
///
|
||||||
|
/// This method is not recommended for use, [`Address::p2tr()`] should be used where possible.
|
||||||
|
pub fn p2tr_tweaked(output_key: TweakedPublicKey, network: Network) -> Address {
|
||||||
|
Address::new(network, Payload::p2tr_tweaked(output_key))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Gets the address type of the address.
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
/// None if unknown, non-standard or related to the future witness version.
|
||||||
|
#[inline]
|
||||||
|
pub fn address_type(&self) -> Option<AddressType> {
|
||||||
|
self.address_type_internal()
|
||||||
|
}
|
||||||
|
|
||||||
/// Checks whether or not the address is following Bitcoin standardness rules.
|
/// Checks whether or not the address is following Bitcoin standardness rules.
|
||||||
///
|
///
|
||||||
/// SegWit addresses with unassigned witness versions or non-standard program sizes are
|
/// SegWit addresses with unassigned witness versions or non-standard program sizes are
|
||||||
|
@ -671,7 +856,7 @@ impl Address {
|
||||||
|
|
||||||
/// Constructs an [`Address`] from an output script (`scriptPubkey`).
|
/// Constructs an [`Address`] from an output script (`scriptPubkey`).
|
||||||
pub fn from_script(script: &script::Script, network: Network) -> Result<Address, Error> {
|
pub fn from_script(script: &script::Script, network: Network) -> Result<Address, Error> {
|
||||||
Ok(Address { payload: Payload::from_script(script)?, network })
|
Ok(Address::new(network, Payload::from_script(script)?))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Generates a script pubkey spending to this address.
|
/// Generates a script pubkey spending to this address.
|
||||||
|
@ -692,40 +877,6 @@ impl Address {
|
||||||
format!("{}:{:#}", schema, self)
|
format!("{}:{:#}", schema, self)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parsed addresses do not always have *one* network. The problem is that legacy testnet,
|
|
||||||
/// regtest and signet addresse use the same prefix instead of multiple different ones. When
|
|
||||||
/// parsing, such addresses are always assumed to be testnet addresses (the same is true for
|
|
||||||
/// bech32 signet addresses). So if one wants to check if an address belongs to a certain
|
|
||||||
/// network a simple comparison is not enough anymore. Instead this function can be used.
|
|
||||||
///
|
|
||||||
/// ```rust
|
|
||||||
/// use bitcoin::{Address, Network};
|
|
||||||
///
|
|
||||||
/// let address: Address = "2N83imGV3gPwBzKJQvWJ7cRUY2SpUyU6A5e".parse().unwrap();
|
|
||||||
/// assert!(address.is_valid_for_network(Network::Testnet));
|
|
||||||
/// assert!(address.is_valid_for_network(Network::Regtest));
|
|
||||||
/// assert!(address.is_valid_for_network(Network::Signet));
|
|
||||||
///
|
|
||||||
/// assert_eq!(address.is_valid_for_network(Network::Bitcoin), false);
|
|
||||||
///
|
|
||||||
/// let address: Address = "32iVBEu4dxkUQk9dJbZUiBiQdmypcEyJRf".parse().unwrap();
|
|
||||||
/// assert!(address.is_valid_for_network(Network::Bitcoin));
|
|
||||||
/// assert_eq!(address.is_valid_for_network(Network::Testnet), false);
|
|
||||||
/// ```
|
|
||||||
pub fn is_valid_for_network(&self, network: Network) -> bool {
|
|
||||||
let is_legacy = match self.address_type() {
|
|
||||||
Some(AddressType::P2pkh) | Some(AddressType::P2sh) => true,
|
|
||||||
_ => false,
|
|
||||||
};
|
|
||||||
|
|
||||||
match (self.network, network) {
|
|
||||||
(a, b) if a == b => true,
|
|
||||||
(Network::Bitcoin, _) | (_, Network::Bitcoin) => false,
|
|
||||||
(Network::Regtest, _) | (_, Network::Regtest) if !is_legacy => false,
|
|
||||||
(Network::Testnet, _) | (Network::Regtest, _) | (Network::Signet, _) => true,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns true if the given pubkey is directly related to the address payload.
|
/// Returns true if the given pubkey is directly related to the address payload.
|
||||||
///
|
///
|
||||||
/// This is determined by directly comparing the address payload with either the
|
/// This is determined by directly comparing the address payload with either the
|
||||||
|
@ -751,26 +902,83 @@ impl Address {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Methods that can be called only on `Address<NetworkUnchecked>`.
|
||||||
|
impl Address<NetworkUnchecked> {
|
||||||
|
/// Parsed addresses do not always have *one* network. The problem is that legacy testnet,
|
||||||
|
/// regtest and signet addresse use the same prefix instead of multiple different ones. When
|
||||||
|
/// parsing, such addresses are always assumed to be testnet addresses (the same is true for
|
||||||
|
/// bech32 signet addresses). So if one wants to check if an address belongs to a certain
|
||||||
|
/// network a simple comparison is not enough anymore. Instead this function can be used.
|
||||||
|
///
|
||||||
|
/// ```rust
|
||||||
|
/// use bitcoin::{Address, Network};
|
||||||
|
/// use bitcoin::address::NetworkUnchecked;
|
||||||
|
///
|
||||||
|
/// let address: Address<NetworkUnchecked> = "2N83imGV3gPwBzKJQvWJ7cRUY2SpUyU6A5e".parse().unwrap();
|
||||||
|
/// assert!(address.is_valid_for_network(Network::Testnet));
|
||||||
|
/// assert!(address.is_valid_for_network(Network::Regtest));
|
||||||
|
/// assert!(address.is_valid_for_network(Network::Signet));
|
||||||
|
///
|
||||||
|
/// assert_eq!(address.is_valid_for_network(Network::Bitcoin), false);
|
||||||
|
///
|
||||||
|
/// let address: Address<NetworkUnchecked> = "32iVBEu4dxkUQk9dJbZUiBiQdmypcEyJRf".parse().unwrap();
|
||||||
|
/// assert!(address.is_valid_for_network(Network::Bitcoin));
|
||||||
|
/// assert_eq!(address.is_valid_for_network(Network::Testnet), false);
|
||||||
|
/// ```
|
||||||
|
pub fn is_valid_for_network(&self, network: Network) -> bool {
|
||||||
|
let is_legacy = match self.address_type_internal() {
|
||||||
|
Some(AddressType::P2pkh) | Some(AddressType::P2sh) => true,
|
||||||
|
_ => false,
|
||||||
|
};
|
||||||
|
|
||||||
|
match (self.network, network) {
|
||||||
|
(a, b) if a == b => true,
|
||||||
|
(Network::Bitcoin, _) | (_, Network::Bitcoin) => false,
|
||||||
|
(Network::Regtest, _) | (_, Network::Regtest) if !is_legacy => false,
|
||||||
|
(Network::Testnet, _) | (Network::Regtest, _) | (Network::Signet, _) => true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Checks whether network of this address is as required.
|
||||||
|
///
|
||||||
|
/// For details about this mechanism, see section [*Parsing addresses*](Address#parsing-addresses)
|
||||||
|
/// on [`Address`].
|
||||||
|
#[inline]
|
||||||
|
pub fn require_network(self, required: Network) -> Result<Address, Error> {
|
||||||
|
if self.is_valid_for_network(required) {
|
||||||
|
Ok(self.assume_checked())
|
||||||
|
} else {
|
||||||
|
Err(Error::NetworkValidation { found: self.network, required })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Marks, without any additional checks, network of this address as checked.
|
||||||
|
///
|
||||||
|
/// Improper use of this method may lead to loss of funds. Reader will most likely prefer
|
||||||
|
/// [`require_network`](Address<NetworkUnchecked>::require_network) as a safe variant.
|
||||||
|
/// For details about this mechanism, see section [*Parsing addresses*](Address#parsing-addresses)
|
||||||
|
/// on [`Address`].
|
||||||
|
#[inline]
|
||||||
|
pub fn assume_checked(self) -> Address {
|
||||||
|
Address::new(self.network, self.payload)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Alternate formatting `{:#}` is used to return uppercase version of bech32 addresses which should
|
// Alternate formatting `{:#}` is used to return uppercase version of bech32 addresses which should
|
||||||
// be used in QR codes, see [`Address::to_qr_uri`].
|
// be used in QR codes, see [`Address::to_qr_uri`].
|
||||||
impl fmt::Display for Address {
|
impl fmt::Display for Address {
|
||||||
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { self.fmt_internal(fmt) }
|
||||||
let p2pkh_prefix = match self.network {
|
}
|
||||||
Network::Bitcoin => PUBKEY_ADDRESS_PREFIX_MAIN,
|
|
||||||
Network::Testnet | Network::Signet | Network::Regtest => PUBKEY_ADDRESS_PREFIX_TEST,
|
impl fmt::Debug for Address {
|
||||||
};
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { self.fmt_internal(f) }
|
||||||
let p2sh_prefix = match self.network {
|
}
|
||||||
Network::Bitcoin => SCRIPT_ADDRESS_PREFIX_MAIN,
|
|
||||||
Network::Testnet | Network::Signet | Network::Regtest => SCRIPT_ADDRESS_PREFIX_TEST,
|
impl fmt::Debug for Address<NetworkUnchecked> {
|
||||||
};
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
let bech32_hrp = match self.network {
|
write!(f, "Address<NetworkUnchecked>(")?;
|
||||||
Network::Bitcoin => "bc",
|
self.fmt_internal(f)?;
|
||||||
Network::Testnet | Network::Signet => "tb",
|
write!(f, ")")
|
||||||
Network::Regtest => "bcrt",
|
|
||||||
};
|
|
||||||
let encoding =
|
|
||||||
AddressEncoding { payload: &self.payload, p2pkh_prefix, p2sh_prefix, bech32_hrp };
|
|
||||||
encoding.fmt(fmt)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -797,10 +1005,11 @@ fn find_bech32_prefix(bech32: &str) -> &str {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FromStr for Address {
|
// Address can be parsed only with NetworkUnchecked.
|
||||||
|
impl FromStr for Address<NetworkUnchecked> {
|
||||||
type Err = Error;
|
type Err = Error;
|
||||||
|
|
||||||
fn from_str(s: &str) -> Result<Address, Error> {
|
fn from_str(s: &str) -> Result<Address<NetworkUnchecked>, Error> {
|
||||||
// try bech32
|
// try bech32
|
||||||
let bech32_network = match find_bech32_prefix(s) {
|
let bech32_network = match find_bech32_prefix(s) {
|
||||||
// note that upper or lowercase is allowed but NOT mixed case
|
// note that upper or lowercase is allowed but NOT mixed case
|
||||||
|
@ -837,7 +1046,7 @@ impl FromStr for Address {
|
||||||
return Err(Error::InvalidBech32Variant { expected, found: variant });
|
return Err(Error::InvalidBech32Variant { expected, found: variant });
|
||||||
}
|
}
|
||||||
|
|
||||||
return Ok(Address { payload: Payload::WitnessProgram { version, program }, network });
|
return Ok(Address::new(network, Payload::WitnessProgram { version, program }));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Base58
|
// Base58
|
||||||
|
@ -861,14 +1070,10 @@ impl FromStr for Address {
|
||||||
x => return Err(Error::Base58(base58::Error::InvalidAddressVersion(x))),
|
x => return Err(Error::Base58(base58::Error::InvalidAddressVersion(x))),
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Address { network, payload })
|
Ok(Address::new(network, payload))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Debug for Address {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Display::fmt(self, f) }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert a byte array of a pubkey hash into a segwit redeem hash
|
/// Convert a byte array of a pubkey hash into a segwit redeem hash
|
||||||
fn segwit_redeem_hash(pubkey_hash: &PubkeyHash) -> crate::hashes::hash160::Hash {
|
fn segwit_redeem_hash(pubkey_hash: &PubkeyHash) -> crate::hashes::hash160::Hash {
|
||||||
let mut sha_engine = sha256::Hash::engine();
|
let mut sha_engine = sha256::Hash::engine();
|
||||||
|
@ -890,7 +1095,7 @@ mod tests {
|
||||||
|
|
||||||
fn roundtrips(addr: &Address) {
|
fn roundtrips(addr: &Address) {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
Address::from_str(&addr.to_string()).unwrap(),
|
Address::from_str(&addr.to_string()).unwrap().assume_checked(),
|
||||||
*addr,
|
*addr,
|
||||||
"string round-trip failed for {}",
|
"string round-trip failed for {}",
|
||||||
addr,
|
addr,
|
||||||
|
@ -905,17 +1110,14 @@ mod tests {
|
||||||
#[cfg(feature = "serde")]
|
#[cfg(feature = "serde")]
|
||||||
{
|
{
|
||||||
let ser = serde_json::to_string(addr).expect("failed to serialize address");
|
let ser = serde_json::to_string(addr).expect("failed to serialize address");
|
||||||
let back: Address = serde_json::from_str(&ser).expect("failed to deserialize address");
|
let back: Address<NetworkUnchecked> = serde_json::from_str(&ser).expect("failed to deserialize address");
|
||||||
assert_eq!(back, *addr, "serde round-trip failed for {}", addr)
|
assert_eq!(back.assume_checked(), *addr, "serde round-trip failed for {}", addr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_p2pkh_address_58() {
|
fn test_p2pkh_address_58() {
|
||||||
let addr = Address {
|
let addr = Address::new(Bitcoin, Payload::PubkeyHash(hex_into!("162c5ea71c0b23f5b9022ef047c4a86470a5b070")));
|
||||||
network: Bitcoin,
|
|
||||||
payload: Payload::PubkeyHash(hex_into!("162c5ea71c0b23f5b9022ef047c4a86470a5b070")),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
addr.script_pubkey(),
|
addr.script_pubkey(),
|
||||||
|
@ -941,10 +1143,7 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_p2sh_address_58() {
|
fn test_p2sh_address_58() {
|
||||||
let addr = Address {
|
let addr = Address::new(Bitcoin, Payload::ScriptHash(hex_into!("162c5ea71c0b23f5b9022ef047c4a86470a5b070")));
|
||||||
network: Bitcoin,
|
|
||||||
payload: Payload::ScriptHash(hex_into!("162c5ea71c0b23f5b9022ef047c4a86470a5b070")),
|
|
||||||
};
|
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
addr.script_pubkey(),
|
addr.script_pubkey(),
|
||||||
|
@ -1029,10 +1228,7 @@ mod tests {
|
||||||
let program = hex!(
|
let program = hex!(
|
||||||
"654f6ea368e0acdfd92976b7c2103a1b26313f430654f6ea368e0acdfd92976b7c2103a1b26313f4"
|
"654f6ea368e0acdfd92976b7c2103a1b26313f430654f6ea368e0acdfd92976b7c2103a1b26313f4"
|
||||||
);
|
);
|
||||||
let addr = Address {
|
let addr = Address::new(Bitcoin, Payload::WitnessProgram { version: WitnessVersion::V13, program });
|
||||||
payload: Payload::WitnessProgram { version: WitnessVersion::V13, program },
|
|
||||||
network: Network::Bitcoin,
|
|
||||||
};
|
|
||||||
roundtrips(&addr);
|
roundtrips(&addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1057,7 +1253,10 @@ mod tests {
|
||||||
("bc1zw508d6qejxtdg4y5r3zarvaryvaxxpcs", None),
|
("bc1zw508d6qejxtdg4y5r3zarvaryvaxxpcs", None),
|
||||||
];
|
];
|
||||||
for (address, expected_type) in &addresses {
|
for (address, expected_type) in &addresses {
|
||||||
let addr = Address::from_str(address).unwrap();
|
let addr = Address::from_str(address)
|
||||||
|
.unwrap()
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet");
|
||||||
assert_eq!(&addr.address_type(), expected_type);
|
assert_eq!(&addr.address_type(), expected_type);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1076,7 +1275,7 @@ mod tests {
|
||||||
("bc1p0xlxvlhemja6c4dqv22uapctqupfhlxm9h8z3k2e72q4k9hcz7vqzk5jj0", "512079be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798")
|
("bc1p0xlxvlhemja6c4dqv22uapctqupfhlxm9h8z3k2e72q4k9hcz7vqzk5jj0", "512079be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798")
|
||||||
];
|
];
|
||||||
for vector in &valid_vectors {
|
for vector in &valid_vectors {
|
||||||
let addr: Address = vector.0.parse().unwrap();
|
let addr: Address = vector.0.parse::<Address<_>>().unwrap().assume_checked();
|
||||||
assert_eq!(&addr.script_pubkey().to_hex_string(), vector.1);
|
assert_eq!(&addr.script_pubkey().to_hex_string(), vector.1);
|
||||||
roundtrips(&addr);
|
roundtrips(&addr);
|
||||||
}
|
}
|
||||||
|
@ -1136,7 +1335,7 @@ mod tests {
|
||||||
"bc1zw508d6qejxtdg4y5r3zarvaryvg6kdaj",
|
"bc1zw508d6qejxtdg4y5r3zarvaryvg6kdaj",
|
||||||
];
|
];
|
||||||
for vector in &invalid_vectors {
|
for vector in &invalid_vectors {
|
||||||
assert!(vector.parse::<Address>().is_err());
|
assert!(vector.parse::<Address<_>>().is_err());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1145,35 +1344,46 @@ mod tests {
|
||||||
fn test_json_serialize() {
|
fn test_json_serialize() {
|
||||||
use serde_json;
|
use serde_json;
|
||||||
|
|
||||||
let addr = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM").unwrap();
|
let addr = Address::from_str("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM").unwrap().assume_checked();
|
||||||
let json = serde_json::to_value(&addr).unwrap();
|
let json = serde_json::to_value(&addr).unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
json,
|
json,
|
||||||
serde_json::Value::String("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM".to_owned())
|
serde_json::Value::String("132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM".to_owned())
|
||||||
);
|
);
|
||||||
let into: Address = serde_json::from_value(json).unwrap();
|
let into: Address = serde_json::from_value::<Address<_>>(json).unwrap().assume_checked();
|
||||||
assert_eq!(addr.to_string(), into.to_string());
|
assert_eq!(addr.to_string(), into.to_string());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
into.script_pubkey(),
|
into.script_pubkey(),
|
||||||
hex_script!("76a914162c5ea71c0b23f5b9022ef047c4a86470a5b07088ac")
|
hex_script!("76a914162c5ea71c0b23f5b9022ef047c4a86470a5b07088ac")
|
||||||
);
|
);
|
||||||
|
|
||||||
let addr = Address::from_str("33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k").unwrap();
|
let addr = Address::from_str("33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k").unwrap().assume_checked();
|
||||||
let json = serde_json::to_value(&addr).unwrap();
|
let json = serde_json::to_value(&addr).unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
json,
|
json,
|
||||||
serde_json::Value::String("33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k".to_owned())
|
serde_json::Value::String("33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k".to_owned())
|
||||||
);
|
);
|
||||||
let into: Address = serde_json::from_value(json).unwrap();
|
let into: Address = serde_json::from_value::<Address<_>>(json).unwrap().assume_checked();
|
||||||
assert_eq!(addr.to_string(), into.to_string());
|
assert_eq!(addr.to_string(), into.to_string());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
into.script_pubkey(),
|
into.script_pubkey(),
|
||||||
hex_script!("a914162c5ea71c0b23f5b9022ef047c4a86470a5b07087")
|
hex_script!("a914162c5ea71c0b23f5b9022ef047c4a86470a5b07087")
|
||||||
);
|
);
|
||||||
|
|
||||||
let addr =
|
let addr: Address<NetworkUnchecked> =
|
||||||
Address::from_str("tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7")
|
Address::from_str("tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7")
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
let json = serde_json::to_value(addr).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
json,
|
||||||
|
serde_json::Value::String(
|
||||||
|
"tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7".to_owned()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
let addr =
|
||||||
|
Address::from_str("tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7")
|
||||||
|
.unwrap().assume_checked();
|
||||||
let json = serde_json::to_value(&addr).unwrap();
|
let json = serde_json::to_value(&addr).unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
json,
|
json,
|
||||||
|
@ -1181,20 +1391,20 @@ mod tests {
|
||||||
"tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7".to_owned()
|
"tb1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3q0sl5k7".to_owned()
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
let into: Address = serde_json::from_value(json).unwrap();
|
let into: Address = serde_json::from_value::<Address<_>>(json).unwrap().assume_checked();
|
||||||
assert_eq!(addr.to_string(), into.to_string());
|
assert_eq!(addr.to_string(), into.to_string());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
into.script_pubkey(),
|
into.script_pubkey(),
|
||||||
hex_script!("00201863143c14c5166804bd19203356da136c985678cd4d27a1b8c6329604903262")
|
hex_script!("00201863143c14c5166804bd19203356da136c985678cd4d27a1b8c6329604903262")
|
||||||
);
|
);
|
||||||
|
|
||||||
let addr = Address::from_str("bcrt1q2nfxmhd4n3c8834pj72xagvyr9gl57n5r94fsl").unwrap();
|
let addr = Address::from_str("bcrt1q2nfxmhd4n3c8834pj72xagvyr9gl57n5r94fsl").unwrap().assume_checked();
|
||||||
let json = serde_json::to_value(&addr).unwrap();
|
let json = serde_json::to_value(&addr).unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
json,
|
json,
|
||||||
serde_json::Value::String("bcrt1q2nfxmhd4n3c8834pj72xagvyr9gl57n5r94fsl".to_owned())
|
serde_json::Value::String("bcrt1q2nfxmhd4n3c8834pj72xagvyr9gl57n5r94fsl".to_owned())
|
||||||
);
|
);
|
||||||
let into: Address = serde_json::from_value(json).unwrap();
|
let into: Address = serde_json::from_value::<Address<_>>(json).unwrap().assume_checked();
|
||||||
assert_eq!(addr.to_string(), into.to_string());
|
assert_eq!(addr.to_string(), into.to_string());
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
into.script_pubkey(),
|
into.script_pubkey(),
|
||||||
|
@ -1207,7 +1417,7 @@ mod tests {
|
||||||
for el in
|
for el in
|
||||||
["132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM", "33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k"].iter()
|
["132F25rTsvBdp9JzLLBHP5mvGY66i1xdiM", "33iFwdLuRpW1uK1RTRqsoi8rR4NpDzk66k"].iter()
|
||||||
{
|
{
|
||||||
let addr = Address::from_str(el).unwrap();
|
let addr = Address::from_str(el).unwrap().require_network(Network::Bitcoin).expect("mainnet");
|
||||||
assert_eq!(addr.to_qr_uri(), format!("bitcoin:{}", el));
|
assert_eq!(addr.to_qr_uri(), format!("bitcoin:{}", el));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1217,7 +1427,7 @@ mod tests {
|
||||||
]
|
]
|
||||||
.iter()
|
.iter()
|
||||||
{
|
{
|
||||||
let addr = Address::from_str(el).unwrap();
|
let addr = Address::from_str(el).unwrap().assume_checked();
|
||||||
assert_eq!(addr.to_qr_uri(), format!("BITCOIN:{}", el.to_ascii_uppercase()));
|
assert_eq!(addr.to_qr_uri(), format!("BITCOIN:{}", el.to_ascii_uppercase()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1248,7 +1458,7 @@ mod tests {
|
||||||
.filter(|ec| ec.contains(addr_net))
|
.filter(|ec| ec.contains(addr_net))
|
||||||
.flat_map(|ec| ec.iter())
|
.flat_map(|ec| ec.iter())
|
||||||
{
|
{
|
||||||
let addr = Address { payload: pl.clone(), network: *addr_net };
|
let addr = Address::new(*addr_net, pl.clone());
|
||||||
assert!(addr.is_valid_for_network(*valid_net));
|
assert!(addr.is_valid_for_network(*valid_net));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1257,7 +1467,7 @@ mod tests {
|
||||||
.filter(|ec| !ec.contains(addr_net))
|
.filter(|ec| !ec.contains(addr_net))
|
||||||
.flat_map(|ec| ec.iter())
|
.flat_map(|ec| ec.iter())
|
||||||
{
|
{
|
||||||
let addr = Address { payload: pl.clone(), network: *addr_net };
|
let addr = Address::new(*addr_net, pl.clone());
|
||||||
assert!(!addr.is_valid_for_network(*invalid_net));
|
assert!(!addr.is_valid_for_network(*invalid_net));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1288,7 +1498,10 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_is_related_to_pubkey_p2wpkh() {
|
fn test_is_related_to_pubkey_p2wpkh() {
|
||||||
let address_string = "bc1qhvd6suvqzjcu9pxjhrwhtrlj85ny3n2mqql5w4";
|
let address_string = "bc1qhvd6suvqzjcu9pxjhrwhtrlj85ny3n2mqql5w4";
|
||||||
let address = Address::from_str(address_string).expect("address");
|
let address = Address::from_str(address_string)
|
||||||
|
.expect("address")
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet");
|
||||||
|
|
||||||
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
||||||
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
||||||
|
@ -1306,7 +1519,10 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_is_related_to_pubkey_p2shwpkh() {
|
fn test_is_related_to_pubkey_p2shwpkh() {
|
||||||
let address_string = "3EZQk4F8GURH5sqVMLTFisD17yNeKa7Dfs";
|
let address_string = "3EZQk4F8GURH5sqVMLTFisD17yNeKa7Dfs";
|
||||||
let address = Address::from_str(address_string).expect("address");
|
let address = Address::from_str(address_string)
|
||||||
|
.expect("address")
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet");
|
||||||
|
|
||||||
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
||||||
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
||||||
|
@ -1324,7 +1540,10 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_is_related_to_pubkey_p2pkh() {
|
fn test_is_related_to_pubkey_p2pkh() {
|
||||||
let address_string = "1J4LVanjHMu3JkXbVrahNuQCTGCRRgfWWx";
|
let address_string = "1J4LVanjHMu3JkXbVrahNuQCTGCRRgfWWx";
|
||||||
let address = Address::from_str(address_string).expect("address");
|
let address = Address::from_str(address_string)
|
||||||
|
.expect("address")
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet");
|
||||||
|
|
||||||
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
let pubkey_string = "0347ff3dacd07a1f43805ec6808e801505a6e18245178609972a68afbc2777ff2b";
|
||||||
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
||||||
|
@ -1342,7 +1561,10 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_is_related_to_pubkey_p2pkh_uncompressed_key() {
|
fn test_is_related_to_pubkey_p2pkh_uncompressed_key() {
|
||||||
let address_string = "msvS7KzhReCDpQEJaV2hmGNvuQqVUDuC6p";
|
let address_string = "msvS7KzhReCDpQEJaV2hmGNvuQqVUDuC6p";
|
||||||
let address = Address::from_str(address_string).expect("address");
|
let address = Address::from_str(address_string)
|
||||||
|
.expect("address")
|
||||||
|
.require_network(Network::Testnet)
|
||||||
|
.expect("testnet");
|
||||||
|
|
||||||
let pubkey_string = "04e96e22004e3db93530de27ccddfdf1463975d2138ac018fc3e7ba1a2e5e0aad8e424d0b55e2436eb1d0dcd5cb2b8bcc6d53412c22f358de57803a6a655fbbd04";
|
let pubkey_string = "04e96e22004e3db93530de27ccddfdf1463975d2138ac018fc3e7ba1a2e5e0aad8e424d0b55e2436eb1d0dcd5cb2b8bcc6d53412c22f358de57803a6a655fbbd04";
|
||||||
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
let pubkey = PublicKey::from_str(pubkey_string).expect("pubkey");
|
||||||
|
@ -1369,6 +1591,8 @@ mod tests {
|
||||||
address,
|
address,
|
||||||
Address::from_str("bc1pgllnmtxs0g058qz7c6qgaqq4qknwrqj9z7rqn9e2dzhmcfmhlu4sfadf5e")
|
Address::from_str("bc1pgllnmtxs0g058qz7c6qgaqq4qknwrqj9z7rqn9e2dzhmcfmhlu4sfadf5e")
|
||||||
.expect("address")
|
.expect("address")
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet")
|
||||||
);
|
);
|
||||||
|
|
||||||
let result = address.is_related_to_pubkey(&pubkey);
|
let result = address.is_related_to_pubkey(&pubkey);
|
||||||
|
@ -1393,6 +1617,8 @@ mod tests {
|
||||||
address,
|
address,
|
||||||
Address::from_str("bc1pgllnmtxs0g058qz7c6qgaqq4qknwrqj9z7rqn9e2dzhmcfmhlu4sfadf5e")
|
Address::from_str("bc1pgllnmtxs0g058qz7c6qgaqq4qknwrqj9z7rqn9e2dzhmcfmhlu4sfadf5e")
|
||||||
.expect("address")
|
.expect("address")
|
||||||
|
.require_network(Network::Bitcoin)
|
||||||
|
.expect("mainnet")
|
||||||
);
|
);
|
||||||
|
|
||||||
let result = address.is_related_to_xonly_pubkey(&xonly_pubkey);
|
let result = address.is_related_to_xonly_pubkey(&xonly_pubkey);
|
||||||
|
|
|
@ -308,15 +308,27 @@ pub mod hex_bytes {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! serde_string_impl {
|
macro_rules! serde_string_serialize_impl {
|
||||||
($name:ident, $expecting:literal) => {
|
($name:ty, $expecting:literal) => {
|
||||||
|
impl $crate::serde::Serialize for $name {
|
||||||
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||||
|
where
|
||||||
|
S: $crate::serde::Serializer,
|
||||||
|
{
|
||||||
|
serializer.collect_str(&self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! serde_string_deserialize_impl {
|
||||||
|
($name:ty, $expecting:literal) => {
|
||||||
impl<'de> $crate::serde::Deserialize<'de> for $name {
|
impl<'de> $crate::serde::Deserialize<'de> for $name {
|
||||||
fn deserialize<D>(deserializer: D) -> Result<$name, D::Error>
|
fn deserialize<D>(deserializer: D) -> Result<$name, D::Error>
|
||||||
where
|
where
|
||||||
D: $crate::serde::de::Deserializer<'de>,
|
D: $crate::serde::de::Deserializer<'de>,
|
||||||
{
|
{
|
||||||
use core::fmt::{self, Formatter};
|
use core::fmt::{self, Formatter};
|
||||||
use core::str::FromStr;
|
|
||||||
|
|
||||||
struct Visitor;
|
struct Visitor;
|
||||||
impl<'de> $crate::serde::de::Visitor<'de> for Visitor {
|
impl<'de> $crate::serde::de::Visitor<'de> for Visitor {
|
||||||
|
@ -330,25 +342,25 @@ macro_rules! serde_string_impl {
|
||||||
where
|
where
|
||||||
E: $crate::serde::de::Error,
|
E: $crate::serde::de::Error,
|
||||||
{
|
{
|
||||||
$name::from_str(v).map_err(E::custom)
|
v.parse::<$name>().map_err(E::custom)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
deserializer.deserialize_str(Visitor)
|
deserializer.deserialize_str(Visitor)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
impl $crate::serde::Serialize for $name {
|
macro_rules! serde_string_impl {
|
||||||
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
($name:ty, $expecting:literal) => {
|
||||||
where
|
$crate::serde_utils::serde_string_deserialize_impl!($name, $expecting);
|
||||||
S: $crate::serde::Serializer,
|
$crate::serde_utils::serde_string_serialize_impl!($name, $expecting);
|
||||||
{
|
|
||||||
serializer.collect_str(&self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
pub(crate) use serde_string_impl;
|
pub(crate) use serde_string_impl;
|
||||||
|
pub(crate) use serde_string_serialize_impl;
|
||||||
|
pub(crate) use serde_string_deserialize_impl;
|
||||||
|
|
||||||
/// A combination macro where the human-readable serialization is done like
|
/// A combination macro where the human-readable serialization is done like
|
||||||
/// serde_string_impl and the non-human-readable impl is done as a struct.
|
/// serde_string_impl and the non-human-readable impl is done as a struct.
|
||||||
|
|
|
@ -1449,7 +1449,7 @@ mod test {
|
||||||
let expected_spk =
|
let expected_spk =
|
||||||
ScriptBuf::from_str(arr["expected"]["scriptPubKey"].as_str().unwrap()).unwrap();
|
ScriptBuf::from_str(arr["expected"]["scriptPubKey"].as_str().unwrap()).unwrap();
|
||||||
let expected_addr =
|
let expected_addr =
|
||||||
Address::from_str(arr["expected"]["bip350Address"].as_str().unwrap()).unwrap();
|
Address::from_str(arr["expected"]["bip350Address"].as_str().unwrap()).unwrap().assume_checked();
|
||||||
|
|
||||||
let tweak = TapTweakHash::from_key_and_tweak(internal_key, merkle_root);
|
let tweak = TapTweakHash::from_key_and_tweak(internal_key, merkle_root);
|
||||||
let (output_key, _parity) = internal_key.tap_tweak(secp, merkle_root);
|
let (output_key, _parity) = internal_key.tap_tweak(secp, merkle_root);
|
||||||
|
|
Loading…
Reference in New Issue