Skip to content

Commit e251dea

Browse files
feat(general): Small assorted improvements (#183)
* Add the purpose getter function to Cip509 * Export Cip509RbacMetadata type * Add the define_hashes macro
1 parent 2aab21d commit e251dea

File tree

5 files changed

+145
-9
lines changed

5 files changed

+145
-9
lines changed
+127
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,127 @@
1+
//! A macro for defining a new type wrappers for the given hash types.
2+
3+
/// Defines a new type wrapper for the given hash types.
4+
///
5+
/// # Examples
6+
///
7+
/// ```
8+
/// use catalyst_types::{define_hashes, hashes::Blake2b128Hash};
9+
///
10+
/// define_hashes!(
11+
/// /// You can document the declared types...
12+
/// (SomeHash, Blake2b128Hash),
13+
/// // ...or not.
14+
/// (AnotherHash, Blake2b128Hash),
15+
/// );
16+
///
17+
/// let hash = SomeHash::new(&[1, 2, 3]);
18+
/// println!("{hash:?}");
19+
/// ```
20+
#[macro_export]
21+
macro_rules! define_hashes {
22+
($($(#[$docs:meta])* ($name:ident, $inner:ty)),+ $(,)?) => {
23+
$(
24+
$(#[$docs])*
25+
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
26+
pub struct $name($inner);
27+
28+
impl $name {
29+
/// Creates a new instance from the given bytes by hashing them.
30+
#[must_use]
31+
pub fn new(input_bytes: &[u8]) -> Self {
32+
Self(<$inner>::new(input_bytes))
33+
}
34+
}
35+
36+
impl From<$name> for Vec<u8> {
37+
fn from(value: $name) -> Self {
38+
value.0.into()
39+
}
40+
}
41+
42+
impl From<$inner> for $name {
43+
fn from(value: $inner) -> Self {
44+
Self(value)
45+
}
46+
}
47+
48+
impl TryFrom<&[u8]> for $name {
49+
type Error = $crate::hashes::Blake2bHashError;
50+
51+
fn try_from(value: &[u8]) -> Result<Self, Self::Error> {
52+
Ok(Self(<$inner>::try_from(value)?))
53+
}
54+
}
55+
56+
impl TryFrom<Vec<u8>> for $name {
57+
type Error = $crate::hashes::Blake2bHashError;
58+
59+
fn try_from(value: Vec<u8>) -> Result<Self, Self::Error> {
60+
value.as_slice().try_into()
61+
}
62+
}
63+
64+
impl std::str::FromStr for $name {
65+
type Err = $crate::hashes::Blake2bHashError;
66+
67+
fn from_str(s: &str) -> Result<Self, Self::Err> {
68+
let hash: $inner = s.parse().map_err($crate::hashes::Blake2bHashError::from)?;
69+
Ok(Self(hash))
70+
}
71+
}
72+
73+
impl<C> minicbor::Encode<C> for $name {
74+
fn encode<W: minicbor::encode::Write>(
75+
&self, e: &mut minicbor::Encoder<W>, ctx: &mut C,
76+
) -> Result<(), minicbor::encode::Error<W::Error>> {
77+
self.0.encode(e, ctx)
78+
}
79+
}
80+
81+
impl<'a, C> minicbor::Decode<'a, C> for $name {
82+
fn decode(
83+
d: &mut minicbor::Decoder<'a>, ctx: &mut C,
84+
) -> Result<Self, minicbor::decode::Error> {
85+
let hash = <$inner>::decode(d, ctx)?;
86+
Ok(Self(hash))
87+
}
88+
}
89+
)+
90+
};
91+
}
92+
93+
#[cfg(test)]
94+
mod tests {
95+
use crate::hashes::Blake2b128Hash;
96+
97+
// Define one type without a trailing comma.
98+
define_hashes!((H1, Blake2b128Hash));
99+
// Define one type with a trailing comma and a doc-comment.
100+
define_hashes!(
101+
/// Some documentation.
102+
(H2, Blake2b128Hash),
103+
);
104+
// Define multiple types at once.
105+
define_hashes!(
106+
/// Documentation.
107+
(H3, Blake2b128Hash),
108+
// No documentation.
109+
(H4, Blake2b128Hash),
110+
/// More documentation.
111+
(H5, Blake2b128Hash),
112+
);
113+
114+
// There is little reason to check the conversion itself, it is mostly a demonstration
115+
// that the methods defined by the macro are working.
116+
#[test]
117+
fn hash_wrapper() {
118+
let hash = H1::new(&[1, 2, 3, 4, 5]);
119+
120+
let v = Vec::from(hash);
121+
let from_slice = H1::try_from(v.as_slice()).unwrap();
122+
assert_eq!(hash, from_slice);
123+
124+
let from_vec = H1::try_from(v).unwrap();
125+
assert_eq!(hash, from_vec);
126+
}
127+
}

rust/catalyst-types/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
33
pub mod cbor_utils;
44
pub mod conversion;
5+
pub mod hash_wrapper;
56
pub mod hashes;
67
pub mod id_uri;
78
pub mod mmap_file;

rust/catalyst-types/src/uuid/uuid_v7.rs

+9-8
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,13 @@
22
use std::fmt::{Display, Formatter};
33

44
use minicbor::{Decode, Decoder, Encode};
5+
use uuid::Uuid;
56

67
use super::{decode_cbor_uuid, encode_cbor_uuid, CborContext, UuidError, INVALID_UUID};
78

89
/// Type representing a `UUIDv7`.
910
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd, serde::Serialize)]
10-
pub struct UuidV7(uuid::Uuid);
11+
pub struct UuidV7(Uuid);
1112

1213
impl UuidV7 {
1314
/// Version for `UUIDv7`.
@@ -17,7 +18,7 @@ impl UuidV7 {
1718
#[must_use]
1819
#[allow(clippy::new_without_default)]
1920
pub fn new() -> Self {
20-
Self(uuid::Uuid::now_v7())
21+
Self(Uuid::now_v7())
2122
}
2223

2324
/// Generates a zeroed out `UUIDv7` that can never be valid.
@@ -34,13 +35,13 @@ impl UuidV7 {
3435

3536
/// Returns the `uuid::Uuid` type.
3637
#[must_use]
37-
pub fn uuid(&self) -> uuid::Uuid {
38+
pub fn uuid(&self) -> Uuid {
3839
self.0
3940
}
4041
}
4142

4243
/// Check if this is a valid `UUIDv7`.
43-
fn is_valid(uuid: &uuid::Uuid) -> bool {
44+
fn is_valid(uuid: &Uuid) -> bool {
4445
uuid != &INVALID_UUID && uuid.get_version_num() == UuidV7::UUID_VERSION_NUMBER
4546
}
4647

@@ -72,10 +73,10 @@ impl Encode<CborContext> for UuidV7 {
7273
}
7374

7475
/// Returns a `UUIDv7` from `uuid::Uuid`.
75-
impl TryFrom<uuid::Uuid> for UuidV7 {
76+
impl TryFrom<Uuid> for UuidV7 {
7677
type Error = UuidError;
7778

78-
fn try_from(uuid: uuid::Uuid) -> Result<Self, Self::Error> {
79+
fn try_from(uuid: Uuid) -> Result<Self, Self::Error> {
7980
if is_valid(&uuid) {
8081
Ok(Self(uuid))
8182
} else {
@@ -87,7 +88,7 @@ impl TryFrom<uuid::Uuid> for UuidV7 {
8788
/// Returns a `uuid::Uuid` from `UUIDv7`.
8889
///
8990
/// NOTE: This does not guarantee that the `UUID` is valid.
90-
impl From<UuidV7> for uuid::Uuid {
91+
impl From<UuidV7> for Uuid {
9192
fn from(value: UuidV7) -> Self {
9293
value.0
9394
}
@@ -96,7 +97,7 @@ impl From<UuidV7> for uuid::Uuid {
9697
impl<'de> serde::Deserialize<'de> for UuidV7 {
9798
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
9899
where D: serde::Deserializer<'de> {
99-
let uuid = uuid::Uuid::deserialize(deserializer)?;
100+
let uuid = Uuid::deserialize(deserializer)?;
100101
if is_valid(&uuid) {
101102
Ok(Self(uuid))
102103
} else {

rust/rbac-registration/src/cardano/cip509/cip509.rs

+6
Original file line numberDiff line numberDiff line change
@@ -202,6 +202,12 @@ impl Cip509 {
202202
self.metadata.as_ref().and_then(|m| m.role_data.get(&role))
203203
}
204204

205+
/// Returns a purpose of this registration.
206+
#[must_use]
207+
pub fn purpose(&self) -> Option<UuidV4> {
208+
self.purpose
209+
}
210+
205211
/// Returns a hash of the previous transaction.
206212
#[must_use]
207213
pub fn previous_transaction(&self) -> Option<Blake2b256Hash> {

rust/rbac-registration/src/cardano/cip509/mod.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,8 @@
33
//! CDDL Reference: <https://github.com/input-output-hk/catalyst-CIPs/blob/x509-envelope-metadata/CIP-XXXX/x509-envelope.cddl>
44
55
pub use cip509::Cip509;
6-
pub use rbac::{C509Cert, SimplePublicKeyType, X509DerCert};
6+
#[allow(clippy::module_name_repetitions)]
7+
pub use rbac::{C509Cert, Cip509RbacMetadata, SimplePublicKeyType, X509DerCert};
78
pub use types::{
89
CertKeyHash, KeyLocalRef, LocalRefInt, Payment, PaymentHistory, PointTxnIdx, RoleData,
910
RoleNumber, TxInputHash, ValidationSignature,

0 commit comments

Comments
 (0)