use crate::error::{Error, Result};
use bls::{PublicKey, PK_SIZE};
use serde::{Deserialize, Serialize};
use std::{
fmt::{Debug, Display},
hash::Hash,
};
use xor_name::{XorName, XOR_NAME_LEN};
#[derive(Clone, Copy, Eq, PartialEq, Ord, PartialOrd, Hash, Serialize, Deserialize)]
pub struct RegisterAddress {
pub(crate) meta: XorName,
pub(crate) owner: PublicKey,
}
impl Display for RegisterAddress {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}({:?})", &self.to_hex()[0..6], self.xorname())
}
}
impl Debug for RegisterAddress {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"RegisterAddress({}) {{ meta: {:?}, owner: {:?} }}",
&self.to_hex()[0..6],
self.meta,
self.owner
)
}
}
impl RegisterAddress {
pub fn new(meta: XorName, owner: PublicKey) -> Self {
Self { meta, owner }
}
pub fn xorname(&self) -> XorName {
let mut bytes = vec![];
bytes.extend_from_slice(&self.meta.0);
bytes.extend_from_slice(&self.owner.to_bytes());
XorName::from_content(&bytes)
}
pub fn to_hex(&self) -> String {
let mut bytes = vec![];
bytes.extend_from_slice(&self.meta.0);
bytes.extend_from_slice(&self.owner.to_bytes());
hex::encode(bytes)
}
pub fn from_hex(hex: &str) -> Result<Self> {
let bytes = hex::decode(hex).map_err(|_| Error::HexDeserializeFailed)?;
let meta_bytes: [u8; XOR_NAME_LEN] = bytes[..XOR_NAME_LEN]
.try_into()
.map_err(|_| Error::HexDeserializeFailed)?;
let meta = XorName(meta_bytes);
let owner_bytes: [u8; PK_SIZE] = bytes[XOR_NAME_LEN..]
.try_into()
.map_err(|_| Error::HexDeserializeFailed)?;
let owner = PublicKey::from_bytes(owner_bytes).map_err(|_| Error::HexDeserializeFailed)?;
Ok(Self { meta, owner })
}
pub fn meta(&self) -> XorName {
self.meta
}
pub fn owner(&self) -> PublicKey {
self.owner
}
}
#[cfg(test)]
mod tests {
use bls::SecretKey;
use super::*;
#[test]
fn test_register_hex_conversion() {
let mut rng = rand::thread_rng();
let owner = SecretKey::random().public_key();
let meta = XorName::random(&mut rng);
let addr = RegisterAddress::new(meta, owner);
let hex = &addr.to_hex();
let addr2 = RegisterAddress::from_hex(hex).unwrap();
assert_eq!(addr, addr2);
let bad_hex = format!("{hex}0");
let err = RegisterAddress::from_hex(&bad_hex);
assert_eq!(err, Err(Error::HexDeserializeFailed));
}
}