1use std::str::FromStr;
6
7use eyre::eyre;
8use fastcrypto::encoding::decode_bytes_hex;
9use serde::{Deserialize, Serialize};
10use serde_repr::{Deserialize_repr, Serialize_repr};
11
12pub const INTENT_PREFIX_LENGTH: usize = 3;
13
14#[derive(Serialize_repr, Deserialize_repr, Copy, Clone, PartialEq, Eq, Debug, Hash)]
18#[repr(u8)]
19pub enum IntentVersion {
20 V0 = 0,
21}
22
23impl TryFrom<u8> for IntentVersion {
24 type Error = eyre::Report;
25 fn try_from(value: u8) -> Result<Self, Self::Error> {
26 bcs::from_bytes(&[value]).map_err(|_| eyre!("Invalid IntentVersion"))
27 }
28}
29
30#[derive(Serialize_repr, Deserialize_repr, Copy, Clone, PartialEq, Eq, Debug, Hash)]
36#[repr(u8)]
37pub enum AppId {
38 Iota = 0,
39 Consensus = 1,
40}
41
42impl TryFrom<u8> for AppId {
44 type Error = eyre::Report;
45 fn try_from(value: u8) -> Result<Self, Self::Error> {
46 bcs::from_bytes(&[value]).map_err(|_| eyre!("Invalid AppId"))
47 }
48}
49
50impl Default for AppId {
51 fn default() -> Self {
52 Self::Iota
53 }
54}
55
56#[derive(Serialize_repr, Deserialize_repr, Copy, Clone, PartialEq, Eq, Debug, Hash)]
60#[repr(u8)]
61pub enum IntentScope {
62 TransactionData = 0, TransactionEffects = 1, CheckpointSummary = 2, PersonalMessage = 3, SenderSignedTransaction = 4, ProofOfPossession = 5, BridgeEventUnused = 6, ConsensusBlock = 7, }
72
73impl TryFrom<u8> for IntentScope {
74 type Error = eyre::Report;
75 fn try_from(value: u8) -> Result<Self, Self::Error> {
76 bcs::from_bytes(&[value]).map_err(|_| eyre!("Invalid IntentScope"))
77 }
78}
79
80#[derive(Debug, PartialEq, Eq, Serialize, Deserialize, Clone, Hash)]
89pub struct Intent {
90 pub scope: IntentScope,
91 pub version: IntentVersion,
92 pub app_id: AppId,
93}
94
95impl Intent {
96 pub fn to_bytes(&self) -> [u8; INTENT_PREFIX_LENGTH] {
97 [self.scope as u8, self.version as u8, self.app_id as u8]
98 }
99
100 pub fn from_bytes(bytes: &[u8]) -> Result<Self, eyre::Report> {
101 if bytes.len() != INTENT_PREFIX_LENGTH {
102 return Err(eyre!("Invalid Intent"));
103 }
104 Ok(Self {
105 scope: bytes[0].try_into()?,
106 version: bytes[1].try_into()?,
107 app_id: bytes[2].try_into()?,
108 })
109 }
110}
111
112impl FromStr for Intent {
113 type Err = eyre::Report;
114 fn from_str(s: &str) -> Result<Self, Self::Err> {
115 let bytes: Vec<u8> = decode_bytes_hex(s).map_err(|_| eyre!("Invalid Intent"))?;
116 Self::from_bytes(bytes.as_slice())
117 }
118}
119
120impl Intent {
121 pub fn iota_app(scope: IntentScope) -> Self {
122 Self {
123 version: IntentVersion::V0,
124 scope,
125 app_id: AppId::Iota,
126 }
127 }
128
129 pub fn iota_transaction() -> Self {
130 Self {
131 scope: IntentScope::TransactionData,
132 version: IntentVersion::V0,
133 app_id: AppId::Iota,
134 }
135 }
136
137 pub fn personal_message() -> Self {
138 Self {
139 scope: IntentScope::PersonalMessage,
140 version: IntentVersion::V0,
141 app_id: AppId::Iota,
142 }
143 }
144
145 pub fn consensus_app(scope: IntentScope) -> Self {
146 Self {
147 scope,
148 version: IntentVersion::V0,
149 app_id: AppId::Consensus,
150 }
151 }
152}
153
154#[derive(Debug, PartialEq, Eq, Serialize, Clone, Hash, Deserialize)]
163pub struct IntentMessage<T> {
164 pub intent: Intent,
165 pub value: T,
166}
167
168impl<T> IntentMessage<T> {
169 pub fn new(intent: Intent, value: T) -> Self {
170 Self { intent, value }
171 }
172}
173
174#[derive(Debug, PartialEq, Eq, Clone, Serialize, Deserialize)]
176pub struct PersonalMessage {
177 pub message: Vec<u8>,
178}
179
180pub trait SecureIntent: Serialize + private::SealedIntent {}
181
182pub(crate) mod private {
183 use super::IntentMessage;
184
185 pub trait SealedIntent {}
186 impl<T> SealedIntent for IntentMessage<T> {}
187}
188
189#[derive(Serialize_repr, Deserialize_repr, Copy, Clone, PartialEq, Eq, Debug, Hash)]
194#[repr(u8)]
195pub enum HashingIntentScope {
196 ChildObjectId = 0xf0,
197 RegularObjectId = 0xf1,
198}