penumbra_asset/asset/
id.rsuse crate::Value;
use ark_ff::ToConstraintField;
use ark_serialize::CanonicalDeserialize;
use base64::Engine;
use decaf377::Fq;
use once_cell::sync::Lazy;
use penumbra_num::Amount;
use penumbra_proto::{penumbra::core::asset::v1 as pb, serializers::bech32str, DomainType};
use serde::{Deserialize, Serialize};
#[derive(Serialize, Deserialize, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[serde(try_from = "pb::AssetId", into = "pb::AssetId")]
pub struct Id(pub Fq);
impl From<Id> for pb::AssetId {
fn from(id: Id) -> Self {
pb::AssetId {
inner: id.0.to_bytes().to_vec(),
alt_bech32m: String::new(),
alt_base_denom: String::new(),
}
}
}
impl TryFrom<pb::AssetId> for Id {
type Error = anyhow::Error;
fn try_from(value: pb::AssetId) -> Result<Self, Self::Error> {
if !value.inner.is_empty() {
if !value.alt_base_denom.is_empty() || !value.alt_bech32m.is_empty() {
anyhow::bail!(
"AssetId proto has both inner and alt_bech32m or alt_base_denom fields set"
);
}
value.inner.as_slice().try_into()
} else if !value.alt_bech32m.is_empty() {
value.alt_bech32m.parse()
} else if !value.alt_base_denom.is_empty() {
Ok(Self::from_raw_denom(&value.alt_base_denom))
} else {
Err(anyhow::anyhow!(
"AssetId proto has neither inner nor alt_bech32m nor alt_base_denom fields set"
))
}
}
}
impl DomainType for Id {
type Proto = pb::AssetId;
}
impl TryFrom<&[u8]> for Id {
type Error = anyhow::Error;
fn try_from(slice: &[u8]) -> Result<Id, Self::Error> {
Ok(Id(Fq::deserialize_compressed(slice)?))
}
}
impl TryFrom<[u8; 32]> for Id {
type Error = anyhow::Error;
fn try_from(bytes: [u8; 32]) -> Result<Id, Self::Error> {
Ok(Id(Fq::from_bytes_checked(&bytes).expect("convert to bytes")))
}
}
impl std::fmt::Debug for Id {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.write_str(&bech32str::encode(
&self.0.to_bytes(),
bech32str::asset_id::BECH32_PREFIX,
bech32str::Bech32m,
))
}
}
impl std::fmt::Display for Id {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
f.write_str(&bech32str::encode(
&self.0.to_bytes(),
bech32str::asset_id::BECH32_PREFIX,
bech32str::Bech32m,
))
}
}
impl std::str::FromStr for Id {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let inner = bech32str::decode(s, bech32str::asset_id::BECH32_PREFIX, bech32str::Bech32m)?;
pb::AssetId {
inner,
alt_bech32m: String::new(),
alt_base_denom: String::new(),
}
.try_into()
}
}
impl ToConstraintField<Fq> for Id {
fn to_field_elements(&self) -> Option<Vec<Fq>> {
let mut elements = Vec::new();
elements.extend_from_slice(&[self.0]);
Some(elements)
}
}
pub static VALUE_GENERATOR_DOMAIN_SEP: Lazy<Fq> = Lazy::new(|| {
Fq::from_le_bytes_mod_order(blake2b_simd::blake2b(b"penumbra.value.generator").as_bytes())
});
impl Id {
pub fn value_generator(&self) -> decaf377::Element {
decaf377::Element::encode_to_curve(&poseidon377::hash_1(
&VALUE_GENERATOR_DOMAIN_SEP,
self.0,
))
}
pub fn to_bytes(&self) -> [u8; 32] {
self.0.to_bytes()
}
pub fn value(&self, amount: Amount) -> Value {
Value {
amount,
asset_id: *self,
}
}
pub(super) fn from_raw_denom(base_denom: &str) -> Self {
Id(Fq::from_le_bytes_mod_order(
blake2b_simd::Params::default()
.personal(b"Penumbra_AssetID")
.hash(base_denom.as_bytes())
.as_bytes(),
))
}
pub fn to_base64(&self) -> String {
base64::engine::general_purpose::STANDARD.encode(self.to_bytes())
}
}
#[cfg(test)]
mod tests {
use super::*;
use hex;
use serde_json;
use std::str::FromStr;
#[test]
fn asset_id_encoding() {
let id = Id::from_raw_denom("upenumbra");
let bech32m_id = format!("{id}");
let id2 = Id::from_str(&bech32m_id).expect("can decode valid asset id");
use penumbra_proto::Message;
let proto = id.encode_to_vec();
let proto2 = pb::AssetId {
alt_bech32m: bech32m_id,
..Default::default()
}
.encode_to_vec();
let proto3 = pb::AssetId {
alt_base_denom: "upenumbra".to_owned(),
..Default::default()
}
.encode_to_vec();
let id3 = Id::decode(proto.as_ref()).expect("can decode valid asset id");
let id4 = Id::decode(proto2.as_ref()).expect("can decode valid asset id");
let id5 = Id::decode(proto3.as_ref()).expect("can decode valid asset id");
assert_eq!(id2, id);
assert_eq!(id3, id);
assert_eq!(id4, id);
assert_eq!(id5, id);
}
#[test]
fn hex_to_bech32() {
let hex_strings = [
"cc0d3c9eef0c7ff4e225eca85a3094603691d289aeaf428ab0d87319ad93a302", "a7a339f42e671b2db1de226d4483d3e63036661cad1554d75f5f76fe04ec1e00", "29ea9c2f3371f6a487e7e95c247041f4a356f983eb064e5d2b3bcf322ca96a10", "76b3e4b10681358c123b381f90638476b7789040e47802de879f0fb3eedc8d0b", "2923a0a87b3a2421f165cc853dbf73a9bdafb5da0d948564b6059cb0217c4407", "07ef660132a4c3235fab272d43d9b9752a8337b2d108597abffaff5f246d0f0f", "5314b33eecfd5ca2e99c0b6d1e0ccafe3d2dd581c952d814fb64fdf51f85c411", "516108d0d0bba3f76e1f982d0a7cde118833307b03c0cd4ccb94e882b53c1f0f", "414e723f74bd987c02ccbc997585ed52b196e2ffe75b3793aa68cc2996626910", "bf8b035dda339b6cda8f221e79773b0fd871f27a472920f84c4aa2b4f98a700d", ];
for hex in hex_strings {
let bytes = hex::decode(hex).expect("valid hex string");
let bytes_array: [u8; 32] = bytes.try_into().expect("hex is 32 bytes");
let id = Id::try_from(bytes_array).expect("valid asset ID bytes");
let bech32_str = id.to_string();
println!("Asset ID for {}:", hex);
println!(" Bech32: {}", bech32_str);
let proto: pb::AssetId = id.into();
println!(" Proto JSON: {}\n", serde_json::to_string(&proto).unwrap());
let id_decoded = Id::from_str(&bech32_str).expect("valid bech32 string");
assert_eq!(id, id_decoded);
}
}
}