penumbra_sdk_stake/
delegation_token.rsuse std::str::FromStr;
use regex::Regex;
use penumbra_sdk_asset::asset;
use super::IdentityKey;
pub struct DelegationToken {
validator_identity: IdentityKey,
base_denom: asset::Metadata,
}
impl From<IdentityKey> for DelegationToken {
fn from(v: IdentityKey) -> Self {
DelegationToken::new(v)
}
}
impl From<&IdentityKey> for DelegationToken {
fn from(v: &IdentityKey) -> Self {
DelegationToken::new(*v)
}
}
impl DelegationToken {
pub fn new(validator_identity: IdentityKey) -> Self {
let base_denom = asset::REGISTRY
.parse_denom(&format!("udelegation_{validator_identity}"))
.expect("base denom format is valid");
DelegationToken {
validator_identity,
base_denom,
}
}
pub fn denom(&self) -> asset::Metadata {
self.base_denom.clone()
}
pub fn default_unit(&self) -> asset::Unit {
self.base_denom.default_unit()
}
pub fn id(&self) -> asset::Id {
self.base_denom.id()
}
pub fn validator(&self) -> IdentityKey {
self.validator_identity
}
}
impl TryFrom<asset::Metadata> for DelegationToken {
type Error = anyhow::Error;
fn try_from(base_denom: asset::Metadata) -> Result<Self, Self::Error> {
let validator_identity =
Regex::new("^udelegation_(?P<data>penumbravalid1[a-zA-HJ-NP-Z0-9]+)$")
.expect("regex is valid")
.captures(&base_denom.to_string())
.ok_or_else(|| {
anyhow::anyhow!(
"base denom {} is not a delegation token",
base_denom.to_string()
)
})?
.name("data")
.expect("data is a named capture")
.as_str()
.parse()?;
Ok(Self {
base_denom,
validator_identity,
})
}
}
impl FromStr for DelegationToken {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
asset::REGISTRY
.parse_denom(s)
.ok_or_else(|| anyhow::anyhow!("could not parse {} as base denomination", s))?
.try_into()
}
}
impl std::fmt::Display for DelegationToken {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.base_denom.fmt(f)
}
}
impl std::fmt::Debug for DelegationToken {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.base_denom.fmt(f)
}
}
impl PartialEq for DelegationToken {
fn eq(&self, other: &Self) -> bool {
self.base_denom.eq(&other.base_denom)
}
}
impl Eq for DelegationToken {}
impl std::hash::Hash for DelegationToken {
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.base_denom.hash(state)
}
}
#[cfg(test)]
mod tests {
use decaf377_rdsa::{SigningKey, SpendAuth, VerificationKey};
use super::*;
#[test]
fn delegation_token_denomination_round_trip() {
use rand_core::OsRng;
let vk = VerificationKey::from(SigningKey::<SpendAuth>::new(OsRng));
let ik = IdentityKey(vk.into());
let token = DelegationToken::new(ik);
let denom = token.to_string();
let token2 = DelegationToken::from_str(&denom).unwrap();
let denom2 = token2.to_string();
assert_eq!(denom, denom2);
assert_eq!(token, token2);
}
}