use super::*;
use crate::{Event, NegativeImbalance};
use frame_support::{
traits::{
BalanceStatus::{Free, Reserved},
Currency,
ExistenceRequirement::{self, AllowDeath, KeepAlive},
InspectLockableCurrency, LockIdentifier, LockableCurrency, NamedReservableCurrency,
ReservableCurrency, WithdrawReasons,
},
StorageNoopGuard,
};
use frame_system::Event as SysEvent;
use sp_runtime::traits::DispatchTransaction;
const ID_1: LockIdentifier = *b"1 ";
const ID_2: LockIdentifier = *b"2 ";
pub const CALL: &<Test as frame_system::Config>::RuntimeCall =
&RuntimeCall::Balances(crate::Call::transfer_allow_death { dest: 0, value: 0 });
#[test]
fn ed_should_work() {
ExtBuilder::default().existential_deposit(1).build_and_execute_with(|| {
assert_ok!(Balances::force_set_balance(RuntimeOrigin::root(), 1, 1000));
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &10, 1000, KeepAlive),
TokenError::NotExpendable
);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &10, 1000, AllowDeath));
});
}
#[test]
fn set_lock_with_amount_zero_removes_lock() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::all());
Balances::set_lock(ID_1, &1, 0, WithdrawReasons::all());
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn set_lock_with_withdraw_reasons_empty_removes_lock() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::all());
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::empty());
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn basic_locking_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
assert_eq!(Balances::free_balance(1), 10);
Balances::set_lock(ID_1, &1, 9, WithdrawReasons::all());
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 5, AllowDeath),
TokenError::Frozen
);
});
}
#[test]
fn inspect_lock_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 10, WithdrawReasons::all());
Balances::set_lock(ID_2, &1, 10, WithdrawReasons::all());
Balances::set_lock(ID_1, &2, 20, WithdrawReasons::all());
assert_eq!(<Balances as InspectLockableCurrency<_>>::balance_locked(ID_1, &1), 10);
assert_eq!(<Balances as InspectLockableCurrency<_>>::balance_locked(ID_2, &1), 10);
assert_eq!(<Balances as InspectLockableCurrency<_>>::balance_locked(ID_1, &2), 20);
assert_eq!(<Balances as InspectLockableCurrency<_>>::balance_locked(ID_2, &2), 0);
assert_eq!(<Balances as InspectLockableCurrency<_>>::balance_locked(ID_1, &3), 0);
})
}
#[test]
fn account_should_be_reaped() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
assert_eq!(Balances::free_balance(1), 10);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 10, AllowDeath));
assert_eq!(System::providers(&1), 0);
assert_eq!(System::consumers(&1), 0);
assert!(!frame_system::Account::<Test>::contains_key(&1));
});
}
#[test]
fn reap_failed_due_to_provider_and_consumer() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
assert_ok!(System::inc_consumers(&1));
assert!(!System::can_dec_provider(&1));
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 10, AllowDeath),
TokenError::Frozen
);
assert!(System::account_exists(&1));
assert_eq!(Balances::free_balance(1), 10);
assert_eq!(System::inc_providers(&1), frame_system::IncRefStatus::Existed);
assert_eq!(System::providers(&1), 2);
assert!(System::can_dec_provider(&1));
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 10, AllowDeath));
assert_eq!(System::providers(&1), 1);
assert!(System::account_exists(&1));
assert_eq!(Balances::free_balance(1), 0);
});
}
#[test]
fn partial_locking_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn lock_removal_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
Balances::remove_lock(ID_1, &1);
assert_eq!(System::consumers(&1), 0);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn lock_replacement_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn double_locking_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
Balances::set_lock(ID_2, &1, 5, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn combination_locking_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
assert_eq!(System::consumers(&1), 0);
Balances::set_lock(ID_1, &1, u64::MAX, WithdrawReasons::empty());
assert_eq!(System::consumers(&1), 0);
Balances::set_lock(ID_2, &1, 0, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 0);
assert_ok!(<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath));
});
}
#[test]
fn lock_value_extension_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
Balances::extend_lock(ID_1, &1, 2, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
Balances::extend_lock(ID_1, &1, 8, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 3, AllowDeath),
TokenError::Frozen
);
});
}
#[test]
fn lock_should_work_reserve() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
pallet_transaction_payment::NextFeeMultiplier::<Test>::put(
Multiplier::saturating_from_integer(1),
);
Balances::set_lock(ID_1, &1, 10, WithdrawReasons::RESERVE);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath),
TokenError::Frozen
);
assert_ok!(Balances::reserve(&1, 9));
assert_noop!(Balances::reserve(&1, 1), DispatchError::ConsumerRemaining,);
assert!(ChargeTransactionPayment::<Test>::validate_and_prepare(
ChargeTransactionPayment::from(1),
Some(1).into(),
CALL,
&info_from_weight(Weight::from_parts(1, 0)),
1,
0,
)
.is_err());
assert!(ChargeTransactionPayment::<Test>::validate_and_prepare(
ChargeTransactionPayment::from(0),
Some(1).into(),
CALL,
&info_from_weight(Weight::from_parts(1, 0)),
1,
0,
)
.is_err());
});
}
#[test]
fn reserve_should_work_for_frozen_balance() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
let account = Balances::account(&1);
assert_eq!(account.free, 10);
assert_eq!(account.frozen, 0);
assert_eq!(account.reserved, 0);
Balances::set_lock(ID_1, &1, 9, WithdrawReasons::RESERVE);
let account = Balances::account(&1);
assert_eq!(account.free, 10);
assert_eq!(account.frozen, 9);
assert_eq!(account.reserved, 0);
assert_ok!(Balances::reserve(&1, 5));
let account = Balances::account(&1);
assert_eq!(account.free, 5);
assert_eq!(account.frozen, 9);
assert_eq!(account.reserved, 5);
assert_ok!(Balances::reserve(&1, 4));
let account = Balances::account(&1);
assert_eq!(account.free, 1);
assert_eq!(account.frozen, 9);
assert_eq!(account.reserved, 9);
let usable_balance = Balances::usable_balance(&1);
assert_eq!(usable_balance, 0);
});
}
#[test]
fn lock_should_work_tx_fee() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 10, WithdrawReasons::TRANSACTION_PAYMENT);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 1, AllowDeath),
TokenError::Frozen
);
assert_ok!(Balances::reserve(&1, 9));
assert_noop!(Balances::reserve(&1, 1), DispatchError::ConsumerRemaining,);
assert!(ChargeTransactionPayment::<Test>::validate_and_prepare(
ChargeTransactionPayment::from(1),
Some(1).into(),
CALL,
&info_from_weight(Weight::from_parts(1, 0)),
1,
0,
)
.is_err());
assert!(ChargeTransactionPayment::<Test>::validate_and_prepare(
ChargeTransactionPayment::from(0),
Some(1).into(),
CALL,
&info_from_weight(Weight::from_parts(1, 0)),
1,
0,
)
.is_err());
});
}
#[test]
fn lock_block_number_extension_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 10, WithdrawReasons::all());
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
Balances::extend_lock(ID_1, &1, 10, WithdrawReasons::all());
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
System::set_block_number(2);
Balances::extend_lock(ID_1, &1, 10, WithdrawReasons::all());
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 3, AllowDeath),
TokenError::Frozen
);
});
}
#[test]
fn lock_reasons_extension_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
Balances::set_lock(ID_1, &1, 10, WithdrawReasons::TRANSFER);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
Balances::extend_lock(ID_1, &1, 10, WithdrawReasons::empty());
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
Balances::extend_lock(ID_1, &1, 10, WithdrawReasons::RESERVE);
assert_noop!(
<Balances as Currency<_>>::transfer(&1, &2, 6, AllowDeath),
TokenError::Frozen
);
});
}
#[test]
fn reserved_balance_should_prevent_reclaim_count() {
ExtBuilder::default()
.existential_deposit(256 * 1)
.monied(true)
.build_and_execute_with(|| {
System::inc_account_nonce(&2);
assert_eq!(Balances::total_balance(&2), 256 * 20);
assert_eq!(System::providers(&2), 1);
System::inc_providers(&2);
assert_eq!(System::providers(&2), 2);
assert_ok!(Balances::reserve(&2, 256 * 19 + 1)); assert_eq!(System::providers(&2), 1);
assert_eq!(Balances::free_balance(2), 255); assert_eq!(Balances::total_balance(&2), 256 * 20); assert_eq!(System::account_nonce(&2), 1);
assert_ok!(Balances::transfer_allow_death(Some(4).into(), 5, 256 * 1 + 0x69));
assert_eq!(Balances::total_balance(&5), 256 * 1 + 0x69);
assert!(Balances::slash_reserved(&2, 256 * 19 + 1).1.is_zero());
assert_ok!(System::dec_providers(&2));
assert_eq!(System::providers(&2), 0);
assert_eq!(System::account_nonce(&2), 0); assert_eq!(Balances::total_balance(&2), 0);
assert_ok!(Balances::transfer_allow_death(Some(4).into(), 6, 256 * 1 + 0x69));
assert_eq!(Balances::total_balance(&6), 256 * 1 + 0x69);
});
}
#[test]
fn reward_should_work() {
ExtBuilder::default().monied(true).build_and_execute_with(|| {
assert_eq!(Balances::total_balance(&1), 10);
assert_ok!(Balances::deposit_into_existing(&1, 10).map(drop));
assert_eq!(
events(),
[
RuntimeEvent::Balances(crate::Event::Deposit { who: 1, amount: 10 }),
RuntimeEvent::Balances(crate::Event::Issued { amount: 10 }),
]
);
assert_eq!(Balances::total_balance(&1), 20);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 120);
});
}
#[test]
fn balance_works() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 42);
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Deposit {
who: 1,
amount: 42,
}));
assert_eq!(Balances::free_balance(1), 42);
assert_eq!(Balances::reserved_balance(1), 0);
assert_eq!(Balances::total_balance(&1), 42);
assert_eq!(Balances::free_balance(2), 0);
assert_eq!(Balances::reserved_balance(2), 0);
assert_eq!(Balances::total_balance(&2), 0);
});
}
#[test]
fn reserving_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
assert_eq!(Balances::total_balance(&1), 111);
assert_eq!(Balances::free_balance(1), 111);
assert_eq!(Balances::reserved_balance(1), 0);
assert_ok!(Balances::reserve(&1, 69));
assert_eq!(Balances::total_balance(&1), 111);
assert_eq!(Balances::free_balance(1), 42);
assert_eq!(Balances::reserved_balance(1), 69);
});
}
#[test]
fn deducting_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
assert_ok!(Balances::reserve(&1, 69));
assert_eq!(Balances::free_balance(1), 42);
});
}
#[test]
fn refunding_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 42);
assert_ok!(Balances::mutate_account(&1, false, |a| a.reserved = 69));
Balances::unreserve(&1, 69);
assert_eq!(Balances::free_balance(1), 111);
assert_eq!(Balances::reserved_balance(1), 0);
});
}
#[test]
fn slashing_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 112);
assert_ok!(Balances::reserve(&1, 69));
assert!(Balances::slash(&1, 42).1.is_zero());
assert_eq!(Balances::free_balance(1), 1);
assert_eq!(Balances::reserved_balance(1), 69);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 70);
});
}
#[test]
fn withdrawing_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&2, 111);
let _ =
Balances::withdraw(&2, 11, WithdrawReasons::TRANSFER, ExistenceRequirement::KeepAlive);
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Withdraw {
who: 2,
amount: 11,
}));
assert_eq!(Balances::free_balance(2), 100);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 100);
});
}
#[test]
fn withdrawing_balance_should_fail_when_not_expendable() {
ExtBuilder::default().build_and_execute_with(|| {
ExistentialDeposit::set(10);
let _ = Balances::deposit_creating(&2, 20);
assert_ok!(Balances::reserve(&2, 5));
assert_noop!(
Balances::withdraw(&2, 6, WithdrawReasons::TRANSFER, ExistenceRequirement::KeepAlive),
Error::<Test>::Expendability,
);
assert_ok!(Balances::withdraw(
&2,
5,
WithdrawReasons::TRANSFER,
ExistenceRequirement::KeepAlive
),);
});
}
#[test]
fn slashing_incomplete_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 42);
assert_ok!(Balances::reserve(&1, 21));
assert_eq!(Balances::slash(&1, 69).1, 49);
assert_eq!(Balances::free_balance(1), 1);
assert_eq!(Balances::reserved_balance(1), 21);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 22);
});
}
#[test]
fn unreserving_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
assert_ok!(Balances::reserve(&1, 110));
Balances::unreserve(&1, 41);
assert_eq!(Balances::reserved_balance(1), 69);
assert_eq!(Balances::free_balance(1), 42);
});
}
#[test]
fn slashing_reserved_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 112);
assert_ok!(Balances::reserve(&1, 111));
assert_eq!(Balances::slash_reserved(&1, 42).1, 0);
assert_eq!(Balances::reserved_balance(1), 69);
assert_eq!(Balances::free_balance(1), 1);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 70);
});
}
#[test]
fn slashing_incomplete_reserved_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
assert_ok!(Balances::reserve(&1, 42));
assert_eq!(Balances::slash_reserved(&1, 69).1, 27);
assert_eq!(Balances::free_balance(1), 69);
assert_eq!(Balances::reserved_balance(1), 0);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 69);
});
}
#[test]
fn repatriating_reserved_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let _ = Balances::deposit_creating(&2, 1);
assert_ok!(Balances::reserve(&1, 110));
assert_ok!(Balances::repatriate_reserved(&1, &2, 41, Free), 0);
System::assert_last_event(RuntimeEvent::Balances(crate::Event::ReserveRepatriated {
from: 1,
to: 2,
amount: 41,
destination_status: Free,
}));
assert_eq!(Balances::reserved_balance(1), 69);
assert_eq!(Balances::free_balance(1), 1);
assert_eq!(Balances::reserved_balance(2), 0);
assert_eq!(Balances::free_balance(2), 42);
});
}
#[test]
fn transferring_reserved_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let _ = Balances::deposit_creating(&2, 1);
assert_ok!(Balances::reserve(&1, 110));
assert_ok!(Balances::repatriate_reserved(&1, &2, 41, Reserved), 0);
assert_eq!(Balances::reserved_balance(1), 69);
assert_eq!(Balances::free_balance(1), 1);
assert_eq!(Balances::reserved_balance(2), 41);
assert_eq!(Balances::free_balance(2), 1);
});
}
#[test]
fn transferring_reserved_balance_to_yourself_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 110);
assert_ok!(Balances::reserve(&1, 50));
assert_ok!(Balances::repatriate_reserved(&1, &1, 50, Free), 0);
assert_eq!(Balances::free_balance(1), 110);
assert_eq!(Balances::reserved_balance(1), 0);
assert_ok!(Balances::reserve(&1, 50));
assert_ok!(Balances::repatriate_reserved(&1, &1, 60, Free), 10);
assert_eq!(Balances::free_balance(1), 110);
assert_eq!(Balances::reserved_balance(1), 0);
});
}
#[test]
fn transferring_reserved_balance_to_nonexistent_should_fail() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
assert_ok!(Balances::reserve(&1, 110));
assert_noop!(
Balances::repatriate_reserved(&1, &2, 42, Free),
Error::<Test, _>::DeadAccount
);
});
}
#[test]
fn transferring_incomplete_reserved_balance_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 110);
let _ = Balances::deposit_creating(&2, 1);
assert_ok!(Balances::reserve(&1, 41));
assert_ok!(Balances::repatriate_reserved(&1, &2, 69, Free), 28);
assert_eq!(Balances::reserved_balance(1), 0);
assert_eq!(Balances::free_balance(1), 69);
assert_eq!(Balances::reserved_balance(2), 0);
assert_eq!(Balances::free_balance(2), 42);
});
}
#[test]
fn transferring_too_high_value_should_not_panic() {
ExtBuilder::default().build_and_execute_with(|| {
Balances::make_free_balance_be(&1, u64::MAX);
Balances::make_free_balance_be(&2, 1);
assert_err!(
<Balances as Currency<_>>::transfer(&1, &2, u64::MAX, AllowDeath),
ArithmeticError::Overflow,
);
assert_eq!(Balances::free_balance(1), u64::MAX);
assert_eq!(Balances::free_balance(2), 1);
});
}
#[test]
fn account_create_on_free_too_low_with_other() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 100);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 100);
let _ = Balances::deposit_creating(&2, 50);
assert_eq!(Balances::free_balance(2), 0);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 100);
})
}
#[test]
fn account_create_on_free_too_low() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
let _ = Balances::deposit_creating(&2, 50);
assert_eq!(Balances::free_balance(2), 0);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 0);
})
}
#[test]
fn account_removal_on_free_too_low() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 0);
let _ = Balances::deposit_creating(&1, 110);
let _ = Balances::deposit_creating(&2, 110);
assert_eq!(Balances::free_balance(1), 110);
assert_eq!(Balances::free_balance(2), 110);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 220);
assert_ok!(Balances::transfer_allow_death(Some(1).into(), 2, 20));
assert_eq!(Balances::free_balance(1), 0);
assert_eq!(Balances::free_balance(2), 130);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 130);
});
}
#[test]
fn burn_must_work() {
ExtBuilder::default().monied(true).build_and_execute_with(|| {
let init_total_issuance = pallet_balances::TotalIssuance::<Test>::get();
let imbalance = <Balances as Currency<_>>::burn(10);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), init_total_issuance - 10);
drop(imbalance);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), init_total_issuance);
});
}
#[test]
#[should_panic = "the balance of any account should always be at least the existential deposit."]
fn cannot_set_genesis_value_below_ed() {
EXISTENTIAL_DEPOSIT.with(|v| *v.borrow_mut() = 11);
let mut t = frame_system::GenesisConfig::<Test>::default().build_storage().unwrap();
let _ = crate::GenesisConfig::<Test> { balances: vec![(1, 10)], ..Default::default() }
.assimilate_storage(&mut t)
.unwrap();
}
#[test]
#[should_panic = "duplicate balances in genesis."]
fn cannot_set_genesis_value_twice() {
let mut t = frame_system::GenesisConfig::<Test>::default().build_storage().unwrap();
let _ = crate::GenesisConfig::<Test> {
balances: vec![(1, 10), (2, 20), (1, 15)],
..Default::default()
}
.assimilate_storage(&mut t)
.unwrap();
}
#[test]
fn existential_deposit_respected_when_reserving() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_ok!(Balances::force_set_balance(RawOrigin::Root.into(), 1, 101));
assert_eq!(Balances::free_balance(1), 101);
assert_eq!(Balances::reserved_balance(1), 0);
assert_ok!(Balances::reserve(&1, 1));
assert_eq!(Balances::free_balance(1), 100);
assert_eq!(Balances::reserved_balance(1), 1);
assert_noop!(Balances::reserve(&1, 1), DispatchError::ConsumerRemaining);
});
}
#[test]
fn slash_fails_when_account_needed() {
ExtBuilder::default().existential_deposit(50).build_and_execute_with(|| {
assert_ok!(Balances::force_set_balance(RawOrigin::Root.into(), 1, 52));
assert_ok!(Balances::reserve(&1, 1));
assert_eq!(Balances::free_balance(1), 51);
assert_eq!(Balances::reserved_balance(1), 1);
let res = Balances::slash(&1, 1);
assert_eq!(res, (NegativeImbalance::new(1), 0));
assert_eq!(Balances::free_balance(1), 50);
assert_eq!(Balances::reserved_balance(1), 1);
let res = Balances::slash(&1, 1);
assert_eq!(res, (NegativeImbalance::new(0), 1));
assert_eq!(Balances::free_balance(1), 50);
assert_eq!(Balances::reserved_balance(1), 1);
});
}
#[test]
fn account_deleted_when_just_dust() {
ExtBuilder::default().existential_deposit(50).build_and_execute_with(|| {
assert_ok!(Balances::force_set_balance(RawOrigin::Root.into(), 1, 50));
assert_eq!(Balances::free_balance(1), 50);
let res = Balances::slash(&1, 1);
assert_eq!(res, (NegativeImbalance::new(1), 0));
assert_eq!(Balances::free_balance(1), 0);
});
}
#[test]
fn emit_events_with_reserve_and_unreserve() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 100);
System::set_block_number(2);
assert_ok!(Balances::reserve(&1, 10));
System::assert_last_event(RuntimeEvent::Balances(crate::Event::Reserved {
who: 1,
amount: 10,
}));
System::set_block_number(3);
assert!(Balances::unreserve(&1, 5).is_zero());
System::assert_last_event(RuntimeEvent::Balances(crate::Event::Unreserved {
who: 1,
amount: 5,
}));
System::set_block_number(4);
assert_eq!(Balances::unreserve(&1, 6), 1);
System::assert_last_event(RuntimeEvent::Balances(crate::Event::Unreserved {
who: 1,
amount: 5,
}));
});
}
#[test]
fn emit_events_with_changing_locks() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 100);
System::reset_events();
Balances::set_lock(*b"LOCK_000", &1, 10, WithdrawReasons::TRANSFER);
assert_eq!(events(), [RuntimeEvent::Balances(crate::Event::Locked { who: 1, amount: 10 })]);
Balances::set_lock(*b"LOCK_000", &1, 15, WithdrawReasons::TRANSFER);
assert_eq!(events(), [RuntimeEvent::Balances(crate::Event::Locked { who: 1, amount: 5 })]);
Balances::set_lock(*b"LOCK_001", &1, 20, WithdrawReasons::TRANSACTION_PAYMENT);
assert_eq!(events(), [RuntimeEvent::Balances(crate::Event::Locked { who: 1, amount: 5 })]);
Balances::set_lock(*b"LOCK_000", &1, 17, WithdrawReasons::TRANSACTION_PAYMENT);
for event in events() {
match event {
RuntimeEvent::Balances(crate::Event::Locked { .. }) => {
assert!(false, "unexpected lock event")
},
RuntimeEvent::Balances(crate::Event::Unlocked { .. }) => {
assert!(false, "unexpected unlock event")
},
_ => continue,
}
}
Balances::set_lock(*b"LOCK_001", &1, 15, WithdrawReasons::TRANSFER);
assert_eq!(
events(),
[RuntimeEvent::Balances(crate::Event::Unlocked { who: 1, amount: 3 })]
);
Balances::remove_lock(*b"LOCK_000", &1);
assert_eq!(
events(),
[RuntimeEvent::Balances(crate::Event::Unlocked { who: 1, amount: 2 })]
);
Balances::remove_lock(*b"LOCK_001", &1);
assert_eq!(
events(),
[RuntimeEvent::Balances(crate::Event::Unlocked { who: 1, amount: 15 })]
);
});
}
#[test]
fn emit_events_with_existential_deposit() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_ok!(Balances::force_set_balance(RawOrigin::Root.into(), 1, 100));
assert_eq!(
events(),
[
RuntimeEvent::System(system::Event::NewAccount { account: 1 }),
RuntimeEvent::Balances(crate::Event::Endowed { account: 1, free_balance: 100 }),
RuntimeEvent::Balances(crate::Event::Issued { amount: 100 }),
RuntimeEvent::Balances(crate::Event::BalanceSet { who: 1, free: 100 }),
]
);
let res = Balances::slash(&1, 1);
assert_eq!(res, (NegativeImbalance::new(1), 0));
assert_eq!(
events(),
[
RuntimeEvent::System(system::Event::KilledAccount { account: 1 }),
RuntimeEvent::Balances(crate::Event::DustLost { account: 1, amount: 99 }),
RuntimeEvent::Balances(crate::Event::BurnedDebt { amount: 99 }),
RuntimeEvent::Balances(crate::Event::Slashed { who: 1, amount: 1 }),
RuntimeEvent::Balances(crate::Event::Rescinded { amount: 1 }),
]
);
});
}
#[test]
fn emit_events_with_no_existential_deposit_suicide() {
ExtBuilder::default().existential_deposit(1).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 100);
assert_eq!(
events(),
[
RuntimeEvent::Balances(crate::Event::BalanceSet { who: 1, free: 100 }),
RuntimeEvent::System(system::Event::NewAccount { account: 1 }),
RuntimeEvent::Balances(crate::Event::Endowed { account: 1, free_balance: 100 }),
RuntimeEvent::Balances(crate::Event::Issued { amount: 100 }),
]
);
let res = Balances::slash(&1, 100);
assert_eq!(res, (NegativeImbalance::new(100), 0));
assert_eq!(
events(),
[
RuntimeEvent::System(system::Event::KilledAccount { account: 1 }),
RuntimeEvent::Balances(crate::Event::Slashed { who: 1, amount: 100 }),
RuntimeEvent::Balances(crate::Event::Rescinded { amount: 100 }),
]
);
});
}
#[test]
fn slash_over_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_eq!(Balances::slash(&1, 1_300), (NegativeImbalance::new(1000), 300));
assert!(!System::account_exists(&1));
});
}
#[test]
fn slash_full_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_eq!(Balances::slash(&1, 1_000), (NegativeImbalance::new(1000), 0));
assert!(!System::account_exists(&1));
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Slashed {
who: 1,
amount: 1000,
}));
});
}
#[test]
fn slash_partial_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_eq!(Balances::slash(&1, 900), (NegativeImbalance::new(900), 0));
assert!(System::account_exists(&1));
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Slashed {
who: 1,
amount: 900,
}));
});
}
#[test]
fn slash_dusting_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_eq!(Balances::slash(&1, 950), (NegativeImbalance::new(950), 0));
assert!(!System::account_exists(&1));
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Slashed {
who: 1,
amount: 950,
}));
});
}
#[test]
fn slash_does_not_take_from_reserve() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(Balances::reserve(&1, 100));
assert_eq!(Balances::slash(&1, 900), (NegativeImbalance::new(800), 100));
assert_eq!(Balances::reserved_balance(&1), 100);
System::assert_has_event(RuntimeEvent::Balances(crate::Event::Slashed {
who: 1,
amount: 800,
}));
});
}
#[test]
fn slash_consumed_slash_full_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(System::inc_consumers(&1)); assert_eq!(Balances::slash(&1, 900), (NegativeImbalance::new(900), 0));
assert!(System::account_exists(&1));
});
}
#[test]
fn slash_consumed_slash_over_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(System::inc_consumers(&1)); assert_eq!(Balances::slash(&1, 1_000), (NegativeImbalance::new(900), 100));
assert!(System::account_exists(&1));
});
}
#[test]
fn slash_consumed_slash_partial_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(System::inc_consumers(&1)); assert_eq!(Balances::slash(&1, 800), (NegativeImbalance::new(800), 0));
assert!(System::account_exists(&1));
});
}
#[test]
fn slash_on_non_existent_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_eq!(Balances::slash(&12345, 1_300), (NegativeImbalance::new(0), 1300));
});
}
#[test]
fn slash_reserved_slash_partial_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(Balances::reserve(&1, 900));
assert_eq!(Balances::slash_reserved(&1, 800), (NegativeImbalance::new(800), 0));
assert_eq!(System::consumers(&1), 1);
assert_eq!(Balances::reserved_balance(&1), 100);
assert_eq!(Balances::free_balance(&1), 100);
});
}
#[test]
fn slash_reserved_slash_everything_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(Balances::reserve(&1, 900));
assert_eq!(System::consumers(&1), 1);
assert_eq!(Balances::slash_reserved(&1, 900), (NegativeImbalance::new(900), 0));
assert_eq!(System::consumers(&1), 0);
assert!(System::account_exists(&1));
});
}
#[test]
fn slash_reserved_overslash_does_not_touch_free_balance() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
Balances::make_free_balance_be(&1, 1_000);
assert_ok!(Balances::reserve(&1, 800));
assert_eq!(Balances::slash_reserved(&1, 900), (NegativeImbalance::new(800), 100));
assert_eq!(Balances::free_balance(&1), 200);
});
}
#[test]
fn slash_reserved_on_non_existent_works() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_eq!(Balances::slash_reserved(&12345, 1_300), (NegativeImbalance::new(0), 1300));
});
}
#[test]
fn operations_on_dead_account_should_not_change_state() {
ExtBuilder::default().existential_deposit(1).build_and_execute_with(|| {
assert!(!frame_system::Account::<Test>::contains_key(&1337));
assert_storage_noop!(assert_eq!(Balances::unreserve(&1337, 42), 42));
assert_noop!(Balances::reserve(&1337, 42), Error::<Test, _>::InsufficientBalance);
assert_storage_noop!(assert_eq!(Balances::slash_reserved(&1337, 42).1, 42));
assert_noop!(
Balances::repatriate_reserved(&1337, &1338, 42, Free),
Error::<Test, _>::DeadAccount
);
assert_storage_noop!(assert_eq!(Balances::slash(&1337, 42).1, 42));
});
}
#[test]
#[should_panic = "The existential deposit must be greater than zero!"]
#[cfg(not(feature = "insecure_zero_ed"))]
fn zero_ed_is_prohibited() {
use frame_support::traits::Hooks;
ExtBuilder::default().existential_deposit(0).build_and_execute_with(|| {
Balances::integrity_test();
});
}
#[test]
fn named_reserve_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let id_1 = TestId::Foo;
let id_2 = TestId::Bar;
let id_3 = TestId::Baz;
assert_noop!(
Balances::reserve_named(&id_1, &1, 112),
Error::<Test, _>::InsufficientBalance
);
assert_ok!(Balances::reserve_named(&id_1, &1, 12));
assert_eq!(Balances::reserved_balance(1), 12);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 12);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 0);
assert_ok!(Balances::reserve_named(&id_1, &1, 2));
assert_eq!(Balances::reserved_balance(1), 14);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 14);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 0);
assert_ok!(Balances::reserve_named(&id_2, &1, 23));
assert_eq!(Balances::reserved_balance(1), 37);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 14);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 23);
assert_ok!(Balances::reserve(&1, 34));
assert_eq!(Balances::reserved_balance(1), 71);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 14);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 23);
assert_eq!(Balances::total_balance(&1), 111);
assert_eq!(Balances::free_balance(1), 40);
assert_noop!(Balances::reserve_named(&id_3, &1, 2), Error::<Test, _>::TooManyReserves);
assert_eq!(Balances::unreserve_named(&id_1, &1, 10), 0);
assert_eq!(Balances::reserved_balance(1), 61);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 4);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 23);
assert_eq!(Balances::unreserve_named(&id_1, &1, 5), 1);
assert_eq!(Balances::reserved_balance(1), 57);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 23);
assert_eq!(Balances::unreserve_named(&id_2, &1, 3), 0);
assert_eq!(Balances::reserved_balance(1), 54);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 20);
assert_eq!(Balances::total_balance(&1), 111);
assert_eq!(Balances::free_balance(1), 57);
assert_ok!(Balances::reserve_named(&id_1, &1, 10));
assert_eq!(Balances::slash_reserved_named(&id_1, &1, 25).1, 15);
assert_eq!(Balances::reserved_balance(1), 54);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 20);
assert_eq!(Balances::total_balance(&1), 101);
assert_eq!(Balances::slash_reserved_named(&id_2, &1, 5).1, 0);
assert_eq!(Balances::reserved_balance(1), 49);
assert_eq!(Balances::reserved_balance_named(&id_1, &1), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 15);
assert_eq!(Balances::total_balance(&1), 96);
let _ = Balances::deposit_creating(&2, 100);
assert_eq!(Balances::repatriate_reserved_named(&id_2, &1, &2, 10, Reserved).unwrap(), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 5);
assert_eq!(Balances::reserved_balance_named(&id_2, &2), 10);
assert_eq!(Balances::reserved_balance(&2), 10);
assert_eq!(Balances::repatriate_reserved_named(&id_2, &2, &1, 11, Reserved).unwrap(), 1);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 15);
assert_eq!(Balances::reserved_balance_named(&id_2, &2), 0);
assert_eq!(Balances::reserved_balance(&2), 0);
assert_eq!(Balances::repatriate_reserved_named(&id_2, &1, &2, 10, Free).unwrap(), 0);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 5);
assert_eq!(Balances::reserved_balance_named(&id_2, &2), 0);
assert_eq!(Balances::free_balance(&2), 110);
assert_eq!(Balances::repatriate_reserved_named(&id_2, &1, &1, 10, Reserved).unwrap(), 5);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 5);
assert_eq!(Balances::free_balance(&1), 47);
assert_eq!(Balances::repatriate_reserved_named(&id_2, &1, &1, 15, Free).unwrap(), 10);
assert_eq!(Balances::reserved_balance_named(&id_2, &1), 0);
assert_eq!(Balances::free_balance(&1), 52);
});
}
#[test]
fn reserve_must_succeed_if_can_reserve_does() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 1);
let _ = Balances::deposit_creating(&2, 2);
let _ = Balances::deposit_creating(&3, 3);
assert!(Balances::can_reserve(&1, 1) == Balances::reserve(&1, 1).is_ok());
assert!(Balances::can_reserve(&2, 1) == Balances::reserve(&2, 1).is_ok());
Balances::set_lock(ID_1, &3, 2, WithdrawReasons::RESERVE);
assert_eq!(Balances::can_reserve(&3, 2), Balances::reserve(&3, 2).is_ok());
});
}
#[test]
fn reserved_named_to_yourself_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 110);
let id = TestId::Foo;
assert_ok!(Balances::reserve_named(&id, &1, 50));
assert_ok!(Balances::repatriate_reserved_named(&id, &1, &1, 50, Free), 0);
assert_eq!(Balances::free_balance(1), 110);
assert_eq!(Balances::reserved_balance_named(&id, &1), 0);
assert_ok!(Balances::reserve_named(&id, &1, 50));
assert_ok!(Balances::repatriate_reserved_named(&id, &1, &1, 60, Free), 10);
assert_eq!(Balances::free_balance(1), 110);
assert_eq!(Balances::reserved_balance_named(&id, &1), 0);
});
}
#[test]
fn ensure_reserved_named_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let id = TestId::Foo;
assert_ok!(Balances::ensure_reserved_named(&id, &1, 15));
assert_eq!(Balances::reserved_balance_named(&id, &1), 15);
assert_ok!(Balances::ensure_reserved_named(&id, &1, 10));
assert_eq!(Balances::reserved_balance_named(&id, &1), 10);
assert_ok!(Balances::ensure_reserved_named(&id, &1, 20));
assert_eq!(Balances::reserved_balance_named(&id, &1), 20);
});
}
#[test]
fn unreserve_all_named_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let id = TestId::Foo;
assert_ok!(Balances::reserve_named(&id, &1, 15));
assert_eq!(Balances::unreserve_all_named(&id, &1), 15);
assert_eq!(Balances::reserved_balance_named(&id, &1), 0);
assert_eq!(Balances::free_balance(&1), 111);
assert_eq!(Balances::unreserve_all_named(&id, &1), 0);
});
}
#[test]
fn slash_all_reserved_named_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let id = TestId::Foo;
assert_ok!(Balances::reserve_named(&id, &1, 15));
assert_eq!(Balances::slash_all_reserved_named(&id, &1).peek(), 15);
assert_eq!(Balances::reserved_balance_named(&id, &1), 0);
assert_eq!(Balances::free_balance(&1), 96);
assert_eq!(Balances::slash_all_reserved_named(&id, &1).peek(), 0);
});
}
#[test]
fn repatriate_all_reserved_named_should_work() {
ExtBuilder::default().build_and_execute_with(|| {
let _ = Balances::deposit_creating(&1, 111);
let _ = Balances::deposit_creating(&2, 10);
let _ = Balances::deposit_creating(&3, 10);
let id = TestId::Foo;
assert_ok!(Balances::reserve_named(&id, &1, 15));
assert_ok!(Balances::repatriate_all_reserved_named(&id, &1, &2, Reserved));
assert_eq!(Balances::reserved_balance_named(&id, &1), 0);
assert_eq!(Balances::reserved_balance_named(&id, &2), 15);
assert_ok!(Balances::repatriate_all_reserved_named(&id, &2, &3, Free));
assert_eq!(Balances::reserved_balance_named(&id, &2), 0);
assert_eq!(Balances::free_balance(&3), 25);
});
}
#[test]
fn freezing_and_locking_should_work() {
ExtBuilder::default()
.existential_deposit(1)
.monied(true)
.build_and_execute_with(|| {
assert_eq!(System::consumers(&1), 0);
assert_ok!(<Balances as fungible::MutateFreeze<_>>::set_freeze(&TestId::Foo, &1, 4));
assert_eq!(System::consumers(&1), 1);
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_eq!(System::consumers(&1), 1);
assert_eq!(get_test_account_data(1).frozen, 5);
assert_ok!(<Balances as fungible::MutateFreeze<_>>::set_freeze(&TestId::Foo, &1, 6));
assert_eq!(get_test_account_data(1).frozen, 6);
assert_ok!(<Balances as fungible::MutateFreeze<_>>::set_freeze(&TestId::Foo, &1, 4));
assert_eq!(get_test_account_data(1).frozen, 5);
Balances::set_lock(ID_1, &1, 3, WithdrawReasons::all());
assert_eq!(get_test_account_data(1).frozen, 4);
Balances::set_lock(ID_1, &1, 5, WithdrawReasons::all());
assert_eq!(get_test_account_data(1).frozen, 5);
Balances::remove_lock(ID_1, &1);
assert_eq!(get_test_account_data(1).frozen, 4);
assert_eq!(System::consumers(&1), 1);
assert_ok!(<Balances as fungible::MutateFreeze<_>>::set_freeze(&TestId::Foo, &1, 0));
assert_eq!(get_test_account_data(1).frozen, 0);
assert_eq!(System::consumers(&1), 0);
});
}
#[test]
fn self_transfer_noop() {
ExtBuilder::default().existential_deposit(100).build_and_execute_with(|| {
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 0);
let _ = Balances::deposit_creating(&1, 100);
assert_eq!(
events(),
[
Event::Deposit { who: 1, amount: 100 }.into(),
SysEvent::NewAccount { account: 1 }.into(),
Event::Endowed { account: 1, free_balance: 100 }.into(),
Event::Issued { amount: 100 }.into(),
]
);
assert_eq!(Balances::free_balance(1), 100);
assert_eq!(pallet_balances::TotalIssuance::<Test>::get(), 100);
let _g = StorageNoopGuard::new();
for i in 0..200 {
let r = Balances::transfer_allow_death(Some(1).into(), 1, i);
if i <= 100 {
assert_ok!(r);
} else {
assert!(r.is_err());
}
assert!(events().is_empty());
assert_eq!(Balances::free_balance(1), 100, "Balance unchanged by self transfer");
assert_eq!(
pallet_balances::TotalIssuance::<Test>::get(),
100,
"TI unchanged by self transfers"
);
}
});
}