Refactor Sapling data and use it in V4 (#1946)

* start refactoring transaction v4 for transaction v5
- move ShieldedData to sapling
- add AnchorVariant
- rename shielded_data to sapling_shielded data in V4
- move value_balance into ShieldedData
- update prop tests for new structure

* add AnchorVariant to Spend
- make anchor types available from sapling crate
- update serialize

* change shielded_balances_match() arguments
* change variable name anchor to shared_anchor in ShieldedData
* fix empty value balance serialization
* use AnchorV in shielded spends
* Rename anchor to per_spend_anchor

* Use nullifiers function directly in non-finalized state
* Use self.value_balance instead of passing it as an argument

* Add missing fields to ShieldedData PartialEq
* Derive Copy for tag types

* Add doc comments for ShieldedData refactor
* Implement a per-spend anchor compatibility iterator

Co-authored-by: teor <teor@riseup.net>
This commit is contained in:
Alfredo Garcia 2021-03-31 18:34:25 -03:00 committed by GitHub
parent 29163cd0b4
commit 48a8a7b851
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 392 additions and 147 deletions

32
Cargo.lock generated
View File

@ -1,5 +1,7 @@
# This file is automatically @generated by Cargo. # This file is automatically @generated by Cargo.
# It is not intended for manual editing. # It is not intended for manual editing.
version = 3
[[package]] [[package]]
name = "abscissa_core" name = "abscissa_core"
version = "0.5.2" version = "0.5.2"
@ -921,6 +923,20 @@ dependencies = [
"syn 1.0.60", "syn 1.0.60",
] ]
[[package]]
name = "ed25519-zebra"
version = "2.2.0"
source = "git+https://github.com/ZcashFoundation/ed25519-zebra?rev=539fad040c443302775b0f508e616418825e6c22#539fad040c443302775b0f508e616418825e6c22"
dependencies = [
"curve25519-dalek",
"hex",
"rand_core 0.6.2",
"serde",
"sha2",
"thiserror",
"zeroize",
]
[[package]] [[package]]
name = "ed25519-zebra" name = "ed25519-zebra"
version = "2.2.0" version = "2.2.0"
@ -935,20 +951,6 @@ dependencies = [
"thiserror", "thiserror",
] ]
[[package]]
name = "ed25519-zebra"
version = "2.2.0"
source = "git+https://github.com/ZcashFoundation/ed25519-zebra?rev=856c96500125e8dd38a525dad13dc64a0ac672cc#856c96500125e8dd38a525dad13dc64a0ac672cc"
dependencies = [
"curve25519-dalek",
"hex",
"rand_core 0.6.2",
"serde",
"sha2",
"thiserror",
"zeroize",
]
[[package]] [[package]]
name = "either" name = "either"
version = "1.6.1" version = "1.6.1"
@ -4070,7 +4072,7 @@ dependencies = [
"chrono", "chrono",
"color-eyre", "color-eyre",
"displaydoc", "displaydoc",
"ed25519-zebra 2.2.0 (git+https://github.com/ZcashFoundation/ed25519-zebra?rev=856c96500125e8dd38a525dad13dc64a0ac672cc)", "ed25519-zebra 2.2.0 (git+https://github.com/ZcashFoundation/ed25519-zebra?rev=539fad040c443302775b0f508e616418825e6c22)",
"equihash", "equihash",
"futures 0.3.13", "futures 0.3.13",
"hex", "hex",

View File

@ -13,6 +13,7 @@ mod tests;
// XXX clean up these modules // XXX clean up these modules
pub mod keys; pub mod keys;
pub mod shielded_data;
pub mod tree; pub mod tree;
pub use address::Address; pub use address::Address;
@ -20,4 +21,5 @@ pub use commitment::{CommitmentRandomness, NoteCommitment, ValueCommitment};
pub use keys::Diversifier; pub use keys::Diversifier;
pub use note::{EncryptedNote, Note, Nullifier, WrappedNoteKey}; pub use note::{EncryptedNote, Note, Nullifier, WrappedNoteKey};
pub use output::Output; pub use output::Output;
pub use shielded_data::{AnchorVariant, PerSpendAnchor, SharedAnchor, ShieldedData};
pub use spend::Spend; pub use spend::Spend;

View File

@ -3,9 +3,9 @@ use proptest::{arbitrary::any, array, collection::vec, prelude::*};
use crate::primitives::Groth16Proof; use crate::primitives::Groth16Proof;
use super::{keys, note, tree, NoteCommitment, Output, Spend, ValueCommitment}; use super::{keys, note, tree, NoteCommitment, Output, PerSpendAnchor, Spend, ValueCommitment};
impl Arbitrary for Spend { impl Arbitrary for Spend<PerSpendAnchor> {
type Parameters = (); type Parameters = ();
fn arbitrary_with(_args: Self::Parameters) -> Self::Strategy { fn arbitrary_with(_args: Self::Parameters) -> Self::Strategy {
@ -16,18 +16,20 @@ impl Arbitrary for Spend {
any::<Groth16Proof>(), any::<Groth16Proof>(),
vec(any::<u8>(), 64), vec(any::<u8>(), 64),
) )
.prop_map(|(anchor, nullifier, rpk_bytes, proof, sig_bytes)| Self { .prop_map(
anchor, |(per_spend_anchor, nullifier, rpk_bytes, proof, sig_bytes)| Self {
cv: ValueCommitment(AffinePoint::identity()), per_spend_anchor,
nullifier, cv: ValueCommitment(AffinePoint::identity()),
rk: redjubjub::VerificationKeyBytes::from(rpk_bytes), nullifier,
zkproof: proof, rk: redjubjub::VerificationKeyBytes::from(rpk_bytes),
spend_auth_sig: redjubjub::Signature::from({ zkproof: proof,
let mut b = [0u8; 64]; spend_auth_sig: redjubjub::Signature::from({
b.copy_from_slice(sig_bytes.as_slice()); let mut b = [0u8; 64];
b b.copy_from_slice(sig_bytes.as_slice());
}), b
}) }),
},
)
.boxed() .boxed()
} }

View File

@ -1,12 +1,60 @@
//! Sapling shielded data for `V4` and `V5` `Transaction`s.
//!
//! Zebra uses a generic shielded data type for `V4` and `V5` transactions.
//! The `value_balance` change is handled using the default zero value.
//! The anchor change is handled using the `AnchorVariant` type trait.
use futures::future::Either; use futures::future::Either;
use crate::{ use crate::{
amount::Amount, amount::Amount,
primitives::redjubjub::{Binding, Signature}, primitives::redjubjub::{Binding, Signature},
sapling::{Nullifier, Output, Spend, ValueCommitment}, sapling::{tree, Nullifier, Output, Spend, ValueCommitment},
serialization::serde_helpers, serialization::serde_helpers,
}; };
use serde::{de::DeserializeOwned, Serialize};
use std::{
cmp::{Eq, PartialEq},
fmt::Debug,
};
/// Per-Spend Sapling anchors, used in Transaction V4 and the
/// `spends_per_anchor` method.
#[derive(Copy, Clone, Debug, Deserialize, Serialize, PartialEq, Eq)]
pub struct PerSpendAnchor {}
/// Shared Sapling anchors, used in Transaction V5.
#[derive(Copy, Clone, Debug, Deserialize, Serialize, PartialEq, Eq)]
pub struct SharedAnchor {}
impl AnchorVariant for PerSpendAnchor {
type Shared = ();
type PerSpend = tree::Root;
}
impl AnchorVariant for SharedAnchor {
type Shared = tree::Root;
type PerSpend = ();
}
/// A type trait to handle structural differences between V4 and V5 Sapling
/// Transaction anchors.
///
/// In Transaction V4, anchors are per-Spend. In Transaction V5, there is a
/// single transaction anchor for all Spends in a transaction.
pub trait AnchorVariant {
/// The type of the shared anchor.
///
/// `()` means "not present in this transaction version".
type Shared: Clone + Debug + DeserializeOwned + Serialize + Eq + PartialEq;
/// The type of the per-spend anchor.
///
/// `()` means "not present in this transaction version".
type PerSpend: Clone + Debug + DeserializeOwned + Serialize + Eq + PartialEq;
}
/// A bundle of [`Spend`] and [`Output`] descriptions and signature data. /// A bundle of [`Spend`] and [`Output`] descriptions and signature data.
/// ///
/// Spend and Output descriptions are optional, but Zcash transactions must /// Spend and Output descriptions are optional, but Zcash transactions must
@ -15,8 +63,29 @@ use crate::{
/// description with the required signature data, so that an /// description with the required signature data, so that an
/// `Option<ShieldedData>` correctly models the presence or absence of any /// `Option<ShieldedData>` correctly models the presence or absence of any
/// shielded data. /// shielded data.
///
/// # Differences between Transaction Versions
///
/// The Sapling `value_balance` field is optional in `Transaction::V5`, but
/// required in `Transaction::V4`. In both cases, if there is no `ShieldedData`,
/// then the field value must be zero. Therefore, only need to store
/// `value_balance` when there is some Sapling `ShieldedData`.
///
/// In `Transaction::V4`, each `Spend` has its own anchor. In `Transaction::V5`,
/// there is a single `shared_anchor` for the entire transaction. This
/// structural difference is modeled using the `AnchorVariant` type trait.
/// A type of `()` means "not present in this transaction version".
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize)]
pub struct ShieldedData { pub struct ShieldedData<AnchorV>
where
AnchorV: AnchorVariant + Clone,
{
/// The net value of Sapling spend transfers minus output transfers.
pub value_balance: Amount,
/// The shared anchor for all `Spend`s in this transaction.
///
/// A type of `()` means "not present in this transaction version".
pub shared_anchor: AnchorV::Shared,
/// Either a spend or output description. /// Either a spend or output description.
/// ///
/// Storing this separately ensures that it is impossible to construct /// Storing this separately ensures that it is impossible to construct
@ -27,12 +96,12 @@ pub struct ShieldedData {
/// methods provide iterators over all of the [`Spend`]s and /// methods provide iterators over all of the [`Spend`]s and
/// [`Output`]s. /// [`Output`]s.
#[serde(with = "serde_helpers::Either")] #[serde(with = "serde_helpers::Either")]
pub first: Either<Spend, Output>, pub first: Either<Spend<AnchorV>, Output>,
/// The rest of the [`Spend`]s for this transaction. /// The rest of the [`Spend`]s for this transaction.
/// ///
/// Note that the [`ShieldedData::spends`] method provides an iterator /// Note that the [`ShieldedData::spends`] method provides an iterator
/// over all spend descriptions. /// over all spend descriptions.
pub rest_spends: Vec<Spend>, pub rest_spends: Vec<Spend<AnchorV>>,
/// The rest of the [`Output`]s for this transaction. /// The rest of the [`Output`]s for this transaction.
/// ///
/// Note that the [`ShieldedData::outputs`] method provides an iterator /// Note that the [`ShieldedData::outputs`] method provides an iterator
@ -42,9 +111,37 @@ pub struct ShieldedData {
pub binding_sig: Signature<Binding>, pub binding_sig: Signature<Binding>,
} }
impl ShieldedData { impl<AnchorV> ShieldedData<AnchorV>
where
AnchorV: AnchorVariant + Clone,
Spend<PerSpendAnchor>: From<(Spend<AnchorV>, AnchorV::Shared)>,
{
/// Iterate over the [`Spend`]s for this transaction. /// Iterate over the [`Spend`]s for this transaction.
pub fn spends(&self) -> impl Iterator<Item = &Spend> { ///
/// Returns `Spend<PerSpendAnchor>` regardless of the underlying transaction
/// version, to allow generic verification over V4 and V5 transactions.
///
/// # Correctness
///
/// Do not use this function for serialization.
pub fn spends_per_anchor(&self) -> impl Iterator<Item = Spend<PerSpendAnchor>> + '_ {
self.spends()
.cloned()
.map(move |spend| Spend::<PerSpendAnchor>::from((spend, self.shared_anchor.clone())))
}
}
impl<AnchorV> ShieldedData<AnchorV>
where
AnchorV: AnchorVariant + Clone,
{
/// Iterate over the [`Spend`]s for this transaction, returning them as
/// their generic type.
///
/// # Correctness
///
/// Use this function for serialization.
pub fn spends(&self) -> impl Iterator<Item = &Spend<AnchorV>> {
match self.first { match self.first {
Either::Left(ref spend) => Some(spend), Either::Left(ref spend) => Some(spend),
Either::Right(_) => None, Either::Right(_) => None,
@ -96,13 +193,11 @@ impl ShieldedData {
/// descriptions of the transaction, and the balancing value. /// descriptions of the transaction, and the balancing value.
/// ///
/// https://zips.z.cash/protocol/protocol.pdf#saplingbalance /// https://zips.z.cash/protocol/protocol.pdf#saplingbalance
pub fn binding_verification_key( pub fn binding_verification_key(&self) -> redjubjub::VerificationKeyBytes<Binding> {
&self,
value_balance: Amount,
) -> redjubjub::VerificationKeyBytes<Binding> {
let cv_old: ValueCommitment = self.spends().map(|spend| spend.cv).sum(); let cv_old: ValueCommitment = self.spends().map(|spend| spend.cv).sum();
let cv_new: ValueCommitment = self.outputs().map(|output| output.cv).sum(); let cv_new: ValueCommitment = self.outputs().map(|output| output.cv).sum();
let cv_balance: ValueCommitment = ValueCommitment::new(jubjub::Fr::zero(), value_balance); let cv_balance: ValueCommitment =
ValueCommitment::new(jubjub::Fr::zero(), self.value_balance);
let key_bytes: [u8; 32] = (cv_old - cv_new - cv_balance).into(); let key_bytes: [u8; 32] = (cv_old - cv_new - cv_balance).into();
@ -114,8 +209,14 @@ impl ShieldedData {
// of a ShieldedData with at least one spend and at least one output, depending // of a ShieldedData with at least one spend and at least one output, depending
// on which goes in the `first` slot. This is annoying but a smallish price to // on which goes in the `first` slot. This is annoying but a smallish price to
// pay for structural validity. // pay for structural validity.
//
// A `ShieldedData<PerSpendAnchor>` can never be equal to a
// `ShieldedData<SharedAnchor>`, even if they have the same effects.
impl std::cmp::PartialEq for ShieldedData { impl<AnchorV> std::cmp::PartialEq for ShieldedData<AnchorV>
where
AnchorV: AnchorVariant + Clone + PartialEq,
{
fn eq(&self, other: &Self) -> bool { fn eq(&self, other: &Self) -> bool {
// First check that the lengths match, so we know it is safe to use zip, // First check that the lengths match, so we know it is safe to use zip,
// which truncates to the shorter of the two iterators. // which truncates to the shorter of the two iterators.
@ -126,11 +227,14 @@ impl std::cmp::PartialEq for ShieldedData {
return false; return false;
} }
// Now check that the binding_sig, spends, outputs match. // Now check that all the fields match
self.binding_sig == other.binding_sig self.value_balance == other.value_balance
&& self.shared_anchor == other.shared_anchor
&& self.binding_sig == other.binding_sig
&& self.spends().zip(other.spends()).all(|(a, b)| a == b) && self.spends().zip(other.spends()).all(|(a, b)| a == b)
&& self.outputs().zip(other.outputs()).all(|(a, b)| a == b) && self.outputs().zip(other.outputs()).all(|(a, b)| a == b)
} }
} }
impl std::cmp::Eq for ShieldedData {} impl<AnchorV> std::cmp::Eq for ShieldedData<AnchorV> where AnchorV: AnchorVariant + Clone + PartialEq
{}

View File

@ -1,3 +1,8 @@
//! Sapling spends for `V4` and `V5` `Transaction`s.
//!
//! Zebra uses a generic spend type for `V4` and `V5` transactions.
//! The anchor change is handled using the `AnchorVariant` type trait.
use std::io; use std::io;
use crate::{ use crate::{
@ -10,17 +15,26 @@ use crate::{
}, },
}; };
use super::{commitment, note, tree}; use super::{commitment, note, tree, AnchorVariant, PerSpendAnchor, SharedAnchor};
/// A _Spend Description_, as described in [protocol specification §7.3][ps]. /// A _Spend Description_, as described in [protocol specification §7.3][ps].
/// ///
/// # Differences between Transaction Versions
///
/// In `Transaction::V4`, each `Spend` has its own anchor. In `Transaction::V5`,
/// there is a single `shared_anchor` for the entire transaction. This
/// structural difference is modeled using the `AnchorVariant` type trait.
/// A type of `()` means "not present in this transaction version".
///
/// [ps]: https://zips.z.cash/protocol/protocol.pdf#spendencoding /// [ps]: https://zips.z.cash/protocol/protocol.pdf#spendencoding
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq)]
pub struct Spend { pub struct Spend<AnchorV: AnchorVariant> {
/// A value commitment to the value of the input note. /// A value commitment to the value of the input note.
pub cv: commitment::ValueCommitment, pub cv: commitment::ValueCommitment,
/// A root of the Sapling note commitment tree at some block height in the past. /// A root of the Sapling note commitment tree at some block height in the past.
pub anchor: tree::Root, ///
/// A type of `()` means "not present in this transaction version".
pub per_spend_anchor: AnchorV::PerSpend,
/// The nullifier of the input note. /// The nullifier of the input note.
pub nullifier: note::Nullifier, pub nullifier: note::Nullifier,
/// The randomized public key for `spend_auth_sig`. /// The randomized public key for `spend_auth_sig`.
@ -31,7 +45,29 @@ pub struct Spend {
pub spend_auth_sig: redjubjub::Signature<SpendAuth>, pub spend_auth_sig: redjubjub::Signature<SpendAuth>,
} }
impl Spend { impl From<(Spend<SharedAnchor>, tree::Root)> for Spend<PerSpendAnchor> {
/// Convert a `Spend<SharedAnchor>` and its shared anchor, into a
/// `Spend<PerSpendAnchor>`.
fn from(shared_spend: (Spend<SharedAnchor>, tree::Root)) -> Self {
Spend::<PerSpendAnchor> {
per_spend_anchor: shared_spend.1,
cv: shared_spend.0.cv,
nullifier: shared_spend.0.nullifier,
rk: shared_spend.0.rk,
zkproof: shared_spend.0.zkproof,
spend_auth_sig: shared_spend.0.spend_auth_sig,
}
}
}
impl From<(Spend<PerSpendAnchor>, ())> for Spend<PerSpendAnchor> {
/// Take the `Spend<PerSpendAnchor>` from a spend + anchor tuple.
fn from(per_spend: (Spend<PerSpendAnchor>, ())) -> Self {
per_spend.0
}
}
impl Spend<PerSpendAnchor> {
/// Encodes the primary inputs for the proof statement as 7 Bls12_381 base /// Encodes the primary inputs for the proof statement as 7 Bls12_381 base
/// field elements, to match bellman::groth16::verify_proof. /// field elements, to match bellman::groth16::verify_proof.
/// ///
@ -49,7 +85,8 @@ impl Spend {
inputs.push(cv_affine.get_u()); inputs.push(cv_affine.get_u());
inputs.push(cv_affine.get_v()); inputs.push(cv_affine.get_v());
inputs.push(jubjub::Fq::from_bytes(&self.anchor.into()).unwrap()); // TODO: V4 only
inputs.push(jubjub::Fq::from_bytes(&self.per_spend_anchor.into()).unwrap());
let nullifier_limbs: [jubjub::Fq; 2] = self.nullifier.into(); let nullifier_limbs: [jubjub::Fq; 2] = self.nullifier.into();
@ -60,10 +97,11 @@ impl Spend {
} }
} }
impl ZcashSerialize for Spend { impl ZcashSerialize for Spend<PerSpendAnchor> {
fn zcash_serialize<W: io::Write>(&self, mut writer: W) -> Result<(), io::Error> { fn zcash_serialize<W: io::Write>(&self, mut writer: W) -> Result<(), io::Error> {
self.cv.zcash_serialize(&mut writer)?; self.cv.zcash_serialize(&mut writer)?;
writer.write_all(&self.anchor.0[..])?; // TODO: V4 only
writer.write_all(&self.per_spend_anchor.0[..])?;
writer.write_32_bytes(&self.nullifier.into())?; writer.write_32_bytes(&self.nullifier.into())?;
writer.write_all(&<[u8; 32]>::from(self.rk)[..])?; writer.write_all(&<[u8; 32]>::from(self.rk)[..])?;
self.zkproof.zcash_serialize(&mut writer)?; self.zkproof.zcash_serialize(&mut writer)?;
@ -72,12 +110,13 @@ impl ZcashSerialize for Spend {
} }
} }
impl ZcashDeserialize for Spend { impl ZcashDeserialize for Spend<PerSpendAnchor> {
fn zcash_deserialize<R: io::Read>(mut reader: R) -> Result<Self, SerializationError> { fn zcash_deserialize<R: io::Read>(mut reader: R) -> Result<Self, SerializationError> {
use crate::sapling::{commitment::ValueCommitment, note::Nullifier}; use crate::sapling::{commitment::ValueCommitment, note::Nullifier};
Ok(Spend { Ok(Spend {
cv: ValueCommitment::zcash_deserialize(&mut reader)?, cv: ValueCommitment::zcash_deserialize(&mut reader)?,
anchor: tree::Root(reader.read_32_bytes()?), // TODO: V4 only
per_spend_anchor: tree::Root(reader.read_32_bytes()?),
nullifier: Nullifier::from(reader.read_32_bytes()?), nullifier: Nullifier::from(reader.read_32_bytes()?),
rk: reader.read_32_bytes()?.into(), rk: reader.read_32_bytes()?.into(),
zkproof: Groth16Proof::zcash_deserialize(&mut reader)?, zkproof: Groth16Proof::zcash_deserialize(&mut reader)?,

View File

@ -1 +1 @@
mod prop;

View File

@ -0,0 +1,68 @@
use proptest::prelude::*;
use super::super::super::transaction::*;
use super::super::shielded_data::*;
use crate::{
block,
serialization::{ZcashDeserializeInto, ZcashSerialize},
};
proptest! {
#[test]
fn shielded_data_roundtrip(shielded in any::<ShieldedData<PerSpendAnchor>>()) {
zebra_test::init();
// shielded data doesn't serialize by itself, so we have to stick it in
// a transaction
let tx = Transaction::V4 {
inputs: Vec::new(),
outputs: Vec::new(),
lock_time: LockTime::min_lock_time(),
expiry_height: block::Height(0),
joinsplit_data: None,
sapling_shielded_data: Some(shielded),
};
let data = tx.zcash_serialize_to_vec().expect("tx should serialize");
let tx_parsed = data.zcash_deserialize_into().expect("randomized tx should deserialize");
prop_assert_eq![tx, tx_parsed];
}
/// Check that ShieldedData serialization is equal if `shielded1 == shielded2`
#[test]
fn shielded_data_serialize_eq(shielded1 in any::<ShieldedData<PerSpendAnchor>>(), shielded2 in any::<ShieldedData<PerSpendAnchor>>()) {
zebra_test::init();
let shielded_eq = shielded1 == shielded2;
// shielded data doesn't serialize by itself, so we have to stick it in
// a transaction
let tx1 = Transaction::V4 {
inputs: Vec::new(),
outputs: Vec::new(),
lock_time: LockTime::min_lock_time(),
expiry_height: block::Height(0),
joinsplit_data: None,
sapling_shielded_data: Some(shielded1),
};
let tx2 = Transaction::V4 {
inputs: Vec::new(),
outputs: Vec::new(),
lock_time: LockTime::min_lock_time(),
expiry_height: block::Height(0),
joinsplit_data: None,
sapling_shielded_data: Some(shielded2),
};
let data1 = tx1.zcash_serialize_to_vec().expect("tx1 should serialize");
let data2 = tx2.zcash_serialize_to_vec().expect("tx2 should serialize");
if shielded_eq {
prop_assert_eq![data1, data2];
} else {
prop_assert_ne![data1, data2];
}
}
}

View File

@ -7,7 +7,6 @@ mod joinsplit;
mod lock_time; mod lock_time;
mod memo; mod memo;
mod serialize; mod serialize;
mod shielded_data;
mod sighash; mod sighash;
#[cfg(any(test, feature = "proptest-impl"))] #[cfg(any(test, feature = "proptest-impl"))]
@ -19,11 +18,9 @@ pub use hash::Hash;
pub use joinsplit::JoinSplitData; pub use joinsplit::JoinSplitData;
pub use lock_time::LockTime; pub use lock_time::LockTime;
pub use memo::Memo; pub use memo::Memo;
pub use shielded_data::ShieldedData;
pub use sighash::HashType; pub use sighash::HashType;
use crate::{ use crate::{
amount::Amount,
block, block,
parameters::NetworkUpgrade, parameters::NetworkUpgrade,
primitives::{Bctv14Proof, Groth16Proof}, primitives::{Bctv14Proof, Groth16Proof},
@ -92,16 +89,12 @@ pub enum Transaction {
lock_time: LockTime, lock_time: LockTime,
/// The latest block height that this transaction can be added to the chain. /// The latest block height that this transaction can be added to the chain.
expiry_height: block::Height, expiry_height: block::Height,
/// The net value of Sapling spend transfers minus output transfers.
value_balance: Amount,
/// The JoinSplit data for this transaction, if any. /// The JoinSplit data for this transaction, if any.
joinsplit_data: Option<JoinSplitData<Groth16Proof>>, joinsplit_data: Option<JoinSplitData<Groth16Proof>>,
/// The shielded data for this transaction, if any. /// The sapling shielded data for this transaction, if any.
shielded_data: Option<ShieldedData>, sapling_shielded_data: Option<sapling::ShieldedData<sapling::PerSpendAnchor>>,
}, },
/// A `version = 5` transaction, which supports `Sapling` and `Orchard`. /// A `version = 5` transaction, which supports `Sapling` and `Orchard`.
// TODO: does this transaction type support `Sprout`?
// Check for ZIP-225 updates after the decision on 2021-03-05.
V5 { V5 {
/// The earliest time or block height that this transaction can be added to the /// The earliest time or block height that this transaction can be added to the
/// chain. /// chain.
@ -224,9 +217,9 @@ impl Transaction {
match self { match self {
// JoinSplits with Groth Proofs // JoinSplits with Groth Proofs
Transaction::V4 { Transaction::V4 {
shielded_data: Some(shielded_data), sapling_shielded_data: Some(sapling_shielded_data),
.. ..
} => Box::new(shielded_data.nullifiers()), } => Box::new(sapling_shielded_data.nullifiers()),
Transaction::V5 { .. } => { Transaction::V5 { .. } => {
unimplemented!("v5 transaction format as specified in ZIP-225") unimplemented!("v5 transaction format as specified in ZIP-225")
} }
@ -235,7 +228,7 @@ impl Transaction {
| Transaction::V2 { .. } | Transaction::V2 { .. }
| Transaction::V3 { .. } | Transaction::V3 { .. }
| Transaction::V4 { | Transaction::V4 {
shielded_data: None, sapling_shielded_data: None,
.. ..
} => Box::new(std::iter::empty()), } => Box::new(std::iter::empty()),
} }

View File

@ -14,7 +14,7 @@ use crate::{
sapling, sprout, transparent, sapling, sprout, transparent,
}; };
use super::{JoinSplitData, LockTime, Memo, ShieldedData, Transaction}; use super::{JoinSplitData, LockTime, Memo, Transaction};
impl Transaction { impl Transaction {
/// Generate a proptest strategy for V1 Transactions /// Generate a proptest strategy for V1 Transactions
@ -79,8 +79,7 @@ impl Transaction {
vec(any::<transparent::Output>(), 0..10), vec(any::<transparent::Output>(), 0..10),
any::<LockTime>(), any::<LockTime>(),
any::<block::Height>(), any::<block::Height>(),
any::<Amount>(), option::of(any::<sapling::ShieldedData<sapling::PerSpendAnchor>>()),
option::of(any::<ShieldedData>()),
option::of(any::<JoinSplitData<Groth16Proof>>()), option::of(any::<JoinSplitData<Groth16Proof>>()),
) )
.prop_map( .prop_map(
@ -89,21 +88,20 @@ impl Transaction {
outputs, outputs,
lock_time, lock_time,
expiry_height, expiry_height,
value_balance, sapling_shielded_data,
shielded_data,
joinsplit_data, joinsplit_data,
)| Transaction::V4 { )| Transaction::V4 {
inputs, inputs,
outputs, outputs,
lock_time, lock_time,
expiry_height, expiry_height,
value_balance, sapling_shielded_data,
shielded_data,
joinsplit_data, joinsplit_data,
}, },
) )
.boxed() .boxed()
} }
/// Generate a proptest strategy for V5 Transactions /// Generate a proptest strategy for V5 Transactions
pub fn v5_strategy(ledger_state: LedgerState) -> BoxedStrategy<Self> { pub fn v5_strategy(ledger_state: LedgerState) -> BoxedStrategy<Self> {
( (
@ -205,29 +203,34 @@ impl<P: ZkSnarkProof + Arbitrary + 'static> Arbitrary for JoinSplitData<P> {
type Strategy = BoxedStrategy<Self>; type Strategy = BoxedStrategy<Self>;
} }
impl Arbitrary for ShieldedData { impl Arbitrary for sapling::ShieldedData<sapling::PerSpendAnchor> {
type Parameters = (); type Parameters = ();
fn arbitrary_with(_args: Self::Parameters) -> Self::Strategy { fn arbitrary_with(_args: Self::Parameters) -> Self::Strategy {
( (
any::<Amount>(),
prop_oneof![ prop_oneof![
any::<sapling::Spend>().prop_map(Either::Left), any::<sapling::Spend<sapling::PerSpendAnchor>>().prop_map(Either::Left),
any::<sapling::Output>().prop_map(Either::Right) any::<sapling::Output>().prop_map(Either::Right)
], ],
vec(any::<sapling::Spend>(), 0..10), vec(any::<sapling::Spend<sapling::PerSpendAnchor>>(), 0..10),
vec(any::<sapling::Output>(), 0..10), vec(any::<sapling::Output>(), 0..10),
vec(any::<u8>(), 64), vec(any::<u8>(), 64),
) )
.prop_map(|(first, rest_spends, rest_outputs, sig_bytes)| Self { .prop_map(
first, |(value_balance, first, rest_spends, rest_outputs, sig_bytes)| Self {
rest_spends, value_balance,
rest_outputs, shared_anchor: (),
binding_sig: redjubjub::Signature::from({ first,
let mut b = [0u8; 64]; rest_spends,
b.copy_from_slice(sig_bytes.as_slice()); rest_outputs,
b binding_sig: redjubjub::Signature::from({
}), let mut b = [0u8; 64];
}) b.copy_from_slice(sig_bytes.as_slice());
b
}),
},
)
.boxed() .boxed()
} }

View File

@ -131,8 +131,7 @@ impl ZcashSerialize for Transaction {
outputs, outputs,
lock_time, lock_time,
expiry_height, expiry_height,
value_balance, sapling_shielded_data,
shielded_data,
joinsplit_data, joinsplit_data,
} => { } => {
// Write version 4 and set the fOverwintered bit. // Write version 4 and set the fOverwintered bit.
@ -142,7 +141,6 @@ impl ZcashSerialize for Transaction {
outputs.zcash_serialize(&mut writer)?; outputs.zcash_serialize(&mut writer)?;
lock_time.zcash_serialize(&mut writer)?; lock_time.zcash_serialize(&mut writer)?;
writer.write_u32::<LittleEndian>(expiry_height.0)?; writer.write_u32::<LittleEndian>(expiry_height.0)?;
value_balance.zcash_serialize(&mut writer)?;
// The previous match arms serialize in one go, because the // The previous match arms serialize in one go, because the
// internal structure happens to nicely line up with the // internal structure happens to nicely line up with the
@ -152,13 +150,16 @@ impl ZcashSerialize for Transaction {
// instead we have to interleave serialization of the // instead we have to interleave serialization of the
// ShieldedData and the JoinSplitData. // ShieldedData and the JoinSplitData.
match shielded_data { match sapling_shielded_data {
None => { None => {
// Signal no value balance.
writer.write_i64::<LittleEndian>(0)?;
// Signal no shielded spends and no shielded outputs. // Signal no shielded spends and no shielded outputs.
writer.write_compactsize(0)?; writer.write_compactsize(0)?;
writer.write_compactsize(0)?; writer.write_compactsize(0)?;
} }
Some(shielded_data) => { Some(shielded_data) => {
shielded_data.value_balance.zcash_serialize(&mut writer)?;
writer.write_compactsize(shielded_data.spends().count() as u64)?; writer.write_compactsize(shielded_data.spends().count() as u64)?;
for spend in shielded_data.spends() { for spend in shielded_data.spends() {
spend.zcash_serialize(&mut writer)?; spend.zcash_serialize(&mut writer)?;
@ -175,7 +176,7 @@ impl ZcashSerialize for Transaction {
Some(jsd) => jsd.zcash_serialize(&mut writer)?, Some(jsd) => jsd.zcash_serialize(&mut writer)?,
} }
match shielded_data { match sapling_shielded_data {
Some(sd) => writer.write_all(&<[u8; 64]>::from(sd.binding_sig)[..])?, Some(sd) => writer.write_all(&<[u8; 64]>::from(sd.binding_sig)[..])?,
None => {} None => {}
} }
@ -194,6 +195,7 @@ impl ZcashSerialize for Transaction {
writer.write_u32::<LittleEndian>(expiry_height.0)?; writer.write_u32::<LittleEndian>(expiry_height.0)?;
inputs.zcash_serialize(&mut writer)?; inputs.zcash_serialize(&mut writer)?;
outputs.zcash_serialize(&mut writer)?; outputs.zcash_serialize(&mut writer)?;
// write the rest // write the rest
writer.write_all(rest)?; writer.write_all(rest)?;
} }
@ -266,22 +268,31 @@ impl ZcashDeserialize for Transaction {
let outputs = Vec::zcash_deserialize(&mut reader)?; let outputs = Vec::zcash_deserialize(&mut reader)?;
let lock_time = LockTime::zcash_deserialize(&mut reader)?; let lock_time = LockTime::zcash_deserialize(&mut reader)?;
let expiry_height = block::Height(reader.read_u32::<LittleEndian>()?); let expiry_height = block::Height(reader.read_u32::<LittleEndian>()?);
let value_balance = (&mut reader).zcash_deserialize_into()?; let value_balance = (&mut reader).zcash_deserialize_into()?;
let mut shielded_spends = Vec::zcash_deserialize(&mut reader)?; let mut shielded_spends = Vec::zcash_deserialize(&mut reader)?;
let mut shielded_outputs = Vec::zcash_deserialize(&mut reader)?; let mut shielded_outputs = Vec::zcash_deserialize(&mut reader)?;
let joinsplit_data = OptV4Jsd::zcash_deserialize(&mut reader)?; let joinsplit_data = OptV4Jsd::zcash_deserialize(&mut reader)?;
use futures::future::Either::*; use futures::future::Either::*;
let shielded_data = if !shielded_spends.is_empty() { // Arbitraily use a spend for `first`, if both are present
Some(ShieldedData { let sapling_shielded_data = if !shielded_spends.is_empty() {
Some(sapling::ShieldedData {
value_balance,
shared_anchor: (),
first: Left(shielded_spends.remove(0)), first: Left(shielded_spends.remove(0)),
rest_spends: shielded_spends, rest_spends: shielded_spends,
rest_outputs: shielded_outputs, rest_outputs: shielded_outputs,
binding_sig: reader.read_64_bytes()?.into(), binding_sig: reader.read_64_bytes()?.into(),
}) })
} else if !shielded_outputs.is_empty() { } else if !shielded_outputs.is_empty() {
Some(ShieldedData { Some(sapling::ShieldedData {
value_balance,
shared_anchor: (),
first: Right(shielded_outputs.remove(0)), first: Right(shielded_outputs.remove(0)),
// the spends are actually empty here, but we use the
// vec for consistency and readability
rest_spends: shielded_spends, rest_spends: shielded_spends,
rest_outputs: shielded_outputs, rest_outputs: shielded_outputs,
binding_sig: reader.read_64_bytes()?.into(), binding_sig: reader.read_64_bytes()?.into(),
@ -295,8 +306,7 @@ impl ZcashDeserialize for Transaction {
outputs, outputs,
lock_time, lock_time,
expiry_height, expiry_height,
value_balance, sapling_shielded_data,
shielded_data,
joinsplit_data, joinsplit_data,
}) })
} }
@ -309,6 +319,7 @@ impl ZcashDeserialize for Transaction {
let expiry_height = block::Height(reader.read_u32::<LittleEndian>()?); let expiry_height = block::Height(reader.read_u32::<LittleEndian>()?);
let inputs = Vec::zcash_deserialize(&mut reader)?; let inputs = Vec::zcash_deserialize(&mut reader)?;
let outputs = Vec::zcash_deserialize(&mut reader)?; let outputs = Vec::zcash_deserialize(&mut reader)?;
let mut rest = Vec::new(); let mut rest = Vec::new();
reader.read_to_end(&mut rest)?; reader.read_to_end(&mut rest)?;

View File

@ -419,11 +419,11 @@ impl<'a> SigHasher<'a> {
let shielded_data = match self.trans { let shielded_data = match self.trans {
V4 { V4 {
shielded_data: Some(shielded_data), sapling_shielded_data: Some(shielded_data),
.. ..
} => shielded_data, } => shielded_data,
V4 { V4 {
shielded_data: None, sapling_shielded_data: None,
.. ..
} => return writer.write_all(&[0; 32]), } => return writer.write_all(&[0; 32]),
V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"), V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"),
@ -439,10 +439,12 @@ impl<'a> SigHasher<'a> {
.personal(ZCASH_SHIELDED_SPENDS_HASH_PERSONALIZATION) .personal(ZCASH_SHIELDED_SPENDS_HASH_PERSONALIZATION)
.to_state(); .to_state();
// TODO: make a generic wrapper in `spends.rs` that does this serialization
for spend in shielded_data.spends() { for spend in shielded_data.spends() {
// This is the canonical transaction serialization, minus the `spendAuthSig`. // This is the canonical transaction serialization, minus the `spendAuthSig`.
spend.cv.zcash_serialize(&mut hash)?; spend.cv.zcash_serialize(&mut hash)?;
hash.write_all(&spend.anchor.0[..])?; // TODO: ZIP-243 Sapling to Canopy only
hash.write_all(&spend.per_spend_anchor.0[..])?;
hash.write_32_bytes(&spend.nullifier.into())?; hash.write_32_bytes(&spend.nullifier.into())?;
hash.write_all(&<[u8; 32]>::from(spend.rk)[..])?; hash.write_all(&<[u8; 32]>::from(spend.rk)[..])?;
spend.zkproof.zcash_serialize(&mut hash)?; spend.zkproof.zcash_serialize(&mut hash)?;
@ -456,11 +458,11 @@ impl<'a> SigHasher<'a> {
let shielded_data = match self.trans { let shielded_data = match self.trans {
V4 { V4 {
shielded_data: Some(shielded_data), sapling_shielded_data: Some(shielded_data),
.. ..
} => shielded_data, } => shielded_data,
V4 { V4 {
shielded_data: None, sapling_shielded_data: None,
.. ..
} => return writer.write_all(&[0; 32]), } => return writer.write_all(&[0; 32]),
V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"), V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"),
@ -484,10 +486,18 @@ impl<'a> SigHasher<'a> {
} }
fn hash_value_balance<W: io::Write>(&self, mut writer: W) -> Result<(), io::Error> { fn hash_value_balance<W: io::Write>(&self, mut writer: W) -> Result<(), io::Error> {
use crate::amount::Amount;
use std::convert::TryFrom;
use Transaction::*; use Transaction::*;
let value_balance = match self.trans { let value_balance = match self.trans {
V4 { value_balance, .. } => value_balance, V4 {
sapling_shielded_data,
..
} => match sapling_shielded_data {
Some(s) => s.value_balance,
None => Amount::try_from(0).unwrap(),
},
V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"), V5 { .. } => unimplemented!("v5 transaction hash as specified in ZIP-225 and ZIP-244"),
V1 { .. } | V2 { .. } | V3 { .. } => unreachable!(ZIP243_EXPLANATION), V1 { .. } | V2 { .. } | V3 { .. } => unreachable!(ZIP243_EXPLANATION),
}; };

View File

@ -20,7 +20,7 @@ use tokio::sync::broadcast::{channel, error::RecvError, Sender};
use tower::{util::ServiceFn, Service}; use tower::{util::ServiceFn, Service};
use tower_batch::{Batch, BatchControl}; use tower_batch::{Batch, BatchControl};
use tower_fallback::Fallback; use tower_fallback::Fallback;
use zebra_chain::sapling::{Output, Spend}; use zebra_chain::sapling::{Output, PerSpendAnchor, Spend};
mod hash_reader; mod hash_reader;
mod params; mod params;
@ -101,8 +101,8 @@ pub type Item = batch::Item<Bls12>;
pub struct ItemWrapper(Item); pub struct ItemWrapper(Item);
impl From<&Spend> for ItemWrapper { impl From<&Spend<PerSpendAnchor>> for ItemWrapper {
fn from(spend: &Spend) -> Self { fn from(spend: &Spend<PerSpendAnchor>) -> Self {
Self(Item::from(( Self(Item::from((
bellman::groth16::Proof::read(&spend.zkproof.0[..]).unwrap(), bellman::groth16::Proof::read(&spend.zkproof.0[..]).unwrap(),
spend.primary_inputs(), spend.primary_inputs(),

View File

@ -30,15 +30,18 @@ where
| Transaction::V2 { .. } | Transaction::V2 { .. }
| Transaction::V3 { .. } | Transaction::V3 { .. }
| Transaction::V5 { .. } => (), | Transaction::V5 { .. } => (),
Transaction::V4 { shielded_data, .. } => { Transaction::V4 {
if let Some(shielded_data) = shielded_data { sapling_shielded_data,
for spend in shielded_data.spends() { ..
} => {
if let Some(shielded_data) = sapling_shielded_data {
for spend in shielded_data.spends_per_anchor() {
tracing::trace!(?spend); tracing::trace!(?spend);
let spend_rsp = spend_verifier let spend_rsp = spend_verifier
.ready_and() .ready_and()
.await? .await?
.call(groth16::ItemWrapper::from(spend).into()); .call(groth16::ItemWrapper::from(&spend).into());
async_checks.push(spend_rsp); async_checks.push(spend_rsp);
} }
@ -110,8 +113,11 @@ where
| Transaction::V2 { .. } | Transaction::V2 { .. }
| Transaction::V3 { .. } | Transaction::V3 { .. }
| Transaction::V5 { .. } => (), | Transaction::V5 { .. } => (),
Transaction::V4 { shielded_data, .. } => { Transaction::V4 {
if let Some(shielded_data) = shielded_data { sapling_shielded_data,
..
} => {
if let Some(shielded_data) = sapling_shielded_data {
for output in shielded_data.outputs() { for output in shielded_data.outputs() {
// This changes the primary inputs to the proof // This changes the primary inputs to the proof
// verification, causing it to fail for this proof. // verification, causing it to fail for this proof.

View File

@ -147,9 +147,8 @@ where
// outputs, // outputs,
// lock_time, // lock_time,
// expiry_height, // expiry_height,
value_balance,
joinsplit_data, joinsplit_data,
shielded_data, sapling_shielded_data,
.. ..
} => { } => {
// A set of asynchronous checks which must all succeed. // A set of asynchronous checks which must all succeed.
@ -213,16 +212,16 @@ where
async_checks.push(rsp.boxed()); async_checks.push(rsp.boxed());
} }
if let Some(shielded_data) = shielded_data { if let Some(shielded_data) = sapling_shielded_data {
check::shielded_balances_match(&shielded_data, *value_balance)?; check::shielded_balances_match(&shielded_data)?;
for spend in shielded_data.spends() { for spend in shielded_data.spends_per_anchor() {
// Consensus rule: cv and rk MUST NOT be of small // Consensus rule: cv and rk MUST NOT be of small
// order, i.e. [h_J]cv MUST NOT be 𝒪_J and [h_J]rk // order, i.e. [h_J]cv MUST NOT be 𝒪_J and [h_J]rk
// MUST NOT be 𝒪_J. // MUST NOT be 𝒪_J.
// //
// https://zips.z.cash/protocol/protocol.pdf#spenddesc // https://zips.z.cash/protocol/protocol.pdf#spenddesc
check::spend_cv_rk_not_small_order(spend)?; check::spend_cv_rk_not_small_order(&spend)?;
// Consensus rule: The proof π_ZKSpend MUST be valid // Consensus rule: The proof π_ZKSpend MUST be valid
// given a primary input formed from the other // given a primary input formed from the other
@ -236,7 +235,7 @@ where
let spend_rsp = spend_verifier let spend_rsp = spend_verifier
.ready_and() .ready_and()
.await? .await?
.call(primitives::groth16::ItemWrapper::from(spend).into()); .call(primitives::groth16::ItemWrapper::from(&spend).into());
async_checks.push(spend_rsp.boxed()); async_checks.push(spend_rsp.boxed());
@ -282,7 +281,7 @@ where
async_checks.push(output_rsp.boxed()); async_checks.push(output_rsp.boxed());
} }
let bvk = shielded_data.binding_verification_key(*value_balance); let bvk = shielded_data.binding_verification_key();
// TODO: enable async verification and remove this block - #1939 // TODO: enable async verification and remove this block - #1939
{ {

View File

@ -3,9 +3,8 @@
//! Code in this file can freely assume that no pre-V4 transactions are present. //! Code in this file can freely assume that no pre-V4 transactions are present.
use zebra_chain::{ use zebra_chain::{
amount::Amount, sapling::{AnchorVariant, Output, PerSpendAnchor, ShieldedData, Spend},
sapling::{Output, Spend}, transaction::Transaction,
transaction::{ShieldedData, Transaction},
}; };
use crate::error::TransactionError; use crate::error::TransactionError;
@ -27,7 +26,7 @@ pub fn has_inputs_and_outputs(tx: &Transaction) -> Result<(), TransactionError>
inputs, inputs,
outputs, outputs,
joinsplit_data, joinsplit_data,
shielded_data, sapling_shielded_data,
.. ..
} => { } => {
let tx_in_count = inputs.len(); let tx_in_count = inputs.len();
@ -36,11 +35,11 @@ pub fn has_inputs_and_outputs(tx: &Transaction) -> Result<(), TransactionError>
.as_ref() .as_ref()
.map(|d| d.joinsplits().count()) .map(|d| d.joinsplits().count())
.unwrap_or(0); .unwrap_or(0);
let n_shielded_spend = shielded_data let n_shielded_spend = sapling_shielded_data
.as_ref() .as_ref()
.map(|d| d.spends().count()) .map(|d| d.spends().count())
.unwrap_or(0); .unwrap_or(0);
let n_shielded_output = shielded_data let n_shielded_output = sapling_shielded_data
.as_ref() .as_ref()
.map(|d| d.outputs().count()) .map(|d| d.outputs().count())
.unwrap_or(0); .unwrap_or(0);
@ -65,12 +64,14 @@ pub fn has_inputs_and_outputs(tx: &Transaction) -> Result<(), TransactionError>
/// Check that if there are no Spends or Outputs, that valueBalance is also 0. /// Check that if there are no Spends or Outputs, that valueBalance is also 0.
/// ///
/// https://zips.z.cash/protocol/protocol.pdf#consensusfrombitcoin /// https://zips.z.cash/protocol/protocol.pdf#consensusfrombitcoin
pub fn shielded_balances_match( pub fn shielded_balances_match<AnchorV>(
shielded_data: &ShieldedData, shielded_data: &ShieldedData<AnchorV>,
value_balance: Amount, ) -> Result<(), TransactionError>
) -> Result<(), TransactionError> { where
AnchorV: AnchorVariant + Clone,
{
if (shielded_data.spends().count() + shielded_data.outputs().count() != 0) if (shielded_data.spends().count() + shielded_data.outputs().count() != 0)
|| i64::from(value_balance) == 0 || i64::from(shielded_data.value_balance) == 0
{ {
Ok(()) Ok(())
} else { } else {
@ -93,9 +94,11 @@ pub fn coinbase_tx_no_joinsplit_or_spend(tx: &Transaction) -> Result<(), Transac
// The ShieldedData contains both Spends and Outputs, and Outputs // The ShieldedData contains both Spends and Outputs, and Outputs
// are allowed post-Heartwood, so we have to count Spends. // are allowed post-Heartwood, so we have to count Spends.
Transaction::V4 { Transaction::V4 {
shielded_data: Some(shielded_data), sapling_shielded_data: Some(sapling_shielded_data),
.. ..
} if shielded_data.spends().count() > 0 => Err(TransactionError::CoinbaseHasSpend), } if sapling_shielded_data.spends().count() > 0 => {
Err(TransactionError::CoinbaseHasSpend)
}
Transaction::V4 { .. } => Ok(()), Transaction::V4 { .. } => Ok(()),
@ -116,7 +119,7 @@ pub fn coinbase_tx_no_joinsplit_or_spend(tx: &Transaction) -> Result<(), Transac
/// i.e. [h_J]cv MUST NOT be 𝒪_J and [h_J]rk MUST NOT be 𝒪_J. /// i.e. [h_J]cv MUST NOT be 𝒪_J and [h_J]rk MUST NOT be 𝒪_J.
/// ///
/// https://zips.z.cash/protocol/protocol.pdf#spenddesc /// https://zips.z.cash/protocol/protocol.pdf#spenddesc
pub fn spend_cv_rk_not_small_order(spend: &Spend) -> Result<(), TransactionError> { pub fn spend_cv_rk_not_small_order(spend: &Spend<PerSpendAnchor>) -> Result<(), TransactionError> {
if bool::from(spend.cv.0.is_small_order()) if bool::from(spend.cv.0.is_small_order())
|| bool::from( || bool::from(
jubjub::AffinePoint::from_bytes(spend.rk.into()) jubjub::AffinePoint::from_bytes(spend.rk.into())

View File

@ -166,13 +166,13 @@ impl UpdateWith<PreparedBlock> for Chain {
.enumerate() .enumerate()
{ {
use transaction::Transaction::*; use transaction::Transaction::*;
let (inputs, shielded_data, joinsplit_data) = match transaction.deref() { let (inputs, joinsplit_data, sapling_shielded_data) = match transaction.deref() {
V4 { V4 {
inputs, inputs,
shielded_data,
joinsplit_data, joinsplit_data,
sapling_shielded_data,
.. ..
} => (inputs, shielded_data, joinsplit_data), } => (inputs, joinsplit_data, sapling_shielded_data),
V5 { .. } => unimplemented!("v5 transaction format as specified in ZIP-225"), V5 { .. } => unimplemented!("v5 transaction format as specified in ZIP-225"),
V1 { .. } | V2 { .. } | V3 { .. } => unreachable!( V1 { .. } | V2 { .. } | V3 { .. } => unreachable!(
"older transaction versions only exist in finalized blocks pre sapling", "older transaction versions only exist in finalized blocks pre sapling",
@ -195,7 +195,7 @@ impl UpdateWith<PreparedBlock> for Chain {
// add sprout anchor and nullifiers // add sprout anchor and nullifiers
self.update_chain_state_with(joinsplit_data); self.update_chain_state_with(joinsplit_data);
// add sapling anchor and nullifier // add sapling anchor and nullifier
self.update_chain_state_with(shielded_data); self.update_chain_state_with(sapling_shielded_data);
} }
} }
@ -226,13 +226,13 @@ impl UpdateWith<PreparedBlock> for Chain {
block.transactions.iter().zip(transaction_hashes.iter()) block.transactions.iter().zip(transaction_hashes.iter())
{ {
use transaction::Transaction::*; use transaction::Transaction::*;
let (inputs, shielded_data, joinsplit_data) = match transaction.deref() { let (inputs, joinsplit_data, sapling_shielded_data) = match transaction.deref() {
V4 { V4 {
inputs, inputs,
shielded_data,
joinsplit_data, joinsplit_data,
sapling_shielded_data,
.. ..
} => (inputs, shielded_data, joinsplit_data), } => (inputs, joinsplit_data, sapling_shielded_data),
V5 { .. } => unimplemented!("v5 transaction format as specified in ZIP-225"), V5 { .. } => unimplemented!("v5 transaction format as specified in ZIP-225"),
V1 { .. } | V2 { .. } | V3 { .. } => unreachable!( V1 { .. } | V2 { .. } | V3 { .. } => unreachable!(
"older transaction versions only exist in finalized blocks pre sapling", "older transaction versions only exist in finalized blocks pre sapling",
@ -252,7 +252,7 @@ impl UpdateWith<PreparedBlock> for Chain {
// remove sprout anchor and nullifiers // remove sprout anchor and nullifiers
self.revert_chain_state_with(joinsplit_data); self.revert_chain_state_with(joinsplit_data);
// remove sapling anchor and nullfier // remove sapling anchor and nullfier
self.revert_chain_state_with(shielded_data); self.revert_chain_state_with(sapling_shielded_data);
} }
} }
} }
@ -336,18 +336,21 @@ impl UpdateWith<Option<transaction::JoinSplitData<Groth16Proof>>> for Chain {
} }
} }
impl UpdateWith<Option<transaction::ShieldedData>> for Chain { impl<AnchorV> UpdateWith<Option<sapling::ShieldedData<AnchorV>>> for Chain
fn update_chain_state_with(&mut self, shielded_data: &Option<transaction::ShieldedData>) { where
AnchorV: sapling::AnchorVariant + Clone,
{
fn update_chain_state_with(&mut self, shielded_data: &Option<sapling::ShieldedData<AnchorV>>) {
if let Some(shielded_data) = shielded_data { if let Some(shielded_data) = shielded_data {
for sapling::Spend { nullifier, .. } in shielded_data.spends() { for nullifier in shielded_data.nullifiers() {
self.sapling_nullifiers.insert(*nullifier); self.sapling_nullifiers.insert(*nullifier);
} }
} }
} }
fn revert_chain_state_with(&mut self, shielded_data: &Option<transaction::ShieldedData>) { fn revert_chain_state_with(&mut self, shielded_data: &Option<sapling::ShieldedData<AnchorV>>) {
if let Some(shielded_data) = shielded_data { if let Some(shielded_data) = shielded_data {
for sapling::Spend { nullifier, .. } in shielded_data.spends() { for nullifier in shielded_data.nullifiers() {
assert!( assert!(
self.sapling_nullifiers.remove(nullifier), self.sapling_nullifiers.remove(nullifier),
"nullifier must be present if block was" "nullifier must be present if block was"