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 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324
/* * Copyright 2020 Nuclei Studio OÜ * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ //! This pallet implements the code to support a multi currency runtime. //! Along with compatibility with the `Currency` trait through the use //! of `NativeCurrencyAdapter`. //! Caveats: //! - for now, we do not support `reasons` and `existence_requirements` //! - for now, we do not support `ExistentialDeposit` //! - for now, we do not support locking or reserving funds #![cfg_attr(not(feature = "std"), no_std)] use frame_support::{ decl_error, decl_event, decl_module, decl_storage, traits::LockIdentifier, weights::Weight, Parameter, }; use frame_system::ensure_signed; use governance_os_support::traits::{Currencies, RoleManager}; use sp_runtime::{ traits::{AtLeast32BitUnsigned, MaybeSerializeDeserialize, Member, StaticLookup, Zero}, DispatchResult, }; use sp_std::cmp::{Eq, PartialEq}; #[cfg(feature = "runtime-benchmarks")] mod benchmarking; #[cfg(test)] mod tests; mod account_data; mod adapter; mod currencies; mod default_weights; mod details; mod imbalances; mod mutations; pub use account_data::AccountCurrencyData; pub use adapter::NativeCurrencyAdapter; pub use details::CurrencyDetails; pub use imbalances::{NegativeImbalance, PositiveImbalance}; pub trait WeightInfo { fn create() -> Weight; fn mint() -> Weight; fn burn() -> Weight; fn update_details() -> Weight; fn transfer() -> Weight; } pub trait RoleBuilder { type CurrencyId; type Role; /// Role for transferring certain units of currencies. This is used to create /// non transferrable assets or could be used for creating permissioned assets /// in the future. fn transfer_currency(id: Self::CurrencyId) -> Self::Role; /// Role for the account(s) that are allowed to `mint` or `burn` units of currency. fn manage_currency(id: Self::CurrencyId) -> Self::Role; /// Role for creating currencies. fn create_currencies() -> Self::Role; } type RoleBuilderOf<T> = <T as Config>::RoleBuilder; pub trait Config: frame_system::Config { /// Because this pallet emits events, it depends on the runtime's definition of an event. type Event: From<Event<Self>> + Into<<Self as frame_system::Config>::Event>; /// The type used to identify currencies type CurrencyId: Parameter + Member + Copy + MaybeSerializeDeserialize + Ord + Default; /// The balance of an account. type Balance: Parameter + Member + AtLeast32BitUnsigned + Default + Copy + MaybeSerializeDeserialize; /// Weight values for this pallet type WeightInfo: WeightInfo; /// Pallet that is in charge of managing the roles based ACL. type RoleManager: RoleManager<AccountId = Self::AccountId>; /// This pallet relies on roles associated to a specific metadata so we need the runtime /// to provide some helper functions to build those so that we can keep the role definition /// code modular. type RoleBuilder: RoleBuilder< CurrencyId = Self::CurrencyId, Role = <RoleManagerOf<Self> as RoleManager>::Role, >; } type RoleManagerOf<T> = <T as Config>::RoleManager; decl_storage! { trait Store for Module<T: Config> as Tokens { /// Store the balances holded by an account. By storing the balances under an account (VS storing /// the accounts under the currency ids) we can enumerate the tokens holded by an account if needed. pub Balances get(fn balances): double_map hasher(blake2_128_concat) T::AccountId, hasher(blake2_128_concat) T::CurrencyId => AccountCurrencyData<T::Balance>; pub Locks get(fn locks): double_map hasher(blake2_128_concat) (T::AccountId, T::CurrencyId), hasher(blake2_128_concat) LockIdentifier => T::Balance; pub TotalIssuances get(fn total_issuances): map hasher(blake2_128_concat) T::CurrencyId => T::Balance; } add_extra_genesis { config(endowed_accounts): Vec<(T::CurrencyId, T::AccountId, T::Balance)>; config(currency_details): Vec<(T::CurrencyId, CurrencyDetails<T::AccountId>)>; build(|config: &GenesisConfig<T>| { config.currency_details.iter().cloned().for_each(|(currency_id, currency_details)| { Module::<T>::set_currency_acl(currency_id, currency_details, None); // If we have an error it means that the currency had some coins issued earlier in // the genesis block, thus we ignore it. drop(Module::<T>::maybe_create_zero_issuance(currency_id)); }); config.endowed_accounts.iter().for_each(|(currency_id, account_id, initial_balance)| { assert!(<Module<T> as Currencies<T::AccountId>>::mint(*currency_id, account_id, *initial_balance).is_ok()); }); }) } } decl_event!( pub enum Event<T> where AccountId = <T as frame_system::Config>::AccountId, Balance = <T as Config>::Balance, CurrencyId = <T as Config>::CurrencyId, CurrencyDetails = CurrencyDetails<<T as frame_system::Config>::AccountId>, { /// A new currency has been created. \[currency id, details\] CurrencyCreated(CurrencyId, CurrencyDetails), /// Some units of currency were issued. \[currency_id, dest, amount\] CurrencyMinted(CurrencyId, AccountId, Balance), /// Some units of currency were destroyed. \[currency_id, source, amount\] CurrencyBurned(CurrencyId, AccountId, Balance), /// Some details about a currency were changed. \[currency_id, details\] CurrencyDetailsChanged(CurrencyId, CurrencyDetails), /// Some units of currency were transferred. \[currency_id, source, dest, amount\] CurrencyTransferred(CurrencyId, AccountId, AccountId, Balance), } ); decl_error! { pub enum Error for Module<T: Config> { /// This operation will cause total issuance to overflow for the given currency TotalIssuanceOverflow, /// This operation will cause total issuance to underflow for the given currency TotalIssuanceUnderflow, /// This operation will cause the balance of an account to overflow for the /// given currency BalanceOverflow, /// There are not enough coins inside the balance of the user to perform the action BalanceTooLow, /// Some of the balance you are trying to withdraw is locked. BalanceLockTriggered, /// The currency ID is already used by another currency CurrencyAlreadyExists, /// This owner(s) of this currency have disabled transfers UnTransferableCurrency, } } decl_module! { pub struct Module<T: Config> for enum Call where origin: T::Origin { type Error = Error<T>; fn deposit_event() = default; /// Creates a new currency with 0 units, to issue units to people one would have to call /// `issue`. This will register the caller of this dispatchable as the owner of the currency /// so they can issue or burn units. This will produce an error if `currency_id` is already /// used by another currency. Use `transferable` to determine if the created asset can be /// transferred between accounts. If not, the only way to move it would be to either be root /// or burn then mint the tokens again. /// /// NOTE: by default, everybody can create new currencies, if it is not wanted you can use the /// `bylaws` pallet to restrict access to this dispatchable. #[weight = T::WeightInfo::create()] pub fn create(origin, currency_id: T::CurrencyId, transferable: bool) { let who = RoleManagerOf::<T>::ensure_has_role(origin, RoleBuilderOf::<T>::create_currencies())?; Self::maybe_create_zero_issuance(currency_id)?; let details = CurrencyDetails { owner: who.clone(), transferable, }; Self::set_currency_acl(currency_id, details.clone(), None); Self::deposit_event(RawEvent::CurrencyCreated(currency_id, details)); } /// Issue some units of the currency identified by `currency_id` and credit them to `dest`. /// Can only be called by the owner of the currency. #[weight = T::WeightInfo::mint()] pub fn mint(origin, currency_id: T::CurrencyId, dest: <T::Lookup as StaticLookup>::Source, amount: T::Balance) { RoleManagerOf::<T>::ensure_has_role(origin, RoleBuilderOf::<T>::manage_currency(currency_id))?; let to = T::Lookup::lookup(dest)?; <Self as Currencies<T::AccountId>>::mint(currency_id, &to, amount)?; Self::deposit_event(RawEvent::CurrencyMinted(currency_id, to, amount)); } /// Destroy some units of the currency identified by `currency_id` from `from`. /// Can only be called by the owner of the currency. #[weight = T::WeightInfo::burn()] pub fn burn(origin, currency_id: T::CurrencyId, from: <T::Lookup as StaticLookup>::Source, amount: T::Balance) { RoleManagerOf::<T>::ensure_has_role(origin, RoleBuilderOf::<T>::manage_currency(currency_id))?; let source = T::Lookup::lookup(from)?; <Self as Currencies<T::AccountId>>::burn(currency_id, &source, amount)?; } /// Update details about the currency identified by `currency_id`. For instance, this /// can be used to change the owner of the currency. Can only be called by the owner. /// /// **NOTE**: this will remove ownership / management access from the caller for the given /// currency if a new owner is specified. However, if other accounts have been granted /// management access to the same currency (for instance through a root action) this will /// not change it. #[weight = T::WeightInfo::update_details()] pub fn update_details(origin, currency_id: T::CurrencyId, details: CurrencyDetails<T::AccountId>) { let who = RoleManagerOf::<T>::ensure_has_role(origin, RoleBuilderOf::<T>::manage_currency(currency_id))?; Self::set_currency_acl(currency_id, details.clone(), Some(who)); Self::deposit_event(RawEvent::CurrencyDetailsChanged(currency_id, details)); } /// Transfer `amount` units of the currency identified by `currency_id` from the origin's /// account to the balance of `dest`. #[weight = T::WeightInfo::transfer()] pub fn transfer(origin, currency_id: T::CurrencyId, dest: <T::Lookup as StaticLookup>::Source, amount: T::Balance) { let from = ensure_signed(origin)?; let to = T::Lookup::lookup(dest)?; <Self as Currencies<T::AccountId>>::transfer(currency_id, &from, &to, amount)?; } } } impl<T: Config> Module<T> { /// Return the `AccountCurrencyData` for the `who` and `currency_id`. fn get_currency_account( currency_id: T::CurrencyId, who: &T::AccountId, ) -> AccountCurrencyData<T::Balance> { Balances::<T>::get(who, currency_id) } /// Register the ACL roles accordingly for a given currency. fn set_currency_acl( currency_id: T::CurrencyId, details: CurrencyDetails<T::AccountId>, maybe_no_longer_owner: Option<T::AccountId>, ) { if let Some(previous_owner) = maybe_no_longer_owner { if details.owner != previous_owner { // Typical error at this stage would be that the role is not granted to // `previous_owner`. There are some edge cases where this is possible, for // instance if a root user revoked is calling the functions. Thus, we prefer // not to fail and drop the result. drop(RoleManagerOf::<T>::revoke_role( Some(&previous_owner), RoleBuilderOf::<T>::manage_currency(currency_id), )); } } // We drop the results as it is possible that the current owner itself is calling the update // function or that none of the acl parameters changed. // Since it is cheaper to fail on `grant_role` rather than calling `has_role` before we drop. drop(RoleManagerOf::<T>::grant_role( Some(&details.owner), RoleBuilderOf::<T>::manage_currency(currency_id), )); if details.transferable { drop(RoleManagerOf::<T>::grant_role( None, RoleBuilderOf::<T>::transfer_currency(currency_id), )); } else { drop(RoleManagerOf::<T>::revoke_role( None, RoleBuilderOf::<T>::transfer_currency(currency_id), )); } } /// Just set the total issuance to 0. This will write to the storage. Use only when /// creating new currencies. /// If this detect that the issuance field is already set this will not reset it to 0. fn maybe_create_zero_issuance(currency_id: T::CurrencyId) -> DispatchResult { // In order to prevent duplication of currencies (aka somebody creating an already existing // currency) have to mark the currency as created and check for this flag on the next calls // to `create`. // The naïve solution would be to add a new storage field to log this. Another one would be // to use the `TotalIssuances` storage and create a storage entry at 0 if a currency is created. // Future calls to `create` will realize that there is a value there and thus prevent the call // from doing any changes. This way, we don't have to use yet another storage map. TotalIssuances::<T>::try_mutate_exists(currency_id, |issuance| match issuance { Some(_) => Err(Error::<T>::CurrencyAlreadyExists.into()), None => { *issuance = Some(Zero::zero()); Ok(()) } }) } }