From 28337442dbe66162866991a5f3097ebd4c59a9be Mon Sep 17 00:00:00 2001 From: Kris Nuttycombe Date: Fri, 16 Feb 2024 14:49:47 -0700 Subject: [PATCH] `zcash_address`: Add support for ZIP 316, Revision 1 --- components/zcash_address/CHANGELOG.md | 9 ++ components/zcash_address/src/encoding.rs | 17 ++- components/zcash_address/src/kind/unified.rs | 118 +++++++++++---- .../zcash_address/src/kind/unified/address.rs | 142 ++++++++++++------ .../zcash_address/src/kind/unified/fvk.rs | 83 ++++++---- .../zcash_address/src/kind/unified/ivk.rs | 79 ++++++---- components/zcash_address/src/test_vectors.rs | 10 +- zcash_keys/src/address.rs | 86 ++++++----- zcash_keys/src/keys.rs | 7 +- 9 files changed, 370 insertions(+), 181 deletions(-) diff --git a/components/zcash_address/CHANGELOG.md b/components/zcash_address/CHANGELOG.md index 39f761d54c..ec1be0182d 100644 --- a/components/zcash_address/CHANGELOG.md +++ b/components/zcash_address/CHANGELOG.md @@ -13,11 +13,20 @@ and this library adheres to Rust's notion of - `MetadataTypecode` - `Item` - `MetadataItem` + - `Revision` + - `Container::revision` + - `Address::revision` + - `Ufvk::revision` + - `Uivk::revision` ### Changed - `zcash_address::unified`: - `Typecode` has changed. Instead of having a variant for each receiver type, it now has two variants, `Typecode::Data` and `Typecode::Metadata`. + - `Encoding::try_from_items` now takes an additional `Revision` argument. + - `Address::try_from_items` now takes an additional `Revision` argument. + - `Ufvk::try_from_items` now takes an additional `Revision` argument. + - `Uivk::try_from_items` now takes an additional `Revision` argument. ### Removed diff --git a/components/zcash_address/src/encoding.rs b/components/zcash_address/src/encoding.rs index 305ba2c2c4..4c29c8628e 100644 --- a/components/zcash_address/src/encoding.rs +++ b/components/zcash_address/src/encoding.rs @@ -164,7 +164,7 @@ mod tests { use super::*; use crate::{ kind::unified, - unified::{Item, Receiver}, + unified::{Item, Receiver, Revision}, }; fn encoding(encoded: &str, decoded: ZcashAddress) { @@ -215,21 +215,30 @@ mod tests { "u1qpatys4zruk99pg59gcscrt7y6akvl9vrhcfyhm9yxvxz7h87q6n8cgrzzpe9zru68uq39uhmlpp5uefxu0su5uqyqfe5zp3tycn0ecl", ZcashAddress { net: Network::Main, - kind: AddressKind::Unified(unified::Address(vec![Item::Data(Receiver::Sapling([0; 43]))])), + kind: AddressKind::Unified(unified::Address { + revision: Revision::R0, + receivers: vec![Item::Data(Receiver::Sapling([0; 43]))] + }), }, ); encoding( "utest10c5kutapazdnf8ztl3pu43nkfsjx89fy3uuff8tsmxm6s86j37pe7uz94z5jhkl49pqe8yz75rlsaygexk6jpaxwx0esjr8wm5ut7d5s", ZcashAddress { net: Network::Test, - kind: AddressKind::Unified(unified::Address(vec![Item::Data(Receiver::Sapling([0; 43]))])), + kind: AddressKind::Unified(unified::Address { + revision: Revision::R0, + receivers: vec![Item::Data(Receiver::Sapling([0; 43]))] + }), }, ); encoding( "uregtest15xk7vj4grjkay6mnfl93dhsflc2yeunhxwdh38rul0rq3dfhzzxgm5szjuvtqdha4t4p2q02ks0jgzrhjkrav70z9xlvq0plpcjkd5z3", ZcashAddress { net: Network::Regtest, - kind: AddressKind::Unified(unified::Address(vec![Item::Data(Receiver::Sapling([0; 43]))])), + kind: AddressKind::Unified(unified::Address { + revision: Revision::R0, + receivers: vec![Item::Data(Receiver::Sapling([0; 43]))] + }), }, ); diff --git a/components/zcash_address/src/kind/unified.rs b/components/zcash_address/src/kind/unified.rs index 7a12600b01..8f7209793a 100644 --- a/components/zcash_address/src/kind/unified.rs +++ b/components/zcash_address/src/kind/unified.rs @@ -120,10 +120,10 @@ impl TryFrom for MetadataTypecode { fn try_from(typecode: u32) -> Result { match typecode { + 0xC0..=0xDF => Ok(MetadataTypecode::Unknown(typecode)), 0xE0 => Ok(MetadataTypecode::ExpiryHeight), 0xE1 => Ok(MetadataTypecode::ExpiryTime), - 0xE2..=0xEF => Ok(MetadataTypecode::MustUnderstand(typecode)), - 0xF0..=0xFC => Ok(MetadataTypecode::Unknown(typecode)), + 0xE2..=0xFC => Ok(MetadataTypecode::MustUnderstand(typecode)), _ => Err(()), } } @@ -212,9 +212,13 @@ pub enum MetadataItem { impl MetadataItem { /// Parse a metadata item for the specified metadata typecode from the provided bytes. - pub fn parse(typecode: MetadataTypecode, data: &[u8]) -> Result { - match typecode { - MetadataTypecode::ExpiryHeight => data + pub fn parse( + revision: Revision, + typecode: MetadataTypecode, + data: &[u8], + ) -> Result { + match (revision, typecode) { + (Revision::R1, MetadataTypecode::ExpiryHeight) => data .try_into() .map(u32::from_le_bytes) .map(MetadataItem::ExpiryHeight) @@ -223,7 +227,7 @@ impl MetadataItem { "Expiry height must be a 32-bit little-endian value.".to_string(), ) }), - MetadataTypecode::ExpiryTime => data + (Revision::R1, MetadataTypecode::ExpiryTime) => data .try_into() .map(u64::from_le_bytes) .map(MetadataItem::ExpiryTime) @@ -232,11 +236,18 @@ impl MetadataItem { "Expiry time must be a 64-bit little-endian value.".to_string(), ) }), - MetadataTypecode::MustUnderstand(tc) => Err(ParseError::NotUnderstood(tc)), - MetadataTypecode::Unknown(typecode) => Ok(MetadataItem::Unknown { + (Revision::R1, MetadataTypecode::Unknown(typecode)) => Ok(MetadataItem::Unknown { typecode, data: data.to_vec(), }), + (Revision::R1, MetadataTypecode::MustUnderstand(tc)) => { + Err(ParseError::NotUnderstood(tc)) + } + (Revision::R0, _) => { + // The entire R0 metadata range has now been reserved for MUST-understand metadata + // in R1, so no R0 metadata range now exists. + Err(ParseError::NotUnderstood(typecode.into())) + } } } @@ -346,8 +357,15 @@ impl fmt::Display for ParseError { impl Error for ParseError {} +/// The revision of the Unified Address standard that an address was parsed under. +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] +pub enum Revision { + R0, + R1, +} + pub(crate) mod private { - use super::{DataTypecode, ParseError, Typecode, PADDING_LEN}; + use super::{DataTypecode, ParseError, Revision, Typecode, PADDING_LEN}; use crate::{ unified::{Item, MetadataItem}, Network, @@ -372,29 +390,47 @@ pub(crate) mod private { /// A Unified Container containing addresses or viewing keys. pub trait SealedContainer: super::Container + std::marker::Sized { - const MAINNET: &'static str; - const TESTNET: &'static str; - const REGTEST: &'static str; + const MAINNET_R0: &'static str; + const TESTNET_R0: &'static str; + const REGTEST_R0: &'static str; + + const MAINNET_R1: &'static str; + const TESTNET_R1: &'static str; + const REGTEST_R1: &'static str; /// Implementations of this method should act as unchecked constructors /// of the container type; the caller is guaranteed to check the /// general invariants that apply to all unified containers. - fn from_inner(items: Vec>) -> Self; + fn from_inner(revision: Revision, items: Vec>) -> Self; + + fn network_hrp(revision: Revision, network: &Network) -> &'static str { + match (revision, network) { + (Revision::R0, Network::Main) => Self::MAINNET_R0, + (Revision::R0, Network::Test) => Self::TESTNET_R0, + (Revision::R0, Network::Regtest) => Self::REGTEST_R0, + (Revision::R1, Network::Main) => Self::MAINNET_R1, + (Revision::R1, Network::Test) => Self::TESTNET_R1, + (Revision::R1, Network::Regtest) => Self::REGTEST_R1, + } + } - fn network_hrp(network: &Network) -> &'static str { - match network { - Network::Main => Self::MAINNET, - Network::Test => Self::TESTNET, - Network::Regtest => Self::REGTEST, + fn hrp_revision(hrp: &str) -> Option { + if hrp == Self::MAINNET_R0 || hrp == Self::TESTNET_R0 || hrp == Self::REGTEST_R0 { + Some(Revision::R0) + } else if hrp == Self::MAINNET_R1 || hrp == Self::TESTNET_R1 || hrp == Self::REGTEST_R1 + { + Some(Revision::R1) + } else { + None } } fn hrp_network(hrp: &str) -> Option { - if hrp == Self::MAINNET { + if hrp == Self::MAINNET_R0 || hrp == Self::MAINNET_R1 { Some(Network::Main) - } else if hrp == Self::TESTNET { + } else if hrp == Self::TESTNET_R0 || hrp == Self::TESTNET_R1 { Some(Network::Test) - } else if hrp == Self::REGTEST { + } else if hrp == Self::REGTEST_R0 || hrp == Self::REGTEST_R1 { Some(Network::Regtest) } else { None @@ -431,11 +467,13 @@ pub(crate) mod private { } /// Parse the items of the unified container. + #[allow(clippy::type_complexity)] fn parse_items>>( hrp: &str, buf: T, - ) -> Result>, ParseError> { - fn read_receiver( + ) -> Result<(Revision, Vec>), ParseError> { + fn read_item( + revision: Revision, mut cursor: &mut std::io::Cursor<&[u8]>, ) -> Result, ParseError> { let typecode = CompactSize::read(&mut cursor) @@ -468,7 +506,9 @@ pub(crate) mod private { let data = &buf[cursor.position() as usize..addr_end as usize]; let result = match Typecode::try_from(typecode)? { Typecode::Data(tc) => Item::Data(R::parse(tc, data)?), - Typecode::Metadata(tc) => Item::Metadata(MetadataItem::parse(tc, data)?), + Typecode::Metadata(tc) => { + Item::Metadata(MetadataItem::parse(revision, tc, data)?) + } }; cursor.set_position(addr_end); Ok(result) @@ -495,19 +535,25 @@ pub(crate) mod private { )), }?; + let revision = Self::hrp_revision(hrp) + .ok_or_else(|| ParseError::UnknownPrefix(hrp.to_string()))?; + let mut cursor = std::io::Cursor::new(encoded); let mut result = vec![]; while cursor.position() < encoded.len().try_into().unwrap() { - result.push(read_receiver(&mut cursor)?); + result.push(read_item(revision, &mut cursor)?); } assert_eq!(cursor.position(), encoded.len().try_into().unwrap()); - Ok(result) + Ok((revision, result)) } /// A private function that constructs a unified container with the /// specified items, which must be in ascending typecode order. - fn try_from_items_internal(items: Vec>) -> Result { + fn try_from_items_internal( + revision: Revision, + items: Vec>, + ) -> Result { assert!(u32::from(Typecode::P2SH) == u32::from(Typecode::P2PKH) + 1); let mut only_transparent = true; @@ -535,12 +581,13 @@ pub(crate) mod private { Err(ParseError::OnlyTransparent) } else { // All checks pass! - Ok(Self::from_inner(items)) + Ok(Self::from_inner(revision, items)) } } fn parse_internal>>(hrp: &str, buf: T) -> Result { - Self::parse_items(hrp, buf).and_then(Self::try_from_items_internal) + Self::parse_items(hrp, buf) + .and_then(|(revision, items)| Self::try_from_items_internal(revision, items)) } } } @@ -559,9 +606,12 @@ pub trait Encoding: private::SealedContainer { /// * the item list may not contain two items having the same typecode /// * the item list may not contain only transparent items (or no items) /// * the item list may not contain both P2PKH and P2SH items. - fn try_from_items(mut items: Vec>) -> Result { + fn try_from_items( + revision: Revision, + mut items: Vec>, + ) -> Result { items.sort_unstable_by(Item::encoding_order); - Self::try_from_items_internal(items) + Self::try_from_items_internal(revision, items) } /// Decodes a unified container from its string representation, preserving @@ -588,7 +638,7 @@ pub trait Encoding: private::SealedContainer { /// ordering of the contained items such that it correctly obeys round-trip /// serialization invariants. fn encode(&self, network: &Network) -> String { - let hrp = Self::network_hrp(network); + let hrp = Self::network_hrp(self.revision(), network); bech32::encode( hrp, self.to_jumbled_bytes(hrp).to_base32(), @@ -607,4 +657,8 @@ pub trait Container { /// /// This API is for advanced usage; in most cases you should use `Self::items`. fn items_as_parsed(&self) -> &[Item]; + + /// Returns the revision of the ZIP 316 standard that this unified container + /// conforms to. + fn revision(&self) -> Revision; } diff --git a/components/zcash_address/src/kind/unified/address.rs b/components/zcash_address/src/kind/unified/address.rs index 41d5c2d354..06af21a1f9 100644 --- a/components/zcash_address/src/kind/unified/address.rs +++ b/components/zcash_address/src/kind/unified/address.rs @@ -1,4 +1,4 @@ -use super::{private::SealedDataItem, DataTypecode, Item, ParseError}; +use super::{private::SealedDataItem, DataTypecode, Item, ParseError, Revision}; use std::convert::TryInto; @@ -61,7 +61,7 @@ impl SealedDataItem for Receiver { /// # use std::convert::Infallible; /// # use std::error::Error; /// use zcash_address::{ -/// unified::{self, Container, Encoding, Item}, +/// unified::{self, Container, Encoding, Item, Revision}, /// ConversionError, TryFromRawAddress, ZcashAddress, /// }; /// @@ -92,13 +92,16 @@ impl SealedDataItem for Receiver { /// let receivers: Vec = ua.receivers(); /// /// // And we can create the UA from a list of receivers: -/// let new_ua = unified::Address::try_from_items(receivers.into_iter().map(Item::Data).collect())?; +/// let new_ua = unified::Address::try_from_items(Revision::R0, receivers.into_iter().map(Item::Data).collect())?; /// assert_eq!(new_ua, ua); /// # Ok(()) /// # } /// ``` #[derive(Clone, Debug, PartialEq, Eq, Hash)] -pub struct Address(pub(crate) Vec>); +pub struct Address { + pub(crate) revision: Revision, + pub(crate) receivers: Vec>, +} impl Address { /// Returns the receiver items for this address, in order of decreasing preference. @@ -107,7 +110,7 @@ impl Address { /// of a type that wallet supports from the result. pub fn receivers(&self) -> Vec { let mut result = self - .0 + .receivers .iter() .filter_map(|item| match item { Item::Data(r) => Some(r.clone()), @@ -121,25 +124,45 @@ impl Address { } impl super::private::SealedContainer for Address { - /// The HRP for a Bech32m-encoded mainnet Unified Address. + /// The HRP for a Bech32m-encoded mainnet Revision 0 Unified Address. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const MAINNET_R0: &'static str = "u"; + + /// The HRP for a Bech32m-encoded testnet Revision 0 Unified Address. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const TESTNET_R0: &'static str = "utest"; + + /// The HRP for a Bech32m-encoded regtest Revision 0 Unified Address. + const REGTEST_R0: &'static str = "uregtest"; + + /// The HRP for a Bech32m-encoded mainnet Revision 1 Unified Address. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const MAINNET: &'static str = "u"; + const MAINNET_R1: &'static str = "ur"; - /// The HRP for a Bech32m-encoded testnet Unified Address. + /// The HRP for a Bech32m-encoded testnet Revision 1 Unified Address. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const TESTNET: &'static str = "utest"; + const TESTNET_R1: &'static str = "urtest"; - /// The HRP for a Bech32m-encoded regtest Unified Address. - const REGTEST: &'static str = "uregtest"; + /// The HRP for a Bech32m-encoded regtest Revision 1 Unified Address. + const REGTEST_R1: &'static str = "urregtest"; - fn from_inner(receivers: Vec>) -> Self { - Self(receivers) + fn from_inner(revision: Revision, receivers: Vec>) -> Self { + Self { + revision, + receivers, + } } } @@ -148,7 +171,11 @@ impl super::Container for Address { type DataItem = Receiver; fn items_as_parsed(&self) -> &[Item] { - &self.0 + &self.receivers + } + + fn revision(&self) -> Revision { + self.revision } } @@ -162,7 +189,7 @@ mod tests { use crate::{ kind::unified::{private::SealedContainer, Encoding}, - unified::{DataTypecode, Item, Typecode}, + unified::{DataTypecode, Item, Revision, Typecode}, Network, }; @@ -234,9 +261,12 @@ mod tests { arb_typecodes() .prop_flat_map(arb_unified_address_receivers) .prop_map(|rs| { - let mut items = rs.into_iter().map(Item::Data).collect::>(); - items.sort_unstable_by(Item::encoding_order); - Address(items) + let mut receivers = rs.into_iter().map(Item::Data).collect::>(); + receivers.sort_unstable_by(Item::encoding_order); + Address { + revision: Revision::R0, + receivers, + } }) } @@ -267,7 +297,7 @@ mod tests { 0x7b, 0x28, 0x69, 0xc9, 0x84, ]; assert_eq!( - Address::parse_internal(Address::MAINNET, &invalid_padding[..]), + Address::parse_internal(Address::MAINNET_R0, &invalid_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -282,7 +312,7 @@ mod tests { 0x4b, 0x31, 0xee, 0x5a, ]; assert_eq!( - Address::parse_internal(Address::MAINNET, &truncated_padding[..]), + Address::parse_internal(Address::MAINNET_R0, &truncated_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -307,7 +337,7 @@ mod tests { 0xc6, 0x5e, 0x68, 0xa2, 0x78, 0x6c, 0x9e, ]; assert_matches!( - Address::parse_internal(Address::MAINNET, &truncated_sapling_data[..]), + Address::parse_internal(Address::MAINNET_R0, &truncated_sapling_data[..]), Err(ParseError::InvalidEncoding(_)) ); @@ -320,7 +350,7 @@ mod tests { 0xe6, 0x70, 0x36, 0x5b, 0x7b, 0x9e, ]; assert_matches!( - Address::parse_internal(Address::MAINNET, &truncated_after_sapling_typecode[..]), + Address::parse_internal(Address::MAINNET_R0, &truncated_after_sapling_typecode[..]), Err(ParseError::InvalidEncoding(_)) ); } @@ -329,13 +359,16 @@ mod tests { fn duplicate_typecode() { // Construct and serialize an invalid UA. This must be done using private // methods, as the public API does not permit construction of such invalid values. - let ua = Address(vec![ - Item::Data(Receiver::Sapling([1; 43])), - Item::Data(Receiver::Sapling([2; 43])), - ]); - let encoded = ua.to_jumbled_bytes(Address::MAINNET); + let ua = Address { + revision: Revision::R0, + receivers: vec![ + Item::Data(Receiver::Sapling([1; 43])), + Item::Data(Receiver::Sapling([2; 43])), + ], + }; + let encoded = ua.to_jumbled_bytes(Address::MAINNET_R0); assert_eq!( - Address::parse_internal(Address::MAINNET, &encoded[..]), + Address::parse_internal(Address::MAINNET_R0, &encoded[..]), Err(ParseError::DuplicateTypecode(Typecode::SAPLING)) ); } @@ -344,14 +377,17 @@ mod tests { fn p2pkh_and_p2sh() { // Construct and serialize an invalid UA. This must be done using private // methods, as the public API does not permit construction of such invalid values. - let ua = Address(vec![ - Item::Data(Receiver::P2pkh([0; 20])), - Item::Data(Receiver::P2sh([0; 20])), - ]); - let encoded = ua.to_jumbled_bytes(Address::MAINNET); + let ua = Address { + revision: Revision::R0, + receivers: vec![ + Item::Data(Receiver::P2pkh([0; 20])), + Item::Data(Receiver::P2sh([0; 20])), + ], + }; + let encoded = ua.to_jumbled_bytes(Address::MAINNET_R0); // ensure that decoding catches the error assert_eq!( - Address::parse_internal(Address::MAINNET, &encoded[..]), + Address::parse_internal(Address::MAINNET_R0, &encoded[..]), Err(ParseError::BothP2phkAndP2sh) ); } @@ -360,14 +396,17 @@ mod tests { fn addresses_out_of_order() { // Construct and serialize an invalid UA. This must be done using private // methods, as the public API does not permit construction of such invalid values. - let ua = Address(vec![ - Item::Data(Receiver::Sapling([0; 43])), - Item::Data(Receiver::P2pkh([0; 20])), - ]); - let encoded = ua.to_jumbled_bytes(Address::MAINNET); + let ua = Address { + revision: Revision::R0, + receivers: vec![ + Item::Data(Receiver::Sapling([0; 43])), + Item::Data(Receiver::P2pkh([0; 20])), + ], + }; + let encoded = ua.to_jumbled_bytes(Address::MAINNET_R0); // ensure that decoding catches the error assert_eq!( - Address::parse_internal(Address::MAINNET, &encoded[..]), + Address::parse_internal(Address::MAINNET_R0, &encoded[..]), Err(ParseError::InvalidTypecodeOrder) ); } @@ -386,7 +425,7 @@ mod tests { // with only one of them we don't have sufficient data for F4Jumble (so we hit a // different error). assert_matches!( - Address::parse_internal(Address::MAINNET, &encoded[..]), + Address::parse_internal(Address::MAINNET_R0, &encoded[..]), Err(ParseError::InvalidEncoding(_)) ); } @@ -394,15 +433,18 @@ mod tests { #[test] fn receivers_are_sorted() { // Construct a UA with receivers in an unsorted order. - let ua = Address(vec![ - Item::Data(Receiver::P2pkh([0; 20])), - Item::Data(Receiver::Orchard([0; 43])), - Item::Data(Receiver::Unknown { - typecode: 0xff, - data: vec![], - }), - Item::Data(Receiver::Sapling([0; 43])), - ]); + let ua = Address { + revision: Revision::R0, + receivers: vec![ + Item::Data(Receiver::P2pkh([0; 20])), + Item::Data(Receiver::Orchard([0; 43])), + Item::Data(Receiver::Unknown { + typecode: 0xff, + data: vec![], + }), + Item::Data(Receiver::Sapling([0; 43])), + ], + }; // `Address::receivers` sorts the receivers in priority order. assert_eq!( diff --git a/components/zcash_address/src/kind/unified/fvk.rs b/components/zcash_address/src/kind/unified/fvk.rs index d20cc65e1e..bb04b3e041 100644 --- a/components/zcash_address/src/kind/unified/fvk.rs +++ b/components/zcash_address/src/kind/unified/fvk.rs @@ -2,7 +2,7 @@ use std::convert::TryInto; use super::{ private::{SealedContainer, SealedDataItem}, - Container, DataTypecode, Encoding, Item, ParseError, + Container, DataTypecode, Encoding, Item, ParseError, Revision, }; /// The set of known FVKs for Unified FVKs. @@ -79,7 +79,7 @@ impl SealedDataItem for Fvk { /// /// ``` /// # use std::error::Error; -/// use zcash_address::unified::{self, Container, Encoding, Item}; +/// use zcash_address::unified::{self, Container, Encoding, Item, Revision}; /// /// # fn main() -> Result<(), Box> { /// # let ufvk_from_user = || "uview1cgrqnry478ckvpr0f580t6fsahp0a5mj2e9xl7hv2d2jd4ldzy449mwwk2l9yeuts85wjls6hjtghdsy5vhhvmjdw3jxl3cxhrg3vs296a3czazrycrr5cywjhwc5c3ztfyjdhmz0exvzzeyejamyp0cr9z8f9wj0953fzht0m4lenk94t70ruwgjxag2tvp63wn9ftzhtkh20gyre3w5s24f6wlgqxnjh40gd2lxe75sf3z8h5y2x0atpxcyf9t3em4h0evvsftluruqne6w4sm066sw0qe5y8qg423grple5fftxrqyy7xmqmatv7nzd7tcjadu8f7mqz4l83jsyxy4t8pkayytyk7nrp467ds85knekdkvnd7hqkfer8mnqd7pv"; @@ -93,13 +93,16 @@ impl SealedDataItem for Fvk { /// let fvks: &[Item] = ufvk.items_as_parsed(); /// /// // And we can create the UFVK from a list of FVKs: -/// let new_ufvk = unified::Ufvk::try_from_items(fvks.to_vec())?; +/// let new_ufvk = unified::Ufvk::try_from_items(Revision::R0, fvks.to_vec())?; /// assert_eq!(new_ufvk, ufvk); /// # Ok(()) /// # } /// ``` #[derive(Clone, Debug, PartialEq, Eq, Hash)] -pub struct Ufvk(pub(crate) Vec>); +pub struct Ufvk { + pub(crate) revision: Revision, + pub(crate) fvks: Vec>, +} impl Container for Ufvk { type DataItem = Fvk; @@ -109,32 +112,53 @@ impl Container for Ufvk { /// /// This API is for advanced usage; in most cases you should use `Ufvk::receivers`. fn items_as_parsed(&self) -> &[Item] { - &self.0 + &self.fvks + } + + fn revision(&self) -> Revision { + self.revision } } impl Encoding for Ufvk {} impl SealedContainer for Ufvk { - /// The HRP for a Bech32m-encoded mainnet Unified FVK. + /// The HRP for a Bech32m-encoded mainnet Revision 0 Unified FVK. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const MAINNET_R0: &'static str = "uview"; + + /// The HRP for a Bech32m-encoded testnet Revision 0 Unified FVK. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const TESTNET_R0: &'static str = "uviewtest"; + + /// The HRP for a Bech32m-encoded regtest Revision 0 Unified FVK. + const REGTEST_R0: &'static str = "uviewregtest"; + + /// The HRP for a Bech32m-encoded mainnet Revision 1 Unified FVK. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const MAINNET: &'static str = "uview"; + const MAINNET_R1: &'static str = "urview"; - /// The HRP for a Bech32m-encoded testnet Unified FVK. + /// The HRP for a Bech32m-encoded testnet Revision 1 Unified FVK. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const TESTNET: &'static str = "uviewtest"; + const TESTNET_R1: &'static str = "urviewtest"; - /// The HRP for a Bech32m-encoded regtest Unified FVK. - const REGTEST: &'static str = "uviewregtest"; + /// The HRP for a Bech32m-encoded regtest Revision 1 Unified FVK. + const REGTEST_R1: &'static str = "urviewregtest"; - fn from_inner(fvks: Vec>) -> Self { - Self(fvks) + fn from_inner(revision: Revision, fvks: Vec>) -> Self { + Self { revision, fvks } } } @@ -147,7 +171,7 @@ mod tests { use super::{Fvk, ParseError, Ufvk}; use crate::{ kind::unified::{private::SealedContainer, Encoding}, - unified::{Item, Typecode}, + unified::{Item, Revision, Typecode}, Network, }; @@ -205,9 +229,9 @@ mod tests { shielded in arb_shielded_fvk(), transparent in prop::option::of(arb_transparent_fvk()), ) -> Ufvk { - let mut items: Vec<_> = transparent.into_iter().chain(shielded).map(Item::Data).collect(); - items.sort_unstable_by(Item::encoding_order); - Ufvk(items) + let mut fvks: Vec<_> = transparent.into_iter().chain(shielded).map(Item::Data).collect(); + fvks.sort_unstable_by(Item::encoding_order); + Ufvk { revision: Revision::R0, fvks } } } @@ -239,7 +263,7 @@ mod tests { 0xdf, 0x63, 0xe7, 0xef, 0x65, 0x6b, 0x18, 0x23, 0xf7, 0x3e, 0x35, 0x7c, 0xf3, 0xc4, ]; assert_eq!( - Ufvk::parse_internal(Ufvk::MAINNET, &invalid_padding[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &invalid_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -257,7 +281,7 @@ mod tests { 0x43, 0x8e, 0xc0, 0x3e, 0x9f, 0xf4, 0xf1, 0x80, 0x32, 0xcf, 0x2f, 0x7e, 0x7f, 0x91, ]; assert_eq!( - Ufvk::parse_internal(Ufvk::MAINNET, &truncated_padding[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &truncated_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -289,7 +313,7 @@ mod tests { 0x8c, 0x7a, 0xbf, 0x7b, 0x9a, 0xdd, 0xee, 0x18, 0x2c, 0x2d, 0xc2, 0xfc, ]; assert_matches!( - Ufvk::parse_internal(Ufvk::MAINNET, &truncated_sapling_data[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &truncated_sapling_data[..]), Err(ParseError::InvalidEncoding(_)) ); @@ -304,7 +328,7 @@ mod tests { 0x54, 0xd1, 0x9e, 0xec, 0x8b, 0xef, 0x35, 0xb8, 0x44, 0xdd, 0xab, 0x9a, 0x8d, ]; assert_matches!( - Ufvk::parse_internal(Ufvk::MAINNET, &truncated_after_sapling_typecode[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &truncated_after_sapling_typecode[..]), Err(ParseError::InvalidEncoding(_)) ); } @@ -313,13 +337,16 @@ mod tests { fn duplicate_typecode() { // Construct and serialize an invalid Ufvk. This must be done using private // methods, as the public API does not permit construction of such invalid values. - let ufvk = Ufvk(vec![ - Item::Data(Fvk::Sapling([1; 128])), - Item::Data(Fvk::Sapling([2; 128])), - ]); - let encoded = ufvk.to_jumbled_bytes(Ufvk::MAINNET); + let ufvk = Ufvk { + revision: Revision::R0, + fvks: vec![ + Item::Data(Fvk::Sapling([1; 128])), + Item::Data(Fvk::Sapling([2; 128])), + ], + }; + let encoded = ufvk.to_jumbled_bytes(Ufvk::MAINNET_R0); assert_eq!( - Ufvk::parse_internal(Ufvk::MAINNET, &encoded[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &encoded[..]), Err(ParseError::DuplicateTypecode(Typecode::SAPLING)) ); } @@ -337,7 +364,7 @@ mod tests { ]; assert_eq!( - Ufvk::parse_internal(Ufvk::MAINNET, &encoded[..]), + Ufvk::parse_internal(Ufvk::MAINNET_R0, &encoded[..]), Err(ParseError::OnlyTransparent) ); } diff --git a/components/zcash_address/src/kind/unified/ivk.rs b/components/zcash_address/src/kind/unified/ivk.rs index 9253c425bf..0b44e4dc07 100644 --- a/components/zcash_address/src/kind/unified/ivk.rs +++ b/components/zcash_address/src/kind/unified/ivk.rs @@ -2,7 +2,7 @@ use std::convert::TryInto; use super::{ private::{SealedContainer, SealedDataItem}, - Container, DataTypecode, Encoding, Item, ParseError, + Container, DataTypecode, Encoding, Item, ParseError, Revision, }; /// The set of known IVKs for Unified IVKs. @@ -84,7 +84,7 @@ impl SealedDataItem for Ivk { /// /// ``` /// # use std::error::Error; -/// use zcash_address::unified::{self, Container, Encoding, Item}; +/// use zcash_address::unified::{self, Container, Encoding, Item, Revision}; /// /// # fn main() -> Result<(), Box> { /// # let uivk_from_user = || "uivk1djetqg3fws7y7qu5tekynvcdhz69gsyq07ewvppmzxdqhpfzdgmx8urnkqzv7ylz78ez43ux266pqjhecd59fzhn7wpe6zarnzh804hjtkyad25ryqla5pnc8p5wdl3phj9fczhz64zprun3ux7y9jc08567xryumuz59rjmg4uuflpjqwnq0j0tzce0x74t4tv3gfjq7nczkawxy6y7hse733ae3vw7qfjd0ss0pytvezxp42p6rrpzeh6t2zrz7zpjk0xhngcm6gwdppxs58jkx56gsfflugehf5vjlmu7vj3393gj6u37wenavtqyhdvcdeaj86s6jczl4zq"; @@ -98,13 +98,16 @@ impl SealedDataItem for Ivk { /// let ivks: &[Item] = uivk.items_as_parsed(); /// /// // And we can create the UIVK from a vector of IVKs: -/// let new_uivk = unified::Uivk::try_from_items(ivks.to_vec())?; +/// let new_uivk = unified::Uivk::try_from_items(Revision::R0, ivks.to_vec())?; /// assert_eq!(new_uivk, uivk); /// # Ok(()) /// # } /// ``` #[derive(Clone, Debug, PartialEq, Eq, Hash)] -pub struct Uivk(pub(crate) Vec>); +pub struct Uivk { + pub(crate) revision: Revision, + pub(crate) ivks: Vec>, +} impl Container for Uivk { type DataItem = Ivk; @@ -114,32 +117,53 @@ impl Container for Uivk { /// /// This API is for advanced usage; in most cases you should use `Uivk::items`. fn items_as_parsed(&self) -> &[Item] { - &self.0 + &self.ivks + } + + fn revision(&self) -> Revision { + self.revision } } impl Encoding for Uivk {} impl SealedContainer for Uivk { - /// The HRP for a Bech32m-encoded mainnet Unified IVK. + /// The HRP for a Bech32m-encoded mainnet Revision 0 Unified IVK. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const MAINNET: &'static str = "uivk"; + const MAINNET_R0: &'static str = "uivk"; - /// The HRP for a Bech32m-encoded testnet Unified IVK. + /// The HRP for a Bech32m-encoded testnet Revision 0 Unified IVK. /// /// Defined in [ZIP 316][zip-0316]. /// /// [zip-0316]: https://zips.z.cash/zip-0316 - const TESTNET: &'static str = "uivktest"; + const TESTNET_R0: &'static str = "uivktest"; + + /// The HRP for a Bech32m-encoded regtest Revision 0 Unified IVK. + const REGTEST_R0: &'static str = "uivkregtest"; - /// The HRP for a Bech32m-encoded regtest Unified IVK. - const REGTEST: &'static str = "uivkregtest"; + /// The HRP for a Bech32m-encoded mainnet Revision 1 Unified IVK. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const MAINNET_R1: &'static str = "urivk"; - fn from_inner(ivks: Vec>) -> Self { - Self(ivks) + /// The HRP for a Bech32m-encoded testnet Revision 1 Unified IVK. + /// + /// Defined in [ZIP 316][zip-0316]. + /// + /// [zip-0316]: https://zips.z.cash/zip-0316 + const TESTNET_R1: &'static str = "urivktest"; + + /// The HRP for a Bech32m-encoded regtest Revision 1 Unified IVK. + const REGTEST_R1: &'static str = "urivkregtest"; + + fn from_inner(revision: Revision, ivks: Vec>) -> Self { + Self { revision, ivks } } } @@ -156,7 +180,7 @@ mod tests { use super::{Ivk, ParseError, Uivk}; use crate::{ kind::unified::{private::SealedContainer, Encoding}, - unified::{Item, Typecode}, + unified::{Item, Revision, Typecode}, Network, }; @@ -198,9 +222,9 @@ mod tests { shielded in arb_shielded_ivk(), transparent in prop::option::of(arb_transparent_ivk()), ) -> Uivk { - let mut items: Vec<_> = transparent.into_iter().chain(shielded).map(Item::Data).collect(); - items.sort_unstable_by(Item::encoding_order); - Uivk(items) + let mut ivks: Vec<_> = transparent.into_iter().chain(shielded).map(Item::Data).collect(); + ivks.sort_unstable_by(Item::encoding_order); + Uivk { revision: Revision::R0, ivks } } } @@ -230,7 +254,7 @@ mod tests { 0x83, 0xe8, 0x92, 0x18, 0x28, 0x70, 0x1e, 0x81, 0x76, 0x56, 0xb6, 0x15, ]; assert_eq!( - Uivk::parse_internal(Uivk::MAINNET, &invalid_padding[..]), + Uivk::parse_internal(Uivk::MAINNET_R0, &invalid_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -246,7 +270,7 @@ mod tests { 0xf9, 0x65, 0x49, 0x14, 0xab, 0x7c, 0x55, 0x7b, 0x39, 0x47, ]; assert_eq!( - Uivk::parse_internal(Uivk::MAINNET, &truncated_padding[..]), + Uivk::parse_internal(Uivk::MAINNET_R0, &truncated_padding[..]), Err(ParseError::InvalidEncoding( "Invalid padding bytes".to_owned() )) @@ -274,7 +298,7 @@ mod tests { 0xf5, 0xd5, 0x8a, 0xb5, 0x1a, ]; assert_matches!( - Uivk::parse_internal(Uivk::MAINNET, &truncated_sapling_data[..]), + Uivk::parse_internal(Uivk::MAINNET_R0, &truncated_sapling_data[..]), Err(ParseError::InvalidEncoding(_)) ); @@ -287,7 +311,7 @@ mod tests { 0xd8, 0x21, 0x5e, 0x8, 0xa, 0x82, 0x95, 0x21, 0x74, ]; assert_matches!( - Uivk::parse_internal(Uivk::MAINNET, &truncated_after_sapling_typecode[..]), + Uivk::parse_internal(Uivk::MAINNET_R0, &truncated_after_sapling_typecode[..]), Err(ParseError::InvalidEncoding(_)) ); } @@ -295,10 +319,13 @@ mod tests { #[test] fn duplicate_typecode() { // Construct and serialize an invalid UIVK. - let uivk = Uivk(vec![ - Item::Data(Ivk::Sapling([1; 64])), - Item::Data(Ivk::Sapling([2; 64])), - ]); + let uivk = Uivk { + revision: Revision::R0, + ivks: vec![ + Item::Data(Ivk::Sapling([1; 64])), + Item::Data(Ivk::Sapling([2; 64])), + ], + }; let encoded = uivk.encode(&Network::Main); assert_eq!( Uivk::decode(&encoded), @@ -319,7 +346,7 @@ mod tests { ]; assert_eq!( - Uivk::parse_internal(Uivk::MAINNET, &encoded[..]), + Uivk::parse_internal(Uivk::MAINNET_R0, &encoded[..]), Err(ParseError::OnlyTransparent) ); } diff --git a/components/zcash_address/src/test_vectors.rs b/components/zcash_address/src/test_vectors.rs index 598096da66..991d16209a 100644 --- a/components/zcash_address/src/test_vectors.rs +++ b/components/zcash_address/src/test_vectors.rs @@ -8,7 +8,7 @@ use { unified::{ self, address::{test_vectors::TEST_VECTORS, Receiver}, - Item, + Item, Revision, }, Network, ToAddress, ZcashAddress, }, @@ -40,7 +40,13 @@ fn unified() { .map(Item::Data) .collect(); - let expected_addr = ZcashAddress::from_unified(Network::Main, unified::Address(receivers)); + let expected_addr = ZcashAddress::from_unified( + Network::Main, + unified::Address { + revision: Revision::R0, + receivers, + }, + ); // Test parsing let addr: ZcashAddress = tv.unified_addr.parse().unwrap(); diff --git a/zcash_keys/src/address.rs b/zcash_keys/src/address.rs index 50677347c8..3f95907ce3 100644 --- a/zcash_keys/src/address.rs +++ b/zcash_keys/src/address.rs @@ -4,7 +4,7 @@ use std::convert::TryFrom; use sapling::PaymentAddress; use zcash_address::{ - unified::{self, Container, Encoding, Item}, + unified::{self, Container, Encoding, Item, Revision}, ConversionError, Network, ToAddress, TryFromRawAddress, ZcashAddress, }; use zcash_primitives::{ @@ -182,6 +182,9 @@ impl UnifiedAddress { } /// Returns any unknown metadata items parsed from the encoded form of the address. + /// + /// Unknown metadata items are guaranteed by construction and parsing to not have keys in the + /// `must-understand` metadata typecode range. pub fn unknown_metadata(&self) -> &[(u32, Vec)] { self.unknown_metadata.as_ref() } @@ -202,43 +205,50 @@ impl UnifiedAddress { #[cfg(not(feature = "orchard"))] let orchard_receiver = None; - let ua = unified::Address::try_from_items({ - let data_items = self - .unknown_data - .iter() - .map(|(typecode, data)| unified::Receiver::Unknown { - typecode: *typecode, - data: data.clone(), - }) - .chain(self.transparent.as_ref().map(|taddr| match taddr { - TransparentAddress::PublicKeyHash(data) => unified::Receiver::P2pkh(*data), - TransparentAddress::ScriptHash(data) => unified::Receiver::P2sh(*data), - })) - .chain( - self.sapling - .as_ref() - .map(|pa| pa.to_bytes()) - .map(unified::Receiver::Sapling), - ) - .chain(orchard_receiver) - .map(Item::Data); - - let meta_items = self - .unknown_metadata - .iter() - .map(|(typecode, data)| unified::MetadataItem::Unknown { - typecode: *typecode, - data: data.clone(), - }) - .chain( - self.expiry_height - .map(|h| unified::MetadataItem::ExpiryHeight(u32::from(h))), - ) - .chain(self.expiry_time.map(unified::MetadataItem::ExpiryTime)) - .map(Item::Metadata); - - data_items.chain(meta_items).collect() - }) + let ua = unified::Address::try_from_items( + if self.expiry_height().is_some() || self.expiry_time().is_some() { + Revision::R1 + } else { + Revision::R0 + }, + { + let data_items = self + .unknown_data + .iter() + .map(|(typecode, data)| unified::Receiver::Unknown { + typecode: *typecode, + data: data.clone(), + }) + .chain(self.transparent.as_ref().map(|taddr| match taddr { + TransparentAddress::PublicKeyHash(data) => unified::Receiver::P2pkh(*data), + TransparentAddress::ScriptHash(data) => unified::Receiver::P2sh(*data), + })) + .chain( + self.sapling + .as_ref() + .map(|pa| pa.to_bytes()) + .map(unified::Receiver::Sapling), + ) + .chain(orchard_receiver) + .map(Item::Data); + + let meta_items = self + .unknown_metadata + .iter() + .map(|(typecode, data)| unified::MetadataItem::Unknown { + typecode: *typecode, + data: data.clone(), + }) + .chain( + self.expiry_height + .map(|h| unified::MetadataItem::ExpiryHeight(u32::from(h))), + ) + .chain(self.expiry_time.map(unified::MetadataItem::ExpiryTime)) + .map(Item::Metadata); + + data_items.chain(meta_items).collect() + }, + ) .expect("UnifiedAddress should only be constructed safely"); ZcashAddress::from_unified(net, ua) } diff --git a/zcash_keys/src/keys.rs b/zcash_keys/src/keys.rs index 2462ec24ef..8f0aa29407 100644 --- a/zcash_keys/src/keys.rs +++ b/zcash_keys/src/keys.rs @@ -1,5 +1,5 @@ //! Helper functions for managing light client key material. -use zcash_address::unified::{self, Container, Encoding, Item, MetadataItem, Typecode}; +use zcash_address::unified::{self, Container, Encoding, Item, MetadataItem, Revision, Typecode}; use zcash_primitives::{ consensus::{self, BlockHeight}, zip32::{AccountId, DiversifierIndex}, @@ -649,6 +649,11 @@ impl UnifiedFullViewingKey { .chain(self.expiry_time.map(unified::MetadataItem::ExpiryTime)); let ufvk = unified::Ufvk::try_from_items( + if self.expiry_height().is_some() || self.expiry_time().is_some() { + Revision::R1 + } else { + Revision::R0 + }, data_items .map(Item::Data) .chain(meta_items.map(Item::Metadata))