1use dcbor::prelude::*;
2
3use crate::{
4 Digest, Error, PrivateKeyBase, PublicKeys, Reference, ReferenceProvider,
5 Result, SigningPrivateKey, SigningPublicKey, tags,
6};
7
8#[derive(Clone, Copy, Eq, PartialEq, Hash)]
29pub struct XID([u8; Self::XID_SIZE]);
30
31impl XID {
32 pub const XID_SIZE: usize = 32;
33
34 pub fn from_data(data: [u8; Self::XID_SIZE]) -> Self { Self(data) }
36
37 pub fn from_data_ref(data: impl AsRef<[u8]>) -> Result<Self> {
41 let data = data.as_ref();
42 if data.len() != Self::XID_SIZE {
43 return Err(Error::invalid_size("XID", Self::XID_SIZE, data.len()));
44 }
45 let mut arr = [0u8; Self::XID_SIZE];
46 arr.copy_from_slice(data.as_ref());
47 Ok(Self::from_data(arr))
48 }
49
50 pub fn data(&self) -> &[u8; Self::XID_SIZE] { self.into() }
52
53 pub fn as_bytes(&self) -> &[u8] { self.as_ref() }
55
56 pub fn new(genesis_key: impl AsRef<SigningPublicKey>) -> Self {
60 let key_cbor_data = genesis_key.as_ref().to_cbor_data();
61 let digest = Digest::from_image(key_cbor_data);
62 Self::from_data(*digest.data())
63 }
64
65 pub fn validate(&self, key: &SigningPublicKey) -> bool {
67 let key_data = key.to_cbor_data();
68 let digest = Digest::from_image(key_data);
69 *digest.data() == self.0
70 }
71
72 pub fn from_hex(hex: impl AsRef<str>) -> Self {
77 Self::from_data_ref(hex::decode(hex.as_ref()).unwrap()).unwrap()
78 }
79
80 pub fn to_hex(&self) -> String { hex::encode(self.0) }
82
83 pub fn short_description(&self) -> String { self.ref_hex_short() }
85
86 pub fn bytewords_identifier(&self, prefix: bool) -> String {
88 self.ref_bytewords(if prefix { Some("๐
ง") } else { None })
89 }
90
91 pub fn bytemoji_identifier(&self, prefix: bool) -> String {
93 self.ref_bytemoji(if prefix { Some("๐
ง") } else { None })
94 }
95}
96
97pub trait XIDProvider {
99 fn xid(&self) -> XID;
101}
102
103impl XIDProvider for XID {
105 fn xid(&self) -> XID { *self }
106}
107
108impl XIDProvider for SigningPublicKey {
110 fn xid(&self) -> XID { XID::new(self) }
111}
112
113impl ReferenceProvider for XID {
115 fn reference(&self) -> Reference { Reference::from_data(*self.data()) }
116}
117
118impl<'a> From<&'a XID> for &'a [u8; XID::XID_SIZE] {
120 fn from(value: &'a XID) -> Self { &value.0 }
121}
122
123impl<'a> From<&'a XID> for &'a [u8] {
125 fn from(value: &'a XID) -> Self { &value.0 }
126}
127
128impl AsRef<[u8]> for XID {
130 fn as_ref(&self) -> &[u8] { &self.0 }
131}
132
133impl std::cmp::PartialOrd for XID {
135 fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
136 Some(self.0.cmp(&other.0))
137 }
138}
139
140impl std::cmp::Ord for XID {
142 fn cmp(&self, other: &Self) -> std::cmp::Ordering { self.0.cmp(&other.0) }
143}
144
145impl std::fmt::Debug for XID {
147 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
148 write!(f, "XID({})", hex::encode(self.0))
149 }
150}
151
152impl std::fmt::Display for XID {
155 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
156 write!(f, "XID({})", self.short_description())
157 }
158}
159
160impl CBORTagged for XID {
162 fn cbor_tags() -> Vec<Tag> { tags_for_values(&[tags::TAG_XID]) }
163}
164
165impl From<XID> for CBOR {
167 fn from(value: XID) -> Self { value.tagged_cbor() }
168}
169
170impl From<&SigningPublicKey> for XID {
172 fn from(key: &SigningPublicKey) -> Self { Self::new(key) }
173}
174
175impl TryFrom<&SigningPrivateKey> for XID {
178 type Error = Error;
179
180 fn try_from(
181 key: &SigningPrivateKey,
182 ) -> std::result::Result<Self, Self::Error> {
183 Ok(Self::new(&key.public_key()?))
184 }
185}
186
187impl From<&PublicKeys> for XID {
190 fn from(key: &PublicKeys) -> Self { Self::new(key.signing_public_key()) }
191}
192
193impl From<&PrivateKeyBase> for XID {
196 fn from(key: &PrivateKeyBase) -> Self {
197 Self::new(key.schnorr_signing_private_key().public_key().unwrap())
198 }
199}
200
201impl CBORTaggedEncodable for XID {
204 fn untagged_cbor(&self) -> CBOR { CBOR::to_byte_string(self.0) }
205}
206
207impl TryFrom<CBOR> for XID {
209 type Error = dcbor::Error;
210
211 fn try_from(cbor: CBOR) -> dcbor::Result<Self> {
212 Self::from_tagged_cbor(cbor)
213 }
214}
215
216impl CBORTaggedDecodable for XID {
219 fn from_untagged_cbor(cbor: CBOR) -> dcbor::Result<Self> {
220 let data = CBOR::try_into_byte_string(cbor)?;
221 Ok(Self::from_data_ref(data)?)
222 }
223}
224
225impl From<XID> for Vec<u8> {
228 fn from(xid: XID) -> Self { xid.0.to_vec() }
229}
230
231#[cfg(test)]
232mod tests {
233 use bc_ur::prelude::*;
234 use hex_literal::hex;
235 use indoc::indoc;
236
237 use super::*;
238 use crate::{ECPrivateKey, SigningPrivateKey};
239
240 #[test]
241 fn test_xid() {
242 crate::register_tags();
243 let xid = XID::from_data_ref(hex!(
244 "de2853684ae55803a08b36dd7f4e566649970601927330299fd333f33fecc037"
245 ))
246 .unwrap();
247 assert_eq!(
248 xid.to_hex(),
249 "de2853684ae55803a08b36dd7f4e566649970601927330299fd333f33fecc037"
250 );
251 assert_eq!(xid.short_description(), "de285368");
252 assert_eq!(
253 xid.data(),
254 &hex!(
255 "de2853684ae55803a08b36dd7f4e566649970601927330299fd333f33fecc037"
256 )
257 );
258 assert_eq!(
259 format!("{:?}", xid),
260 "XID(de2853684ae55803a08b36dd7f4e566649970601927330299fd333f33fecc037)"
261 );
262 assert_eq!(format!("{}", xid), "XID(de285368)");
263
264 let xid_string = xid.ur_string();
265 assert_eq!(
266 xid_string,
267 "ur:xid/hdcxuedeguisgevwhdaxnbluenutlbglhfiygamsamadmojkdydtneteeowffhwprtemcaatledk"
268 );
269 assert_eq!(XID::from_ur_string(xid_string).unwrap(), xid);
270 assert_eq!(xid.bytewords_identifier(true), "๐
ง URGE DICE GURU IRIS");
271 assert_eq!(xid.bytemoji_identifier(true), "๐
ง ๐ป ๐ป ๐ ๐");
272 }
273
274 #[test]
275 fn test_xid_from_key() {
276 crate::register_tags();
277 let private_key = SigningPrivateKey::new_schnorr(
278 ECPrivateKey::from_data(hex!(
279 "322b5c1dd5a17c3481c2297990c85c232ed3c17b52ce9905c6ec5193ad132c36"
280 )),
281 );
282 let public_key = private_key.public_key().unwrap();
283
284 let key_cbor = public_key.to_cbor();
285 #[rustfmt::skip]
286 assert_eq!(key_cbor.diagnostic(), indoc! {"
287 40022(
288 h'e8251dc3a17e0f2c07865ed191139ecbcddcbdd070ec1ff65df5148c7ef4005a'
289 )
290 "}.trim());
291 #[rustfmt::skip]
292 assert_eq!(key_cbor.hex_annotated(), indoc! {"
293 d9 9c56 # tag(40022) signing-public-key
294 5820 # bytes(32)
295 e8251dc3a17e0f2c07865ed191139ecbcddcbdd070ec1ff65df5148c7ef4005a
296 "}.trim());
297 let key_cbor_data = key_cbor.to_cbor_data();
298 assert_eq!(
299 key_cbor_data,
300 hex!(
301 "d99c565820e8251dc3a17e0f2c07865ed191139ecbcddcbdd070ec1ff65df5148c7ef4005a"
302 )
303 );
304 let digest = Digest::from_image(&key_cbor_data);
305 assert_eq!(
306 digest.data(),
307 &hex!(
308 "d40e0602674df1b732f5e025d04c45f2e74ed1652c5ae1740f6a5502dbbdcd47"
309 )
310 );
311
312 let xid = XID::new(&public_key);
313 assert_eq!(
314 format!("{:?}", xid),
315 "XID(d40e0602674df1b732f5e025d04c45f2e74ed1652c5ae1740f6a5502dbbdcd47)"
316 );
317 xid.validate(&public_key);
318
319 assert_eq!(format!("{}", xid), "XID(d40e0602)");
320 let reference = xid.reference();
321 assert_eq!(format!("{reference}"), "Reference(d40e0602)");
322
323 assert_eq!(reference.bytewords_identifier(None), "TINY BETA ATOM ALSO");
324 assert_eq!(reference.bytemoji_identifier(None), "๐งฆ ๐คจ ๐ ๐");
325
326 assert_eq!(digest.data(), xid.data());
327 }
328}