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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#![cfg_attr(not(feature = "std"), no_std)]
use scale_info::prelude::string::String;
#[cfg(feature = "runtime-benchmarks")]
pub mod benchmarks;
#[cfg(test)]
mod mock;
#[cfg(test)]
mod tests;
mod weights;
pub use pallet::*;
use weights::{SubstrateWeight, WeightInfo};
#[frame_support::pallet]
pub mod pallet {
use super::*;
use frame_support::{
dispatch::WithPostDispatchInfo,
pallet_prelude::*,
traits::{Currency, ExistenceRequirement::AllowDeath},
};
use frame_system::pallet_prelude::*;
use sp_runtime::traits::Zero;
type BalanceOf<T> =
<<T as Config>::Currency as Currency<<T as frame_system::Config>::AccountId>>::Balance;
#[derive(Encode, Decode, Clone, Debug, PartialEq, Eq, Ord, PartialOrd)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
#[cfg_attr(feature = "serde", serde(rename_all = "camelCase"))]
#[derive(scale_info_derive::TypeInfo)]
#[scale_info(omit_prefix)]
pub struct CheqdAddress(String);
impl CheqdAddress {
pub fn new<T>(value: String) -> Result<Self, Error<T>> {
let (prefix, addr) =
bech32::decode(&value).map_err(|_| Error::<T>::AddressMustBeValidBech32)?;
ensure!(
prefix.as_str() == "cheqd",
Error::<T>::AddressMustStartWithCheqd
);
ensure!(addr.len() == 20, Error::<T>::InvalidAddressLength);
Ok(Self(value))
}
}
#[pallet::config]
pub trait Config: frame_system::Config {
type Event: From<Event<Self>> + IsType<<Self as frame_system::Config>::Event>;
type Currency: Currency<Self::AccountId>;
type BurnDestination: Get<Self::AccountId>;
}
#[pallet::error]
pub enum Error<T> {
AddressMustStartWithCheqd,
InvalidAddressLength,
AddressMustBeValidBech32,
BalanceIsZero,
}
#[pallet::pallet]
pub struct Pallet<T>(_);
#[pallet::call]
impl<T: Config> Pallet<T> {
#[pallet::weight(SubstrateWeight::<T::DbWeight>::migrate())]
pub fn migrate(origin: OriginFor<T>, cheqd_address: String) -> DispatchResultWithPostInfo {
let dock_account = ensure_signed(origin)?;
let cheqd_account = CheqdAddress::new::<T>(cheqd_address)
.map_err(DispatchError::from)
.map_err(|error| {
error.with_weight(SubstrateWeight::<T::DbWeight>::migrate_validation_failure())
})?;
let dock_tokens_amount = T::Currency::free_balance(&dock_account);
ensure!(!dock_tokens_amount.is_zero(), Error::<T>::BalanceIsZero);
let dest = T::BurnDestination::get();
T::Currency::transfer(&dock_account, &dest, dock_tokens_amount, AllowDeath)?;
Self::deposit_event(Event::Migrated {
dock_account,
cheqd_account,
dock_tokens_amount,
accepted_terms_and_conditions: true,
});
Ok(Pays::Yes.into())
}
}
#[pallet::event]
#[pallet::generate_deposit(pub(super) fn deposit_event)]
pub enum Event<T: Config> {
Migrated {
dock_account: T::AccountId,
cheqd_account: CheqdAddress,
dock_tokens_amount: BalanceOf<T>,
accepted_terms_and_conditions: bool,
},
}
}