Skip to content

Commit

Permalink
Only allow construction of Unified keys & addresses with at least one…
Browse files Browse the repository at this point in the history
… data item.
  • Loading branch information
nuttycom committed Dec 10, 2024
1 parent dea0c51 commit 7852a31
Show file tree
Hide file tree
Showing 4 changed files with 217 additions and 63 deletions.
5 changes: 4 additions & 1 deletion zcash_client_backend/src/data_api/testing/orchard.rs
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ impl ShieldedPoolTester for OrchardPoolTester {
None,
None,
)
.expect("Address construction is valid")
.into()
}

Expand Down Expand Up @@ -154,7 +155,9 @@ impl ShieldedPoolTester for OrchardPoolTester {
return result.map(|(note, addr, memo)| {
(
Note::Orchard(note),
UnifiedAddress::from_receivers(Some(addr), None, None).into(),
UnifiedAddress::from_receivers(Some(addr), None, None)
.expect("Address construction is valid")

Check warning on line 159 in zcash_client_backend/src/data_api/testing/orchard.rs

View check run for this annotation

Codecov / codecov/patch

zcash_client_backend/src/data_api/testing/orchard.rs#L159

Added line #L159 was not covered by tests
.into(),
MemoBytes::from_bytes(&memo).expect("correct length"),
)
});
Expand Down
6 changes: 6 additions & 0 deletions zcash_keys/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,9 @@ and this library adheres to Rust's notion of

## [Unreleased]

### Added
- `zcash_keys::keys::UnifiedKeyError`

### Changed
- The `UnifiedAddressRequest` argument to the following methods is now optional:
- `zcash_keys::keys::UnifiedSpendingKey::address`
Expand All @@ -22,6 +25,9 @@ and this library adheres to Rust's notion of
- Migrated to `zcash_primitives 0.20.0`
- MSRV is now 1.77.0.

### Removed
- `zcash_keys::keys::DerivationError` use `UnifiedKeyError` instead.

## [0.4.0] - 2024-10-04

### Added
Expand Down
63 changes: 48 additions & 15 deletions zcash_keys/src/address.rs
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ impl TryFrom<unified::Address> for UnifiedAddress {
}
}

Ok(Self {
Self::from_checked_parts(
#[cfg(feature = "orchard")]
orchard,
#[cfg(feature = "sapling")]
Expand All @@ -103,7 +103,8 @@ impl TryFrom<unified::Address> for UnifiedAddress {
expiry_height,
expiry_time,
unknown_metadata,
})
)
.ok_or("Unified addresses without data fields are not permitted.")
}
}

Expand All @@ -118,36 +119,55 @@ impl UnifiedAddress {
#[cfg(feature = "orchard")] orchard: Option<orchard::Address>,
#[cfg(feature = "sapling")] sapling: Option<PaymentAddress>,
transparent: Option<TransparentAddress>,
) -> Self {
Self::new_internal(
) -> Option<Self> {
Self::from_checked_parts(
#[cfg(feature = "orchard")]
orchard,
#[cfg(feature = "sapling")]
sapling,
transparent,
vec![],
None,
None,
vec![],

Check warning on line 132 in zcash_keys/src/address.rs

View check run for this annotation

Codecov / codecov/patch

zcash_keys/src/address.rs#L124-L132

Added lines #L124 - L132 were not covered by tests
)
}

pub(crate) fn new_internal(
pub(crate) fn from_checked_parts(
#[cfg(feature = "orchard")] orchard: Option<orchard::Address>,
#[cfg(feature = "sapling")] sapling: Option<PaymentAddress>,
transparent: Option<TransparentAddress>,
unknown_data: Vec<(u32, Vec<u8>)>,
expiry_height: Option<BlockHeight>,
expiry_time: Option<u64>,
) -> Self {
Self {
unknown_metadata: Vec<(u32, Vec<u8>)>,
) -> Option<Self> {
let has_transparent = transparent.is_some();

#[allow(unused_mut)]
let mut has_shielded = false;
#[cfg(feature = "sapling")]
{
has_shielded = has_shielded || sapling.is_some();
}
#[cfg(feature = "orchard")]
{
has_shielded = has_shielded || orchard.is_some();
}

let has_unknown = !unknown_data.is_empty();

(has_transparent || has_shielded || has_unknown).then_some(Self {
#[cfg(feature = "orchard")]
orchard,
#[cfg(feature = "sapling")]
sapling,
transparent,
unknown_data: vec![],
unknown_data,
expiry_height,
expiry_time,
unknown_metadata: vec![],
}
unknown_metadata,
})
}

/// Returns whether this address has an Orchard receiver.
Expand Down Expand Up @@ -593,15 +613,25 @@ mod tests {
let transparent = None;

#[cfg(all(feature = "orchard", feature = "sapling"))]
let ua = UnifiedAddress::new_internal(orchard, sapling, transparent, None, None);
let ua = UnifiedAddress::from_checked_parts(
orchard,
sapling,
transparent,
vec![],
None,
None,
vec![],
);

#[cfg(all(not(feature = "orchard"), feature = "sapling"))]
let ua = UnifiedAddress::new_internal(sapling, transparent, None, None);
let ua =
UnifiedAddress::from_checked_parts(sapling, transparent, vec![], None, None, vec![]);

#[cfg(all(feature = "orchard", not(feature = "sapling")))]
let ua = UnifiedAddress::new_internal(orchard, transparent, None, None);
let ua =
UnifiedAddress::from_checked_parts(orchard, transparent, vec![], None, None, vec![]);

let addr = Address::from(ua);
let addr = Address::from(ua.expect("test UAs are constructed in valid configurations"));
let addr_str = addr.encode(&MAIN_NETWORK);
assert_eq!(Address::decode(&MAIN_NETWORK, &addr_str), Some(addr));
}
Expand All @@ -610,7 +640,10 @@ mod tests {
#[cfg(not(any(feature = "orchard", feature = "sapling")))]
fn ua_round_trip() {
let transparent = None;
assert_eq!(UnifiedAddress::new_internal(transparent, None, None), None)
assert_eq!(
UnifiedAddress::from_checked_parts(transparent, vec![], None, None, vec![]),
None
)
}

#[test]
Expand Down
Loading

0 comments on commit 7852a31

Please sign in to comment.