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
// Copyright 2020-2022 IOTA Stiftung
// SPDX-License-Identifier: Apache-2.0

mod simple;

pub use self::simple::SimpleTokenScheme;
use crate::types::block::Error;

///
#[derive(Clone, Eq, PartialEq, Ord, PartialOrd, Hash, derive_more::From, packable::Packable)]
#[packable(unpack_error = Error)]
#[packable(tag_type = u8, with_error = Error::InvalidTokenSchemeKind)]
pub enum TokenScheme {
    ///
    #[packable(tag = SimpleTokenScheme::KIND)]
    Simple(SimpleTokenScheme),
}

impl core::fmt::Debug for TokenScheme {
    fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
        match self {
            Self::Simple(scheme) => scheme.fmt(f),
        }
    }
}

impl TokenScheme {
    /// Returns the token scheme kind of a [`TokenScheme`].
    pub fn kind(&self) -> u8 {
        match self {
            Self::Simple(_) => SimpleTokenScheme::KIND,
        }
    }

    /// Checks whether the token scheme is a [`SimpleTokenScheme`].
    pub fn is_simple(&self) -> bool {
        matches!(self, Self::Simple(_))
    }

    /// Gets the token scheme as an actual [`SimpleTokenScheme`].
    /// PANIC: do not call on a non-simple token scheme.
    pub fn as_simple(&self) -> &SimpleTokenScheme {
        let Self::Simple(scheme) = self;
        scheme
    }
}

#[cfg(feature = "serde")]
pub(crate) mod dto {
    use derive_more::From;
    use serde::{Deserialize, Serialize};

    pub use super::simple::dto::SimpleTokenSchemeDto;
    use super::*;
    use crate::types::block::Error;

    #[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize, From)]
    #[serde(untagged)]
    pub enum TokenSchemeDto {
        /// A simple token scheme.
        Simple(SimpleTokenSchemeDto),
    }

    impl From<&TokenScheme> for TokenSchemeDto {
        fn from(value: &TokenScheme) -> Self {
            match value {
                TokenScheme::Simple(v) => Self::Simple(v.into()),
            }
        }
    }

    impl TryFrom<TokenSchemeDto> for TokenScheme {
        type Error = Error;

        fn try_from(value: TokenSchemeDto) -> Result<Self, Self::Error> {
            Ok(match value {
                TokenSchemeDto::Simple(v) => Self::Simple(v.try_into()?),
            })
        }
    }
}