1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
use crate::{UREncodable, URDecodable};

/// A type that can be encoded to and decoded from a UR.
pub trait URCodable: UREncodable + URDecodable {}

#[cfg(test)]
mod tests {
    use dcbor::{CBOR, Tag, CBOREncodable, CBORTaggedEncodable, CBORDecodable, CBORTaggedDecodable, CBORTagged};

    use super::*;

    struct Test {
        s: String,
    }

    impl Test {
        fn new(s: &str) -> Self {
            Self { s: s.to_string() }
        }
    }

    impl CBORTagged for Test {
        const CBOR_TAG: Tag = Tag::new_with_static_name(24, "leaf");
    }

    impl CBOREncodable for Test {
        // This ensures that asking for the CBOR for this type will always
        // return a tagged CBOR value.
        fn cbor(&self) -> CBOR {
            self.tagged_cbor()
        }
    }

    impl From<Test> for CBOR {
        fn from(value: Test) -> Self {
            value.cbor()
        }
    }

    impl CBORTaggedEncodable for Test {
        // This is the core of the CBOR encoding for this type. It is the
        // untagged CBOR encoding.
        fn untagged_cbor(&self) -> CBOR {
            self.s.cbor()
        }
    }

    impl UREncodable for Test {}

    impl CBORDecodable for Test {
        // This ensures that asking for the CBOR for this type will always
        // expect a tagged CBOR value.
        fn from_cbor(cbor: &CBOR) -> anyhow::Result<Self> {
            Self::from_tagged_cbor(cbor)
        }
    }

    impl TryFrom<CBOR> for Test {
        type Error = anyhow::Error;

        fn try_from(cbor: CBOR) -> Result<Self, Self::Error> {
            Self::from_cbor(&cbor)
        }
    }

    impl CBORTaggedDecodable for Test {
        // This is the core of the CBOR decoding for this type. It is the
        // untagged CBOR decoding.
        fn from_untagged_cbor(cbor: &CBOR) -> anyhow::Result<Self> {
            Ok(Self::new(&String::from_cbor(cbor)?))
        }
    }

    impl URDecodable for Test {}

    impl URCodable for Test {}

    #[test]
    fn test_ur_codable() {
        let test = Test::new("test");
        let ur = test.ur();
        let ur_string = ur.string();
        assert_eq!(ur_string, "ur:leaf/iejyihjkjygupyltla");
        let test2 = Test::from_ur_string(ur_string).unwrap();
        assert_eq!(test.s, test2.s);
    }
}