1use crate::balances::credits::TokenAmount;
2use crate::data_contract::associated_token::token_configuration_convention::TokenConfigurationConvention;
3use crate::data_contract::associated_token::token_marketplace_rules::v0::TokenTradeMode;
4use crate::data_contract::associated_token::token_perpetual_distribution::TokenPerpetualDistribution;
5use crate::data_contract::change_control_rules::authorized_action_takers::AuthorizedActionTakers;
6use crate::data_contract::GroupContractPosition;
7use crate::ProtocolError;
8use bincode::Encode;
9use platform_serialization::de::Decode;
10use platform_serialization_derive::{PlatformDeserialize, PlatformSerialize};
11use platform_value::Identifier;
12#[cfg(feature = "serde-conversion")]
13use serde::{Deserialize, Serialize};
14use std::fmt;
15
16#[derive(
17 Debug,
18 Clone,
19 Default,
20 PartialOrd,
21 Encode,
22 Decode,
23 PlatformSerialize,
24 PlatformDeserialize,
25 PartialEq,
26 Eq,
27)]
28#[cfg_attr(
29 feature = "serde-conversion",
30 derive(Serialize, Deserialize),
31 serde(rename_all = "camelCase")
32)]
33pub enum TokenConfigurationChangeItem {
34 #[default]
35 TokenConfigurationNoChange,
36 Conventions(TokenConfigurationConvention),
37 ConventionsControlGroup(AuthorizedActionTakers),
38 ConventionsAdminGroup(AuthorizedActionTakers),
39 MaxSupply(Option<TokenAmount>),
40 MaxSupplyControlGroup(AuthorizedActionTakers),
41 MaxSupplyAdminGroup(AuthorizedActionTakers),
42 PerpetualDistribution(Option<TokenPerpetualDistribution>),
43 PerpetualDistributionControlGroup(AuthorizedActionTakers),
44 PerpetualDistributionAdminGroup(AuthorizedActionTakers),
45 NewTokensDestinationIdentity(Option<Identifier>),
46 NewTokensDestinationIdentityControlGroup(AuthorizedActionTakers),
47 NewTokensDestinationIdentityAdminGroup(AuthorizedActionTakers),
48 MintingAllowChoosingDestination(bool),
49 MintingAllowChoosingDestinationControlGroup(AuthorizedActionTakers),
50 MintingAllowChoosingDestinationAdminGroup(AuthorizedActionTakers),
51 ManualMinting(AuthorizedActionTakers),
52 ManualMintingAdminGroup(AuthorizedActionTakers),
53 ManualBurning(AuthorizedActionTakers),
54 ManualBurningAdminGroup(AuthorizedActionTakers),
55 Freeze(AuthorizedActionTakers),
56 FreezeAdminGroup(AuthorizedActionTakers),
57 Unfreeze(AuthorizedActionTakers),
58 UnfreezeAdminGroup(AuthorizedActionTakers),
59 DestroyFrozenFunds(AuthorizedActionTakers),
60 DestroyFrozenFundsAdminGroup(AuthorizedActionTakers),
61 EmergencyAction(AuthorizedActionTakers),
62 EmergencyActionAdminGroup(AuthorizedActionTakers),
63 MarketplaceTradeMode(TokenTradeMode),
64 MarketplaceTradeModeControlGroup(AuthorizedActionTakers),
65 MarketplaceTradeModeAdminGroup(AuthorizedActionTakers),
66 MainControlGroup(Option<GroupContractPosition>),
67}
68impl TokenConfigurationChangeItem {
69 pub fn payload_serialization(&self) -> Result<Option<Vec<u8>>, ProtocolError> {
70 Ok(match self {
71 TokenConfigurationChangeItem::TokenConfigurationNoChange => None,
72 TokenConfigurationChangeItem::Conventions(convention) => Some(
73 bincode::encode_to_vec(convention, bincode::config::standard())
74 .map_err(|e| ProtocolError::EncodingError(e.to_string()))?,
75 ),
76 TokenConfigurationChangeItem::ConventionsControlGroup(a)
77 | TokenConfigurationChangeItem::ConventionsAdminGroup(a)
78 | TokenConfigurationChangeItem::MaxSupplyControlGroup(a)
79 | TokenConfigurationChangeItem::MaxSupplyAdminGroup(a)
80 | TokenConfigurationChangeItem::PerpetualDistributionControlGroup(a)
81 | TokenConfigurationChangeItem::PerpetualDistributionAdminGroup(a)
82 | TokenConfigurationChangeItem::NewTokensDestinationIdentityControlGroup(a)
83 | TokenConfigurationChangeItem::NewTokensDestinationIdentityAdminGroup(a)
84 | TokenConfigurationChangeItem::MintingAllowChoosingDestinationControlGroup(a)
85 | TokenConfigurationChangeItem::MintingAllowChoosingDestinationAdminGroup(a)
86 | TokenConfigurationChangeItem::ManualMinting(a)
87 | TokenConfigurationChangeItem::ManualMintingAdminGroup(a)
88 | TokenConfigurationChangeItem::ManualBurning(a)
89 | TokenConfigurationChangeItem::ManualBurningAdminGroup(a)
90 | TokenConfigurationChangeItem::Freeze(a)
91 | TokenConfigurationChangeItem::FreezeAdminGroup(a)
92 | TokenConfigurationChangeItem::Unfreeze(a)
93 | TokenConfigurationChangeItem::UnfreezeAdminGroup(a)
94 | TokenConfigurationChangeItem::DestroyFrozenFunds(a)
95 | TokenConfigurationChangeItem::DestroyFrozenFundsAdminGroup(a)
96 | TokenConfigurationChangeItem::EmergencyAction(a)
97 | TokenConfigurationChangeItem::EmergencyActionAdminGroup(a)
98 | TokenConfigurationChangeItem::MarketplaceTradeModeControlGroup(a)
99 | TokenConfigurationChangeItem::MarketplaceTradeModeAdminGroup(a) => Some(a.to_bytes()),
100 TokenConfigurationChangeItem::MaxSupply(max_supply) => {
101 max_supply.map(|amount| amount.to_be_bytes().to_vec())
102 }
103 TokenConfigurationChangeItem::PerpetualDistribution(distribution) => distribution
104 .as_ref()
105 .map(|dist| {
106 bincode::encode_to_vec(dist, bincode::config::standard())
107 .map_err(|e| ProtocolError::EncodingError(e.to_string()))
108 })
109 .transpose()?,
110 TokenConfigurationChangeItem::NewTokensDestinationIdentity(identity) => {
111 identity.map(|id| id.to_vec())
112 }
113 TokenConfigurationChangeItem::MintingAllowChoosingDestination(allow) => {
114 Some(vec![*allow as u8])
115 }
116 TokenConfigurationChangeItem::MarketplaceTradeMode(mode) => Some(
117 bincode::encode_to_vec(mode, bincode::config::standard())
118 .map_err(|e| ProtocolError::EncodingError(e.to_string()))?,
119 ),
120 TokenConfigurationChangeItem::MainControlGroup(position) => {
121 position.map(|pos| pos.to_be_bytes().to_vec())
122 }
123 })
124 }
125 pub fn u8_item_index(&self) -> u8 {
126 match self {
127 TokenConfigurationChangeItem::TokenConfigurationNoChange => 0,
128 TokenConfigurationChangeItem::Conventions(_) => 1,
129 TokenConfigurationChangeItem::ConventionsControlGroup(_) => 2,
130 TokenConfigurationChangeItem::ConventionsAdminGroup(_) => 3,
131 TokenConfigurationChangeItem::MaxSupply(_) => 4,
132 TokenConfigurationChangeItem::MaxSupplyControlGroup(_) => 5,
133 TokenConfigurationChangeItem::MaxSupplyAdminGroup(_) => 6,
134 TokenConfigurationChangeItem::PerpetualDistribution(_) => 7,
135 TokenConfigurationChangeItem::PerpetualDistributionControlGroup(_) => 8,
136 TokenConfigurationChangeItem::PerpetualDistributionAdminGroup(_) => 9,
137 TokenConfigurationChangeItem::NewTokensDestinationIdentity(_) => 10,
138 TokenConfigurationChangeItem::NewTokensDestinationIdentityControlGroup(_) => 11,
139 TokenConfigurationChangeItem::NewTokensDestinationIdentityAdminGroup(_) => 12,
140 TokenConfigurationChangeItem::MintingAllowChoosingDestination(_) => 13,
141 TokenConfigurationChangeItem::MintingAllowChoosingDestinationControlGroup(_) => 14,
142 TokenConfigurationChangeItem::MintingAllowChoosingDestinationAdminGroup(_) => 15,
143 TokenConfigurationChangeItem::ManualMinting(_) => 16,
144 TokenConfigurationChangeItem::ManualMintingAdminGroup(_) => 17,
145 TokenConfigurationChangeItem::ManualBurning(_) => 18,
146 TokenConfigurationChangeItem::ManualBurningAdminGroup(_) => 19,
147 TokenConfigurationChangeItem::Freeze(_) => 20,
148 TokenConfigurationChangeItem::FreezeAdminGroup(_) => 21,
149 TokenConfigurationChangeItem::Unfreeze(_) => 22,
150 TokenConfigurationChangeItem::UnfreezeAdminGroup(_) => 23,
151 TokenConfigurationChangeItem::DestroyFrozenFunds(_) => 24,
152 TokenConfigurationChangeItem::DestroyFrozenFundsAdminGroup(_) => 25,
153 TokenConfigurationChangeItem::EmergencyAction(_) => 26,
154 TokenConfigurationChangeItem::EmergencyActionAdminGroup(_) => 27,
155 TokenConfigurationChangeItem::MarketplaceTradeMode(_) => 28,
156 TokenConfigurationChangeItem::MarketplaceTradeModeControlGroup(_) => 29,
157 TokenConfigurationChangeItem::MarketplaceTradeModeAdminGroup(_) => 30,
158 TokenConfigurationChangeItem::MainControlGroup(_) => 31,
159 }
160 }
161}
162
163impl fmt::Display for TokenConfigurationChangeItem {
164 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
165 match self {
166 TokenConfigurationChangeItem::TokenConfigurationNoChange => {
167 write!(f, "No Change in Token Configuration")
168 }
169 TokenConfigurationChangeItem::Conventions(convention) => {
170 write!(f, "Conventions: {}", convention)
171 }
172 TokenConfigurationChangeItem::ConventionsControlGroup(control_group) => {
173 write!(f, "Conventions Control Group: {}", control_group)
174 }
175 TokenConfigurationChangeItem::ConventionsAdminGroup(admin_group) => {
176 write!(f, "Conventions Admin Group: {}", admin_group)
177 }
178 TokenConfigurationChangeItem::MaxSupply(max_supply) => match max_supply {
179 Some(amount) => write!(f, "Max Supply: {}", amount),
180 None => write!(f, "Max Supply: No Limit"),
181 },
182 TokenConfigurationChangeItem::MaxSupplyControlGroup(control_group) => {
183 write!(f, "Max Supply Control Group: {}", control_group)
184 }
185 TokenConfigurationChangeItem::MaxSupplyAdminGroup(admin_group) => {
186 write!(f, "Max Supply Admin Group: {}", admin_group)
187 }
188 TokenConfigurationChangeItem::PerpetualDistribution(distribution) => match distribution
189 {
190 Some(dist) => write!(f, "Perpetual Distribution: {}", dist),
191 None => write!(f, "Perpetual Distribution: None"),
192 },
193 TokenConfigurationChangeItem::PerpetualDistributionControlGroup(control_group) => {
194 write!(f, "Perpetual Distribution Control Group: {}", control_group)
195 }
196 TokenConfigurationChangeItem::PerpetualDistributionAdminGroup(admin_group) => {
197 write!(f, "Perpetual Distribution Admin Group: {}", admin_group)
198 }
199 TokenConfigurationChangeItem::NewTokensDestinationIdentity(identity) => {
200 match identity {
201 Some(id) => write!(f, "New Tokens Destination Identity: {}", id),
202 None => write!(f, "New Tokens Destination Identity: None"),
203 }
204 }
205 TokenConfigurationChangeItem::NewTokensDestinationIdentityControlGroup(
206 control_group,
207 ) => {
208 write!(
209 f,
210 "New Tokens Destination Identity Control Group: {}",
211 control_group
212 )
213 }
214 TokenConfigurationChangeItem::NewTokensDestinationIdentityAdminGroup(admin_group) => {
215 write!(
216 f,
217 "New Tokens Destination Identity Admin Group: {}",
218 admin_group
219 )
220 }
221 TokenConfigurationChangeItem::MintingAllowChoosingDestination(allow) => {
222 write!(f, "Minting Allow Choosing Destination: {}", allow)
223 }
224 TokenConfigurationChangeItem::MintingAllowChoosingDestinationControlGroup(
225 control_group,
226 ) => {
227 write!(
228 f,
229 "Minting Allow Choosing Destination Control Group: {}",
230 control_group
231 )
232 }
233 TokenConfigurationChangeItem::MintingAllowChoosingDestinationAdminGroup(
234 admin_group,
235 ) => {
236 write!(
237 f,
238 "Minting Allow Choosing Destination Admin Group: {}",
239 admin_group
240 )
241 }
242 TokenConfigurationChangeItem::ManualMinting(control_group) => {
243 write!(f, "Manual Minting: {}", control_group)
244 }
245 TokenConfigurationChangeItem::ManualMintingAdminGroup(admin_group) => {
246 write!(f, "Manual Minting Admin Group: {}", admin_group)
247 }
248 TokenConfigurationChangeItem::ManualBurning(control_group) => {
249 write!(f, "Manual Burning: {}", control_group)
250 }
251 TokenConfigurationChangeItem::ManualBurningAdminGroup(admin_group) => {
252 write!(f, "Manual Burning Admin Group: {}", admin_group)
253 }
254 TokenConfigurationChangeItem::Freeze(control_group) => {
255 write!(f, "Freeze: {}", control_group)
256 }
257 TokenConfigurationChangeItem::FreezeAdminGroup(admin_group) => {
258 write!(f, "Freeze Admin Group: {}", admin_group)
259 }
260 TokenConfigurationChangeItem::Unfreeze(control_group) => {
261 write!(f, "Unfreeze: {}", control_group)
262 }
263 TokenConfigurationChangeItem::UnfreezeAdminGroup(admin_group) => {
264 write!(f, "Unfreeze Admin Group: {}", admin_group)
265 }
266 TokenConfigurationChangeItem::DestroyFrozenFunds(control_group) => {
267 write!(f, "Destroy Frozen Funds: {}", control_group)
268 }
269 TokenConfigurationChangeItem::DestroyFrozenFundsAdminGroup(admin_group) => {
270 write!(f, "Destroy Frozen Funds Admin Group: {}", admin_group)
271 }
272 TokenConfigurationChangeItem::EmergencyAction(control_group) => {
273 write!(f, "Emergency Action: {}", control_group)
274 }
275 TokenConfigurationChangeItem::EmergencyActionAdminGroup(admin_group) => {
276 write!(f, "Emergency Action Admin Group: {}", admin_group)
277 }
278 TokenConfigurationChangeItem::MainControlGroup(position) => match position {
279 Some(pos) => write!(f, "Main Control Group: {}", pos),
280 None => write!(f, "Main Control Group: None"),
281 },
282 TokenConfigurationChangeItem::MarketplaceTradeMode(mode) => {
283 write!(f, "Marketplace Trade Mode: {:?}", mode)
284 }
285 TokenConfigurationChangeItem::MarketplaceTradeModeControlGroup(control_group) => {
286 write!(f, "Marketplace Trade Mode Control Group: {}", control_group)
287 }
288 TokenConfigurationChangeItem::MarketplaceTradeModeAdminGroup(admin_group) => {
289 write!(f, "Marketplace Trade Mode Admin Group: {}", admin_group)
290 }
291 }
292 }
293}
294
295#[cfg(test)]
296mod tests {
297 use super::*;
298 use std::collections::BTreeSet;
299
300 fn all_variants() -> Vec<TokenConfigurationChangeItem> {
302 let aat = AuthorizedActionTakers::NoOne;
303 vec![
304 TokenConfigurationChangeItem::TokenConfigurationNoChange,
305 TokenConfigurationChangeItem::Conventions(
306 TokenConfigurationConvention::V0(
307 crate::data_contract::associated_token::token_configuration_convention::v0::TokenConfigurationConventionV0::default(),
308 ),
309 ),
310 TokenConfigurationChangeItem::ConventionsControlGroup(aat.clone()),
311 TokenConfigurationChangeItem::ConventionsAdminGroup(aat.clone()),
312 TokenConfigurationChangeItem::MaxSupply(None),
313 TokenConfigurationChangeItem::MaxSupplyControlGroup(aat.clone()),
314 TokenConfigurationChangeItem::MaxSupplyAdminGroup(aat.clone()),
315 TokenConfigurationChangeItem::PerpetualDistribution(None),
316 TokenConfigurationChangeItem::PerpetualDistributionControlGroup(aat.clone()),
317 TokenConfigurationChangeItem::PerpetualDistributionAdminGroup(aat.clone()),
318 TokenConfigurationChangeItem::NewTokensDestinationIdentity(None),
319 TokenConfigurationChangeItem::NewTokensDestinationIdentityControlGroup(aat.clone()),
320 TokenConfigurationChangeItem::NewTokensDestinationIdentityAdminGroup(aat.clone()),
321 TokenConfigurationChangeItem::MintingAllowChoosingDestination(false),
322 TokenConfigurationChangeItem::MintingAllowChoosingDestinationControlGroup(aat.clone()),
323 TokenConfigurationChangeItem::MintingAllowChoosingDestinationAdminGroup(aat.clone()),
324 TokenConfigurationChangeItem::ManualMinting(aat.clone()),
325 TokenConfigurationChangeItem::ManualMintingAdminGroup(aat.clone()),
326 TokenConfigurationChangeItem::ManualBurning(aat.clone()),
327 TokenConfigurationChangeItem::ManualBurningAdminGroup(aat.clone()),
328 TokenConfigurationChangeItem::Freeze(aat.clone()),
329 TokenConfigurationChangeItem::FreezeAdminGroup(aat.clone()),
330 TokenConfigurationChangeItem::Unfreeze(aat.clone()),
331 TokenConfigurationChangeItem::UnfreezeAdminGroup(aat.clone()),
332 TokenConfigurationChangeItem::DestroyFrozenFunds(aat.clone()),
333 TokenConfigurationChangeItem::DestroyFrozenFundsAdminGroup(aat.clone()),
334 TokenConfigurationChangeItem::EmergencyAction(aat.clone()),
335 TokenConfigurationChangeItem::EmergencyActionAdminGroup(aat.clone()),
336 TokenConfigurationChangeItem::MarketplaceTradeMode(TokenTradeMode::default()),
337 TokenConfigurationChangeItem::MarketplaceTradeModeControlGroup(aat.clone()),
338 TokenConfigurationChangeItem::MarketplaceTradeModeAdminGroup(aat.clone()),
339 TokenConfigurationChangeItem::MainControlGroup(None),
340 ]
341 }
342
343 #[test]
346 fn u8_item_index_values_are_unique() {
347 let variants = all_variants();
348 let indices: Vec<u8> = variants.iter().map(|v| v.u8_item_index()).collect();
349 let unique: BTreeSet<u8> = indices.iter().cloned().collect();
350 assert_eq!(
351 indices.len(),
352 unique.len(),
353 "Duplicate u8_item_index values found: {:?}",
354 indices
355 );
356 }
357
358 #[test]
359 fn u8_item_index_covers_0_through_31() {
360 let variants = all_variants();
361 let indices: BTreeSet<u8> = variants.iter().map(|v| v.u8_item_index()).collect();
362 for i in 0u8..=31 {
363 assert!(indices.contains(&i), "Missing u8_item_index value: {}", i);
364 }
365 }
366
367 #[test]
368 fn u8_item_index_all_within_range() {
369 let variants = all_variants();
370 for v in &variants {
371 let idx = v.u8_item_index();
372 assert!(idx <= 31, "Index {} exceeds expected max of 31", idx);
373 }
374 }
375
376 #[test]
377 fn u8_item_index_specific_known_values() {
378 assert_eq!(
379 TokenConfigurationChangeItem::TokenConfigurationNoChange.u8_item_index(),
380 0
381 );
382 assert_eq!(
383 TokenConfigurationChangeItem::MaxSupply(Some(100)).u8_item_index(),
384 4
385 );
386 assert_eq!(
387 TokenConfigurationChangeItem::ManualMinting(AuthorizedActionTakers::NoOne)
388 .u8_item_index(),
389 16
390 );
391 assert_eq!(
392 TokenConfigurationChangeItem::MainControlGroup(Some(5)).u8_item_index(),
393 31
394 );
395 }
396
397 #[test]
398 fn u8_item_index_variant_count() {
399 let variants = all_variants();
401 assert_eq!(variants.len(), 32);
402 }
403}