penumbra_sdk_shielded_pool/output/
plan.rsuse decaf377::{Fq, Fr};
use decaf377_ka as ka;
use penumbra_sdk_asset::{Balance, Value, STAKING_TOKEN_ASSET_ID};
use penumbra_sdk_keys::{
keys::{IncomingViewingKey, OutgoingViewingKey},
symmetric::WrappedMemoKey,
Address, PayloadKey,
};
use penumbra_sdk_proto::{core::component::shielded_pool::v1 as pb, DomainType};
use rand_core::{CryptoRng, RngCore};
use serde::{Deserialize, Serialize};
use super::{Body, Output, OutputProof, OutputProofPrivate, OutputProofPublic};
use crate::{Note, Rseed};
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(try_from = "pb::OutputPlan", into = "pb::OutputPlan")]
pub struct OutputPlan {
pub value: Value,
pub dest_address: Address,
pub rseed: Rseed,
pub value_blinding: Fr,
pub proof_blinding_r: Fq,
pub proof_blinding_s: Fq,
}
impl OutputPlan {
pub fn new<R: RngCore + CryptoRng>(
rng: &mut R,
value: Value,
dest_address: Address,
) -> OutputPlan {
let rseed = Rseed::generate(rng);
let value_blinding = Fr::rand(rng);
Self {
value,
dest_address,
rseed,
value_blinding,
proof_blinding_r: Fq::rand(rng),
proof_blinding_s: Fq::rand(rng),
}
}
pub fn dummy<R: CryptoRng + RngCore>(rng: &mut R) -> OutputPlan {
let dummy_address = Address::dummy(rng);
Self::new(
rng,
Value {
amount: 0u64.into(),
asset_id: *STAKING_TOKEN_ASSET_ID,
},
dummy_address,
)
}
pub fn output(&self, ovk: &OutgoingViewingKey, memo_key: &PayloadKey) -> Output {
Output {
body: self.output_body(ovk, memo_key),
proof: self.output_proof(),
}
}
pub fn output_note(&self) -> Note {
Note::from_parts(self.dest_address.clone(), self.value, self.rseed)
.expect("transmission key in address is always valid")
}
pub fn output_proof(&self) -> OutputProof {
let note = self.output_note();
let balance_commitment = self.balance().commit(self.value_blinding);
let note_commitment = note.commit();
OutputProof::prove(
self.proof_blinding_r,
self.proof_blinding_s,
&penumbra_sdk_proof_params::OUTPUT_PROOF_PROVING_KEY,
OutputProofPublic {
balance_commitment,
note_commitment,
},
OutputProofPrivate {
note,
balance_blinding: self.value_blinding,
},
)
.expect("can generate ZKOutputProof")
}
pub fn output_body(&self, ovk: &OutgoingViewingKey, memo_key: &PayloadKey) -> Body {
let note = self.output_note();
let balance_commitment = self.balance().commit(self.value_blinding);
let esk: ka::Secret = note.ephemeral_secret_key();
let ovk_wrapped_key = note.encrypt_key(ovk, balance_commitment);
let wrapped_memo_key = WrappedMemoKey::encrypt(
memo_key,
esk,
note.transmission_key(),
¬e.diversified_generator(),
);
Body {
note_payload: note.payload(),
balance_commitment,
ovk_wrapped_key,
wrapped_memo_key,
}
}
pub fn is_viewed_by(&self, ivk: &IncomingViewingKey) -> bool {
ivk.views_address(&self.dest_address)
}
pub fn balance(&self) -> Balance {
-Balance::from(self.value)
}
}
impl DomainType for OutputPlan {
type Proto = pb::OutputPlan;
}
impl From<OutputPlan> for pb::OutputPlan {
fn from(msg: OutputPlan) -> Self {
Self {
value: Some(msg.value.into()),
dest_address: Some(msg.dest_address.into()),
rseed: msg.rseed.to_bytes().to_vec(),
value_blinding: msg.value_blinding.to_bytes().to_vec(),
proof_blinding_r: msg.proof_blinding_r.to_bytes().to_vec(),
proof_blinding_s: msg.proof_blinding_s.to_bytes().to_vec(),
}
}
}
impl TryFrom<pb::OutputPlan> for OutputPlan {
type Error = anyhow::Error;
fn try_from(msg: pb::OutputPlan) -> Result<Self, Self::Error> {
Ok(Self {
value: msg
.value
.ok_or_else(|| anyhow::anyhow!("missing value"))?
.try_into()?,
dest_address: msg
.dest_address
.ok_or_else(|| anyhow::anyhow!("missing address"))?
.try_into()?,
rseed: Rseed(msg.rseed.as_slice().try_into()?),
value_blinding: Fr::from_bytes_checked(msg.value_blinding.as_slice().try_into()?)
.expect("value_blinding malformed"),
proof_blinding_r: Fq::from_bytes_checked(msg.proof_blinding_r.as_slice().try_into()?)
.expect("proof_blinding_r malformed"),
proof_blinding_s: Fq::from_bytes_checked(msg.proof_blinding_s.as_slice().try_into()?)
.expect("proof_blinding_s malformed"),
})
}
}
#[cfg(test)]
mod test {
use crate::output::OutputProofPublic;
use super::OutputPlan;
use penumbra_sdk_asset::Value;
use penumbra_sdk_keys::{
keys::{Bip44Path, SeedPhrase, SpendKey},
PayloadKey,
};
use penumbra_sdk_proof_params::OUTPUT_PROOF_VERIFICATION_KEY;
use rand_core::OsRng;
#[test]
fn check_output_proof_verification() {
let mut rng = OsRng;
let seed_phrase = SeedPhrase::generate(rng);
let sk = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let ovk = sk.full_viewing_key().outgoing();
let dummy_memo_key: PayloadKey = [0; 32].into();
let value: Value = "1234.02penumbra".parse().unwrap();
let dest_address = "penumbra1rqcd3hfvkvc04c4c9vc0ac87lh4y0z8l28k4xp6d0cnd5jc6f6k0neuzp6zdwtpwyfpswtdzv9jzqtpjn5t6wh96pfx3flq2dhqgc42u7c06kj57dl39w2xm6tg0wh4zc8kjjk".parse().unwrap();
let output_plan = OutputPlan::new(&mut rng, value, dest_address);
let blinding_factor = output_plan.value_blinding;
let _body = output_plan.output_body(ovk, &dummy_memo_key);
let balance_commitment = output_plan.balance().commit(blinding_factor);
let note_commitment = output_plan.output_note().commit();
let output_proof = output_plan.output_proof();
output_proof
.verify(
&OUTPUT_PROOF_VERIFICATION_KEY,
OutputProofPublic {
balance_commitment,
note_commitment,
},
)
.unwrap();
}
}