penumbra_sdk_transaction/view/
transaction_perspective.rsuse anyhow::anyhow;
use pbjson_types::Any;
use penumbra_sdk_asset::{asset, EstimatedPrice, Value, ValueView};
use penumbra_sdk_dex::BatchSwapOutputData;
use penumbra_sdk_keys::{Address, AddressView, PayloadKey};
use penumbra_sdk_proto::core::transaction::v1::{
self as pb, NullifierWithNote, PayloadKeyWithCommitment,
};
use penumbra_sdk_sct::Nullifier;
use penumbra_sdk_shielded_pool::{note, Note, NoteView};
use penumbra_sdk_txhash::TransactionId;
use std::collections::BTreeMap;
#[derive(Debug, Clone, Default)]
pub struct TransactionPerspective {
pub payload_keys: BTreeMap<note::StateCommitment, PayloadKey>,
pub spend_nullifiers: BTreeMap<Nullifier, Note>,
pub advice_notes: BTreeMap<note::StateCommitment, Note>,
pub address_views: Vec<AddressView>,
pub denoms: asset::Cache,
pub transaction_id: TransactionId,
pub prices: Vec<EstimatedPrice>,
pub extended_metadata: BTreeMap<asset::Id, Any>,
pub creation_transaction_ids_by_nullifier: BTreeMap<Nullifier, TransactionId>,
pub nullification_transaction_ids_by_commitment: BTreeMap<note::StateCommitment, TransactionId>,
pub batch_swap_output_data: Vec<BatchSwapOutputData>,
}
impl TransactionPerspective {
pub fn view_value(&self, value: Value) -> ValueView {
value
.view_with_cache(&self.denoms)
.with_prices(&self.prices, &self.denoms)
.with_extended_metadata(self.extended_metadata.get(&value.asset_id).cloned())
}
pub fn view_note(&self, note: Note) -> NoteView {
NoteView {
address: self.view_address(note.address()),
value: self.view_value(note.value()),
rseed: note.rseed(),
}
}
pub fn view_address(&self, address: Address) -> AddressView {
match self.address_views.iter().find(|av| av.address() == address) {
Some(av) => av.clone(),
None => AddressView::Opaque { address },
}
}
pub fn get_and_view_advice_note(&self, commitment: ¬e::StateCommitment) -> Option<NoteView> {
self.advice_notes
.get(commitment)
.cloned()
.map(|note| self.view_note(note))
}
}
impl TransactionPerspective {}
impl From<TransactionPerspective> for pb::TransactionPerspective {
fn from(msg: TransactionPerspective) -> Self {
let mut payload_keys = Vec::new();
let mut spend_nullifiers = Vec::new();
let mut advice_notes = Vec::new();
let mut address_views = Vec::new();
let mut denoms = Vec::new();
for (commitment, payload_key) in msg.payload_keys {
payload_keys.push(PayloadKeyWithCommitment {
payload_key: Some(payload_key.to_owned().into()),
commitment: Some(commitment.to_owned().into()),
});
}
for (nullifier, note) in msg.spend_nullifiers {
spend_nullifiers.push(NullifierWithNote {
nullifier: Some(nullifier.into()),
note: Some(note.into()),
})
}
for note in msg.advice_notes.into_values() {
advice_notes.push(note.into());
}
for address_view in msg.address_views {
address_views.push(address_view.into());
}
for denom in msg.denoms.values() {
denoms.push(denom.clone().into());
}
Self {
payload_keys,
spend_nullifiers,
advice_notes,
address_views,
denoms,
transaction_id: Some(msg.transaction_id.into()),
prices: msg.prices.into_iter().map(Into::into).collect(),
extended_metadata: msg
.extended_metadata
.into_iter()
.map(|(k, v)| pb::transaction_perspective::ExtendedMetadataById {
asset_id: Some(k.into()),
extended_metadata: Some(v),
})
.collect(),
creation_transaction_ids_by_nullifier: msg
.creation_transaction_ids_by_nullifier
.into_iter()
.map(
|(k, v)| pb::transaction_perspective::CreationTransactionIdByNullifier {
nullifier: Some(k.into()),
transaction_id: Some(v.into()),
},
)
.collect(),
nullification_transaction_ids_by_commitment: msg
.nullification_transaction_ids_by_commitment
.into_iter()
.map(
|(k, v)| pb::transaction_perspective::NullificationTransactionIdByCommitment {
commitment: Some(k.into()),
transaction_id: Some(v.into()),
},
)
.collect(),
batch_swap_output_data: msg
.batch_swap_output_data
.into_iter()
.map(Into::into)
.collect(),
}
}
}
impl TryFrom<pb::TransactionPerspective> for TransactionPerspective {
type Error = anyhow::Error;
fn try_from(msg: pb::TransactionPerspective) -> Result<Self, Self::Error> {
let mut payload_keys = BTreeMap::new();
let mut spend_nullifiers = BTreeMap::new();
let mut advice_notes = BTreeMap::new();
let mut address_views = Vec::new();
let mut denoms = BTreeMap::new();
for pk in msg.payload_keys {
if pk.commitment.is_some() {
payload_keys.insert(
pk.commitment
.ok_or_else(|| anyhow!("missing commitment in payload key"))?
.try_into()?,
pk.payload_key
.ok_or_else(|| anyhow!("missing payload key"))?
.try_into()?,
);
};
}
for nwn in msg.spend_nullifiers {
spend_nullifiers.insert(
nwn.nullifier
.ok_or_else(|| anyhow!("missing nullifier in spend nullifier"))?
.try_into()?,
nwn.note
.ok_or_else(|| anyhow!("missing note in spend nullifier"))?
.try_into()?,
);
}
for note in msg.advice_notes {
let note: Note = note.try_into()?;
advice_notes.insert(note.commit(), note);
}
for address_view in msg.address_views {
address_views.push(address_view.try_into()?);
}
for denom in msg.denoms {
denoms.insert(
denom
.penumbra_asset_id
.clone()
.ok_or_else(|| anyhow!("missing penumbra asset ID in denom"))?
.try_into()?,
denom.try_into()?,
);
}
let transaction_id: TransactionId = match msg.transaction_id {
Some(tx_id) => tx_id.try_into()?,
None => TransactionId::default(),
};
Ok(Self {
payload_keys,
spend_nullifiers,
advice_notes,
address_views,
denoms: denoms.try_into()?,
transaction_id,
prices: msg
.prices
.into_iter()
.map(TryInto::try_into)
.collect::<Result<_, _>>()?,
extended_metadata: msg
.extended_metadata
.into_iter()
.map(|em| {
Ok((
em.asset_id
.ok_or_else(|| anyhow!("missing asset ID in extended metadata"))?
.try_into()?,
em.extended_metadata
.ok_or_else(|| anyhow!("missing extended metadata"))?,
))
})
.collect::<Result<_, anyhow::Error>>()?,
creation_transaction_ids_by_nullifier: msg
.creation_transaction_ids_by_nullifier
.into_iter()
.map(|ct| {
Ok((
ct.nullifier
.ok_or_else(|| anyhow!("missing nullifier in creation transaction ID"))?
.try_into()?,
ct.transaction_id
.ok_or_else(|| {
anyhow!("missing transaction ID in creation transaction ID")
})?
.try_into()?,
))
})
.collect::<Result<_, anyhow::Error>>()?,
nullification_transaction_ids_by_commitment: msg
.nullification_transaction_ids_by_commitment
.into_iter()
.map(|nt| {
Ok((
nt.commitment
.ok_or_else(|| {
anyhow!("missing commitment in nullification transaction ID")
})?
.try_into()?,
nt.transaction_id
.ok_or_else(|| {
anyhow!("missing transaction ID in nullification transaction ID")
})?
.try_into()?,
))
})
.collect::<Result<_, anyhow::Error>>()?,
batch_swap_output_data: msg
.batch_swap_output_data
.into_iter()
.map(TryInto::try_into)
.collect::<Result<_, _>>()?,
})
}
}