1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
use std::fmt;
use serde::{Deserialize, Serialize};
use crate::{
amount::{self, Amount, NegativeAllowed},
fmt::HexDebug,
primitives::{ed25519, ZkSnarkProof},
sprout::{self, JoinSplit, Nullifier},
};
/// A bundle of [`JoinSplit`] descriptions and signature data.
///
/// JoinSplit descriptions are optional, but Zcash transactions must include a
/// JoinSplit signature and verification key if and only if there is at least one
/// JoinSplit description. This wrapper type bundles at least one JoinSplit
/// description with the required signature data, so that an
/// `Option<JoinSplitData>` correctly models the presence or absence of any
/// JoinSplit data.
#[derive(Clone, PartialEq, Eq, Serialize, Deserialize)]
pub struct JoinSplitData<P: ZkSnarkProof> {
/// The first JoinSplit description in the transaction,
/// using proofs of type `P`.
///
/// Storing this separately from `rest` ensures that it is impossible
/// to construct an invalid `JoinSplitData` with no `JoinSplit`s.
///
/// However, it's not necessary to access or process `first` and `rest`
/// separately, as the [`JoinSplitData::joinsplits`] method provides an
/// iterator over all of the `JoinSplit`s.
#[serde(bound(
serialize = "JoinSplit<P>: Serialize",
deserialize = "JoinSplit<P>: Deserialize<'de>"
))]
pub first: JoinSplit<P>,
/// The rest of the JoinSplit descriptions, using proofs of type `P`,
/// in the order they appear in the transaction.
///
/// The [`JoinSplitData::joinsplits`] method provides an iterator over
/// all `JoinSplit`s.
#[serde(bound(
serialize = "JoinSplit<P>: Serialize",
deserialize = "JoinSplit<P>: Deserialize<'de>"
))]
pub rest: Vec<JoinSplit<P>>,
/// The public key for the JoinSplit signature, denoted as `joinSplitPubKey` in the spec.
pub pub_key: ed25519::VerificationKeyBytes,
/// The JoinSplit signature, denoted as `joinSplitSig` in the spec.
pub sig: ed25519::Signature,
}
impl<P: ZkSnarkProof> fmt::Debug for JoinSplitData<P> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("JoinSplitData")
.field("first", &self.first)
.field("rest", &self.rest)
.field("pub_key", &self.pub_key)
.field("sig", &HexDebug(&self.sig.to_bytes()))
.finish()
}
}
impl<P: ZkSnarkProof> fmt::Display for JoinSplitData<P> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let mut fmter =
f.debug_struct(format!("JoinSplitData<{}>", std::any::type_name::<P>()).as_str());
fmter.field("joinsplits", &self.joinsplits().count());
fmter.field("value_balance", &self.value_balance());
fmter.finish()
}
}
impl<P: ZkSnarkProof> JoinSplitData<P> {
/// Iterate over the [`JoinSplit`]s in `self`, in the order they appear
/// in the transaction.
pub fn joinsplits(&self) -> impl Iterator<Item = &JoinSplit<P>> {
std::iter::once(&self.first).chain(self.rest.iter())
}
/// Modify the [`JoinSplit`]s in `self`,
/// in the order they appear in the transaction.
#[cfg(any(test, feature = "proptest-impl"))]
pub fn joinsplits_mut(&mut self) -> impl Iterator<Item = &mut JoinSplit<P>> {
std::iter::once(&mut self.first).chain(self.rest.iter_mut())
}
/// Iterate over the [`Nullifier`]s in `self`.
pub fn nullifiers(&self) -> impl Iterator<Item = &Nullifier> {
self.joinsplits()
.flat_map(|joinsplit| joinsplit.nullifiers.iter())
}
/// Return the sprout value balance,
/// the change in the transaction value pool due to sprout [`JoinSplit`]s.
///
/// <https://zebra.zfnd.org/dev/rfcs/0012-value-pools.html#definitions>
///
/// See [`sprout_value_balance`][svb] for details.
///
/// [svb]: crate::transaction::Transaction::sprout_value_balance
pub fn value_balance(&self) -> Result<Amount<NegativeAllowed>, amount::Error> {
self.joinsplit_value_balances().sum()
}
/// Return a list of sprout value balances,
/// the changes in the transaction value pool due to each sprout [`JoinSplit`].
///
/// See [`sprout_value_balance`][svb] for details.
///
/// [svb]: crate::transaction::Transaction::sprout_value_balance
pub fn joinsplit_value_balances(
&self,
) -> Box<dyn Iterator<Item = Amount<NegativeAllowed>> + '_> {
Box::new(self.joinsplits().map(JoinSplit::value_balance))
}
/// Collect the Sprout note commitments for this transaction, if it contains `Output`s,
/// in the order they appear in the transaction.
pub fn note_commitments(&self) -> impl Iterator<Item = &sprout::commitment::NoteCommitment> {
self.joinsplits()
.flat_map(|joinsplit| &joinsplit.commitments)
}
}