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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/// the authenticator API, consisting of "operations"
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
pub enum Operation {
    MakeCredential,
    GetAssertion,
    GetNextAssertion,
    GetInfo,
    ClientPin,
    Reset,
    // new in v2.1
    BioEnrollment,
    CredentialManagement,
    Selection,
    LargeBlobs,
    Config,
    PreviewBioEnrollment,
    PreviewCredentialManagement,
    /// vendors are assigned the range 0x40..=0x7f for custom operations
    Vendor(VendorOperation),
}

impl From<Operation> for u8 {
    fn from(operation: Operation) -> u8 {
        use Operation::*;
        match operation {
            MakeCredential => 0x01,
            GetAssertion => 0x02,
            GetNextAssertion => 0x08,
            GetInfo => 0x04,
            ClientPin => 0x06,
            Reset => 0x07,
            BioEnrollment => 0x09,
            CredentialManagement => 0x0A,
            Selection => 0x0B,
            LargeBlobs => 0x0C,
            Config => 0x0D,
            PreviewBioEnrollment => 0x40,
            PreviewCredentialManagement => 0x41,
            Vendor(operation) => operation.into(),
        }
    }
}

impl Operation {
    pub fn into_u8(self) -> u8 {
        self.into()
    }
}

/// Vendor CTAP2 operations, from 0x40 to 0x7f.
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
pub struct VendorOperation(u8);

impl VendorOperation {
    pub const FIRST: u8 = 0x40;
    pub const LAST: u8 = 0x7f;
}

impl TryFrom<u8> for VendorOperation {
    type Error = ();

    fn try_from(from: u8) -> core::result::Result<Self, ()> {
        match from {
            code @ Self::FIRST..=Self::LAST => Ok(VendorOperation(code)),
            _ => Err(()),
        }
    }
}

impl From<VendorOperation> for u8 {
    fn from(operation: VendorOperation) -> u8 {
        operation.0
    }
}

impl TryFrom<u8> for Operation {
    type Error = ();

    fn try_from(from: u8) -> core::result::Result<Operation, ()> {
        use Operation::*;
        Ok(match from {
            0x01 => MakeCredential,
            0x02 => GetAssertion,
            0x08 => GetNextAssertion,
            0x04 => GetInfo,
            0x06 => ClientPin,
            0x07 => Reset,
            0x09 => BioEnrollment,
            0x0A => CredentialManagement,
            0x0B => Selection,
            0x0C => LargeBlobs,
            0x0D => Config,
            0x40 => PreviewBioEnrollment,
            0x41 => PreviewCredentialManagement,
            code @ VendorOperation::FIRST..=VendorOperation::LAST => {
                Vendor(VendorOperation::try_from(code)?)
            }
            _ => return Err(()),
        })
    }
}