use crate::{mutations::Mutation, Config, Locks, Module, RawEvent};
use frame_support::{
traits::{BalanceStatus, LockIdentifier},
IterableStorageDoubleMap, StorageDoubleMap,
};
use governance_os_support::traits::{Currencies, LockableCurrencies, ReservableCurrencies};
use sp_runtime::{
traits::{Saturating, Zero},
DispatchError, DispatchResult,
};
impl<T: Config> Currencies<T::AccountId> for Module<T> {
type CurrencyId = T::CurrencyId;
type Balance = T::Balance;
fn total_issuance(currency_id: Self::CurrencyId) -> Self::Balance {
Self::total_issuances(currency_id)
}
fn burn(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.sub_free_balance(who, amount)?;
mutation.apply()?;
Self::deposit_event(RawEvent::CurrencyBurned(currency_id, who.clone(), amount));
Ok(())
}
fn mint(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.add_free_balance(who, amount)?;
mutation.apply()?;
Ok(())
}
fn free_balance(currency_id: Self::CurrencyId, who: &T::AccountId) -> Self::Balance {
Self::get_currency_account(currency_id, who).free
}
fn total_balance(currency_id: Self::CurrencyId, who: &T::AccountId) -> Self::Balance {
Self::get_currency_account(currency_id, who).total()
}
fn ensure_can_withdraw(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.ensure_must_be_transferable_for(who)?;
mutation.sub_free_balance(who, amount)?;
Ok(())
}
fn transfer(
currency_id: Self::CurrencyId,
source: &T::AccountId,
dest: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.ensure_must_be_transferable_for(source)?;
mutation.sub_free_balance(source, amount)?;
mutation.add_free_balance(dest, amount)?;
mutation.apply()?;
Self::deposit_event(RawEvent::CurrencyTransferred(
currency_id,
source.clone(),
dest.clone(),
amount,
));
Ok(())
}
}
impl<T: Config> ReservableCurrencies<T::AccountId> for Module<T> {
fn can_reserve(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> bool {
Self::ensure_can_withdraw(currency_id, who, amount).is_ok()
}
fn slash_reserved(
currency_id: Self::CurrencyId,
who: &T::AccountId,
value: Self::Balance,
) -> Self::Balance {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
let actual = mutation.sub_up_to_reserved_balance(who, value);
mutation
.apply()
.expect("we assume the coins were created and added to the total issuance previously");
value - actual
}
fn reserved_balance(currency_id: Self::CurrencyId, who: &T::AccountId) -> Self::Balance {
Self::get_currency_account(currency_id, who).reserved
}
fn reserve(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.sub_free_balance(who, amount)?;
mutation.add_reserved_balance(who, amount)?;
mutation.apply()?;
Ok(())
}
fn unreserve(
currency_id: Self::CurrencyId,
who: &T::AccountId,
amount: Self::Balance,
) -> Self::Balance {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
let unreserved = mutation.sub_up_to_reserved_balance(who, amount);
mutation
.add_free_balance(who, unreserved)
.expect("we are merely reallocating balances");
mutation
.apply()
.expect("we are not modifiying the total issuance and thus do not expect any errors");
amount - unreserved
}
fn repatriate_reserved(
currency_id: Self::CurrencyId,
slashed: &T::AccountId,
beneficiary: &T::AccountId,
value: Self::Balance,
status: BalanceStatus,
) -> Result<Self::Balance, DispatchError> {
if slashed == beneficiary {
return match status {
BalanceStatus::Free => Ok(Self::unreserve(currency_id, slashed, value)),
BalanceStatus::Reserved => {
Ok(value.saturating_sub(Self::reserved_balance(currency_id, slashed)))
}
};
}
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
let actual = mutation.sub_up_to_reserved_balance(slashed, value);
match status {
BalanceStatus::Free => {
mutation.add_free_balance(beneficiary, actual)?;
}
BalanceStatus::Reserved => {
mutation.add_reserved_balance(beneficiary, actual)?;
}
};
mutation.apply()?;
Ok(value - actual)
}
}
impl<T: Config> Module<T> {
fn conditionally_write_lock<Cond: FnOnce(T::Balance) -> bool>(
currency_id: T::CurrencyId,
lock_id: LockIdentifier,
who: &T::AccountId,
amount: T::Balance,
condition: Cond,
) -> DispatchResult {
Locks::<T>::try_mutate_exists(
(who, currency_id),
lock_id,
|maybe_existing_lock| -> DispatchResult {
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
if maybe_existing_lock.is_none() {
if mutation.frozen(who) < amount {
mutation.add_frozen(who, amount)?;
}
if frame_system::Pallet::<T>::inc_consumers(who).is_err() {
log::warn!(
target: "runtime::tokens",
"Warning: Attempt to introduce lock consumer reference, yet no providers. \
This is unexpected but should be safe."
);
}
} else {
let existing_lock = maybe_existing_lock
.take()
.expect("we just did a is_none check");
if !condition(existing_lock) {
return Ok(());
}
if mutation.frozen(who).saturating_sub(existing_lock) < amount {
mutation.sub_frozen(who, existing_lock)?;
mutation.add_frozen(who, amount)?;
}
}
mutation.forget_issuance_changes();
mutation.apply()?;
*maybe_existing_lock = Some(amount);
Ok(())
},
)
}
}
impl<T: Config> LockableCurrencies<T::AccountId> for Module<T> {
fn locked_balance(currency_id: Self::CurrencyId, who: &T::AccountId) -> Self::Balance {
Self::get_currency_account(currency_id, who).frozen
}
fn set_lock(
currency_id: Self::CurrencyId,
lock_id: LockIdentifier,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
Self::conditionally_write_lock(currency_id, lock_id, who, amount, |_| true)
}
fn extend_lock(
currency_id: Self::CurrencyId,
lock_id: LockIdentifier,
who: &T::AccountId,
amount: Self::Balance,
) -> DispatchResult {
Self::conditionally_write_lock(currency_id, lock_id, who, amount, |existing_lock| {
existing_lock < amount
})
}
fn remove_lock(
currency_id: Self::CurrencyId,
lock_id: LockIdentifier,
who: &T::AccountId,
) -> DispatchResult {
Locks::<T>::remove((who, currency_id), lock_id);
frame_system::Pallet::<T>::dec_consumers(who);
let highest_lock_value = Locks::<T>::iter_prefix((who, currency_id)).fold(
Zero::zero(),
|acc, (_lock_id, lock_val)| {
if acc < lock_val {
lock_val
} else {
acc
}
},
);
let mut mutation = Mutation::<T>::new_for_currency(currency_id);
mutation.overwrite_frozen_balance(who, highest_lock_value);
mutation.apply()?;
Ok(())
}
}