penumbra_sdk_mock_client/
lib.rsuse anyhow::Error;
use cnidarium::StateRead;
use penumbra_sdk_compact_block::{component::StateReadExt as _, CompactBlock, StatePayload};
use penumbra_sdk_dex::{swap::SwapPlaintext, swap_claim::SwapClaimPlan};
use penumbra_sdk_keys::{keys::SpendKey, FullViewingKey};
use penumbra_sdk_sct::{
component::{clock::EpochRead, tree::SctRead},
Nullifier,
};
use penumbra_sdk_shielded_pool::{note, Note, SpendPlan};
use penumbra_sdk_tct as tct;
use penumbra_sdk_transaction::{AuthorizationData, Transaction, TransactionPlan, WitnessData};
use rand_core::OsRng;
use std::collections::BTreeMap;
pub struct MockClient {
latest_height: u64,
sk: SpendKey,
pub fvk: FullViewingKey,
pub notes: BTreeMap<note::StateCommitment, Note>,
pub nullifiers: BTreeMap<note::StateCommitment, Nullifier>,
pub spent_notes: BTreeMap<note::StateCommitment, ()>,
swaps: BTreeMap<tct::StateCommitment, SwapPlaintext>,
pub sct: penumbra_sdk_tct::Tree,
}
impl MockClient {
pub fn new(sk: SpendKey) -> MockClient {
Self {
latest_height: u64::MAX,
fvk: sk.full_viewing_key().clone(),
sk,
notes: Default::default(),
spent_notes: Default::default(),
nullifiers: Default::default(),
sct: Default::default(),
swaps: Default::default(),
}
}
pub async fn with_sync_to_storage(
mut self,
storage: impl AsRef<cnidarium::Storage>,
) -> anyhow::Result<Self> {
let latest = storage.as_ref().latest_snapshot();
self.sync_to_latest(latest).await?;
Ok(self)
}
pub async fn with_sync_to_inner_storage(
mut self,
storage: cnidarium::Storage,
) -> anyhow::Result<Self> {
let latest = storage.latest_snapshot();
self.sync_to_latest(latest).await?;
Ok(self)
}
pub async fn sync_to_latest<R: StateRead>(&mut self, state: R) -> anyhow::Result<()> {
let height = state.get_block_height().await?;
self.sync_to(height, state).await?;
Ok(())
}
pub async fn sync_to<R: StateRead>(
&mut self,
target_height: u64,
state: R,
) -> anyhow::Result<()> {
let start_height = self.latest_height.wrapping_add(1);
for height in start_height..=target_height {
let compact_block = state
.compact_block(height)
.await?
.ok_or_else(|| anyhow::anyhow!("missing compact block for height {}", height))?;
self.scan_block(compact_block.try_into()?)?;
let (latest_height, root) = self.latest_height_and_sct_root();
anyhow::ensure!(latest_height == height, "latest height should be updated");
let expected_root = state
.get_anchor_by_height(height)
.await?
.ok_or_else(|| anyhow::anyhow!("missing sct anchor for height {}", height))?;
anyhow::ensure!(
root == expected_root,
format!(
"client sct root should match chain state: {:?} != {:?}",
root, expected_root
)
);
}
Ok(())
}
pub fn scan_block(&mut self, block: CompactBlock) -> anyhow::Result<()> {
use penumbra_sdk_tct::Witness::*;
if self.latest_height.wrapping_add(1) != block.height {
anyhow::bail!(
"wrong block height {} for latest height {}",
block.height,
self.latest_height
);
}
for payload in block.state_payloads {
match payload {
StatePayload::Note { note: payload, .. } => {
match payload.trial_decrypt(&self.fvk) {
Some(note) => {
self.sct.insert(Keep, payload.note_commitment)?;
let nullifier = self
.nullifier(payload.note_commitment)
.expect("newly inserted note should be present in sct");
self.notes.insert(payload.note_commitment, note.clone());
self.nullifiers.insert(payload.note_commitment, nullifier);
}
None => {
self.sct.insert(Forget, payload.note_commitment)?;
}
}
}
StatePayload::Swap { swap: payload, .. } => {
match payload.trial_decrypt(&self.fvk) {
Some(swap) => {
self.sct.insert(Keep, payload.commitment)?;
self.swaps.insert(payload.commitment, swap.clone());
let batch_data =
block.swap_outputs.get(&swap.trading_pair).ok_or_else(|| {
anyhow::anyhow!("server gave invalid compact block")
})?;
let (output_1, output_2) = swap.output_notes(batch_data);
self.notes.insert(output_1.commit(), output_1.clone());
self.notes.insert(output_2.commit(), output_2.clone());
}
None => {
self.sct.insert(Forget, payload.commitment)?;
}
}
}
StatePayload::RolledUp { commitment, .. } => {
if self.notes.contains_key(&commitment) {
self.sct.insert(Keep, commitment)?;
} else {
self.sct.insert(Forget, commitment)?;
}
}
}
}
for nullifier in block.nullifiers {
if !self.nullifiers.values().any(move |n| *n == nullifier) {
continue;
}
self.spent_notes.insert(
*self
.nullifiers
.iter()
.find_map(|(k, v)| if *v == nullifier { Some(k) } else { None })
.unwrap(),
(),
);
}
self.sct.end_block()?;
if block.epoch_root.is_some() {
self.sct.end_epoch()?;
}
self.latest_height = block.height;
Ok(())
}
pub fn latest_height_and_sct_root(&self) -> (u64, penumbra_sdk_tct::Root) {
(self.latest_height, self.sct.root())
}
pub fn note_by_commitment(&self, commitment: ¬e::StateCommitment) -> Option<Note> {
self.notes.get(commitment).cloned()
}
pub fn swap_by_commitment(&self, commitment: ¬e::StateCommitment) -> Option<SwapPlaintext> {
self.swaps.get(commitment).cloned()
}
pub fn position(
&self,
commitment: note::StateCommitment,
) -> Option<penumbra_sdk_tct::Position> {
self.sct.witness(commitment).map(|proof| proof.position())
}
pub fn nullifier(&self, commitment: note::StateCommitment) -> Option<Nullifier> {
let position = self.position(commitment);
if position.is_none() {
return None;
}
let nk = self.fvk.nullifier_key();
Some(Nullifier::derive(&nk, position.unwrap(), &commitment))
}
pub fn witness_commitment(
&self,
commitment: note::StateCommitment,
) -> Option<penumbra_sdk_tct::Proof> {
self.sct.witness(commitment)
}
pub fn witness_plan(&self, plan: &TransactionPlan) -> Result<WitnessData, Error> {
let spend_commitment = |spend: &SpendPlan| spend.note.commit();
let spends = plan.spend_plans().map(spend_commitment);
let swap_claim_commitment = |swap: &SwapClaimPlan| swap.swap_plaintext.swap_commitment();
let swap_claims = plan.swap_claim_plans().map(swap_claim_commitment);
let witness = |commitment| {
self.sct
.witness(commitment)
.ok_or_else(|| anyhow::anyhow!("note commitment {commitment:?} unknown to client"))
.map(|proof| (commitment, proof))
};
Ok(WitnessData {
anchor: self.sct.root(),
state_commitment_proofs: spends
.chain(swap_claims)
.map(witness)
.collect::<Result<_, Error>>()?,
})
}
pub fn authorize_plan(&self, plan: &TransactionPlan) -> Result<AuthorizationData, Error> {
plan.authorize(OsRng, &self.sk)
}
pub async fn witness_auth_build(&self, plan: &TransactionPlan) -> Result<Transaction, Error> {
let witness_data = self.witness_plan(plan)?;
let auth_data = self.authorize_plan(plan)?;
plan.clone()
.build_concurrent(&self.fvk, &witness_data, &auth_data)
.await
}
pub fn notes_by_asset(
&self,
asset_id: penumbra_sdk_asset::asset::Id,
) -> impl Iterator<Item = &Note> + '_ {
self.notes
.values()
.filter(move |n| n.asset_id() == asset_id)
}
pub fn spent_note(&self, commitment: ¬e::StateCommitment) -> bool {
self.spent_notes.contains_key(commitment)
}
pub fn spendable_notes_by_asset(
&self,
asset_id: penumbra_sdk_asset::asset::Id,
) -> impl Iterator<Item = &Note> + '_ {
self.notes
.values()
.filter(move |n| n.asset_id() == asset_id && !self.spent_note(&n.commit()))
}
}