use super::*;
use crate::{
asset,
session_rotation::{Eras, Rotator},
Pallet as Staking,
};
use alloc::collections::BTreeMap;
pub use frame_benchmarking::{
impl_benchmark_test_suite, v2::*, whitelist_account, whitelisted_caller, BenchmarkError,
};
use frame_election_provider_support::SortedListProvider;
use frame_support::{
assert_ok,
pallet_prelude::*,
storage::bounded_vec::BoundedVec,
traits::{fungible::Inspect, TryCollect},
};
use frame_system::RawOrigin;
use pallet_staking_async_rc_client as rc_client;
use sp_runtime::{
traits::{Bounded, One, StaticLookup, Zero},
Perbill, Percent, Saturating,
};
use sp_staking::currency_to_vote::CurrencyToVote;
use testing_utils::*;
const SEED: u32 = 0;
pub(crate) fn create_validator_with_nominators<T: Config>(
n: u32,
upper_bound: u32,
dead_controller: bool,
unique_controller: bool,
destination: RewardDestination<T::AccountId>,
) -> Result<(T::AccountId, Vec<(T::AccountId, T::AccountId)>, EraIndex), &'static str> {
clear_validators_and_nominators::<T>();
let mut points_total = 0;
let mut points_individual = Vec::new();
let (v_stash, v_controller) = if unique_controller {
create_unique_stash_controller::<T>(0, 100, destination.clone(), false)?
} else {
create_stash_controller::<T>(0, 100, destination.clone())?
};
let validator_prefs =
ValidatorPrefs { commission: Perbill::from_percent(50), ..Default::default() };
Staking::<T>::validate(RawOrigin::Signed(v_controller).into(), validator_prefs)?;
let stash_lookup = T::Lookup::unlookup(v_stash.clone());
points_total += 10;
points_individual.push((v_stash.clone(), 10));
let original_nominator_count = Nominators::<T>::count();
let mut nominators = Vec::new();
for i in 0..upper_bound {
let (n_stash, n_controller) = if !dead_controller {
create_stash_controller::<T>(u32::MAX - i, 100, destination.clone())?
} else {
create_unique_stash_controller::<T>(u32::MAX - i, 100, destination.clone(), true)?
};
if i < n {
Staking::<T>::nominate(
RawOrigin::Signed(n_controller.clone()).into(),
vec![stash_lookup.clone()],
)?;
nominators.push((n_stash, n_controller));
}
}
ValidatorCount::<T>::put(1);
let new_validators = Rotator::<T>::legacy_insta_plan_era();
let planned_era = CurrentEra::<T>::get().unwrap_or_default();
assert_eq!(new_validators.len(), 1, "New validators is not 1");
assert_eq!(new_validators[0], v_stash, "Our validator was not selected");
assert_ne!(Validators::<T>::count(), 0, "New validators count wrong");
assert_eq!(
Nominators::<T>::count(),
original_nominator_count + nominators.len() as u32,
"New nominators count wrong"
);
let reward = EraRewardPoints::<T> {
total: points_total,
individual: points_individual.into_iter().try_collect()?,
};
ErasRewardPoints::<T>::insert(planned_era, reward);
let total_payout = asset::existential_deposit::<T>()
.saturating_mul(upper_bound.into())
.saturating_mul(1000u32.into());
<ErasValidatorReward<T>>::insert(planned_era, total_payout);
Ok((v_stash, nominators, planned_era))
}
struct ListScenario<T: Config> {
origin_stash1: T::AccountId,
origin_controller1: T::AccountId,
dest_weight: BalanceOf<T>,
}
impl<T: Config> ListScenario<T> {
fn new(origin_weight: BalanceOf<T>, is_increase: bool) -> Result<Self, &'static str> {
ensure!(!origin_weight.is_zero(), "origin weight must be greater than 0");
let i = asset::burn::<T>(asset::total_issuance::<T>());
core::mem::forget(i);
let validator_account = account("random_validator", 0, SEED);
let validator_stake = asset::existential_deposit::<T>() * 1000u32.into();
asset::set_stakeable_balance::<T>(&validator_account, validator_stake);
assert_ok!(Staking::<T>::bond(
RawOrigin::Signed(validator_account.clone()).into(),
validator_stake / 2u32.into(),
RewardDestination::Staked
));
assert_ok!(Staking::<T>::validate(
RawOrigin::Signed(validator_account.clone()).into(),
Default::default()
));
let (origin_stash1, origin_controller1) = create_stash_controller_with_balance::<T>(
USER_SEED + 2,
origin_weight,
RewardDestination::Staked,
)?;
Staking::<T>::nominate(
RawOrigin::Signed(origin_controller1.clone()).into(),
vec![T::Lookup::unlookup(validator_account.clone())],
)?;
let (_origin_stash2, origin_controller2) = create_stash_controller_with_balance::<T>(
USER_SEED + 3,
origin_weight,
RewardDestination::Staked,
)?;
Staking::<T>::nominate(
RawOrigin::Signed(origin_controller2).into(),
vec![T::Lookup::unlookup(validator_account.clone())],
)?;
let dest_weight_as_vote =
T::VoterList::score_update_worst_case(&origin_stash1, is_increase);
let total_issuance = asset::total_issuance::<T>();
let dest_weight =
T::CurrencyToVote::to_currency(dest_weight_as_vote as u128, total_issuance);
let (_dest_stash1, dest_controller1) = create_stash_controller_with_balance::<T>(
USER_SEED + 1,
dest_weight,
RewardDestination::Staked,
)?;
Staking::<T>::nominate(
RawOrigin::Signed(dest_controller1).into(),
vec![T::Lookup::unlookup(validator_account)],
)?;
Ok(ListScenario { origin_stash1, origin_controller1, dest_weight })
}
}
const USER_SEED: u32 = 999666;
#[benchmarks]
mod benchmarks {
use super::*;
use alloc::format;
#[benchmark]
fn bond() {
let stash = create_funded_user::<T>("stash", USER_SEED, 100);
let reward_destination = RewardDestination::Staked;
let amount = asset::existential_deposit::<T>() * 10u32.into();
whitelist_account!(stash);
#[extrinsic_call]
_(RawOrigin::Signed(stash.clone()), amount, reward_destination);
assert!(Bonded::<T>::contains_key(stash.clone()));
assert!(Ledger::<T>::contains_key(stash));
}
#[benchmark]
fn bond_extra() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let max_additional = scenario.dest_weight - origin_weight;
let stash = scenario.origin_stash1.clone();
let controller = scenario.origin_controller1;
let original_bonded: BalanceOf<T> = Ledger::<T>::get(&controller)
.map(|l| l.active)
.ok_or("ledger not created after")?;
let _ = asset::mint_into_existing::<T>(
&stash,
max_additional + asset::existential_deposit::<T>(),
)
.unwrap();
whitelist_account!(stash);
#[extrinsic_call]
_(RawOrigin::Signed(stash), max_additional);
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created after")?;
let new_bonded: BalanceOf<T> = ledger.active;
assert!(original_bonded < new_bonded);
Ok(())
}
#[benchmark]
fn unbond() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = BalanceOf::<T>::try_from(952_994_955_240_703u128)
.map_err(|_| "balance expected to be a u128")
.unwrap();
let scenario = ListScenario::<T>::new(origin_weight, false)?;
let controller = scenario.origin_controller1.clone();
let amount = origin_weight - scenario.dest_weight;
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created before")?;
let original_bonded: BalanceOf<T> = ledger.active;
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller.clone()), amount);
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created after")?;
let new_bonded: BalanceOf<T> = ledger.active;
assert!(original_bonded > new_bonded);
Ok(())
}
#[benchmark]
fn withdraw_unbonded_update() -> Result<(), BenchmarkError> {
let (_, controller) = create_stash_controller::<T>(0, 100, RewardDestination::Staked)?;
let amount = asset::existential_deposit::<T>() * 5u32.into(); Staking::<T>::unbond(RawOrigin::Signed(controller.clone()).into(), amount)?;
set_active_era::<T>(EraIndex::max_value());
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created before")?;
let original_total: BalanceOf<T> = ledger.total;
whitelist_account!(controller);
#[extrinsic_call]
withdraw_unbonded(RawOrigin::Signed(controller.clone()), 0);
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created after")?;
let new_total: BalanceOf<T> = ledger.total;
assert!(original_total > new_total);
Ok(())
}
#[benchmark]
fn withdraw_unbonded_kill() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let controller = scenario.origin_controller1.clone();
let stash = scenario.origin_stash1;
assert!(T::VoterList::contains(&stash));
let ed = asset::existential_deposit::<T>();
let mut ledger = Ledger::<T>::get(&controller).unwrap();
ledger.active = ed - One::one();
Ledger::<T>::insert(&controller, ledger);
set_active_era::<T>(EraIndex::max_value());
whitelist_account!(controller);
#[extrinsic_call]
withdraw_unbonded(RawOrigin::Signed(controller.clone()), 0);
assert!(!Ledger::<T>::contains_key(controller));
assert!(!T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn validate() -> Result<(), BenchmarkError> {
let (stash, controller) = create_stash_controller::<T>(
MaxNominationsOf::<T>::get() - 1,
100,
RewardDestination::Staked,
)?;
assert!(!T::VoterList::contains(&stash));
let prefs = ValidatorPrefs::default();
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller), prefs);
assert!(Validators::<T>::contains_key(&stash));
assert!(T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn kick(
k: Linear<1, 128>,
) -> Result<(), BenchmarkError> {
let rest_of_validators =
create_validators_with_seed::<T>(MaxNominationsOf::<T>::get() - 1, 100, 415)?;
let (stash, controller) = create_stash_controller::<T>(
MaxNominationsOf::<T>::get() - 1,
100,
RewardDestination::Staked,
)?;
let stash_lookup = T::Lookup::unlookup(stash.clone());
Staking::<T>::validate(RawOrigin::Signed(controller.clone()).into(), Default::default())?;
let mut nominator_stashes = Vec::with_capacity(k as usize);
for i in 0..k {
let (n_stash, n_controller) = create_stash_controller::<T>(
MaxNominationsOf::<T>::get() + i,
100,
RewardDestination::Staked,
)?;
let mut nominations = rest_of_validators.clone();
nominations.insert(i as usize % (nominations.len() + 1), stash_lookup.clone());
Staking::<T>::nominate(RawOrigin::Signed(n_controller.clone()).into(), nominations)?;
nominator_stashes.push(n_stash);
}
for n in nominator_stashes.iter() {
assert!(Nominators::<T>::get(n).unwrap().targets.contains(&stash));
}
let kicks = nominator_stashes
.iter()
.map(|n| T::Lookup::unlookup(n.clone()))
.collect::<Vec<_>>();
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller), kicks);
for n in nominator_stashes.iter() {
assert!(!Nominators::<T>::get(n).unwrap().targets.contains(&stash));
}
Ok(())
}
#[benchmark]
fn nominate(n: Linear<1, { MaxNominationsOf::<T>::get() }>) -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
ListScenario::<T>::new(origin_weight, true)?;
let (stash, controller) = create_stash_controller_with_balance::<T>(
SEED + MaxNominationsOf::<T>::get() + 1,
origin_weight,
RewardDestination::Staked,
)
.unwrap();
assert!(!Nominators::<T>::contains_key(&stash));
assert!(!T::VoterList::contains(&stash));
let validators = create_validators::<T>(n, 100).unwrap();
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller), validators);
assert!(Nominators::<T>::contains_key(&stash));
assert!(T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn chill() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let controller = scenario.origin_controller1.clone();
let stash = scenario.origin_stash1;
assert!(T::VoterList::contains(&stash));
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller));
assert!(!T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn set_payee() -> Result<(), BenchmarkError> {
let (stash, controller) =
create_stash_controller::<T>(USER_SEED, 100, RewardDestination::Staked)?;
assert_eq!(Payee::<T>::get(&stash), Some(RewardDestination::Staked));
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller.clone()), RewardDestination::Account(controller.clone()));
assert_eq!(Payee::<T>::get(&stash), Some(RewardDestination::Account(controller)));
Ok(())
}
#[benchmark]
fn update_payee() -> Result<(), BenchmarkError> {
let (stash, controller) =
create_stash_controller::<T>(USER_SEED, 100, RewardDestination::Staked)?;
Payee::<T>::insert(&stash, {
#[allow(deprecated)]
RewardDestination::Controller
});
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller.clone()), controller.clone());
assert_eq!(Payee::<T>::get(&stash), Some(RewardDestination::Account(controller)));
Ok(())
}
#[benchmark]
fn set_controller() -> Result<(), BenchmarkError> {
let (stash, ctlr) =
create_unique_stash_controller::<T>(9000, 100, RewardDestination::Staked, false)?;
assert!(!Ledger::<T>::contains_key(&stash));
assert!(Ledger::<T>::contains_key(&ctlr));
assert_eq!(Bonded::<T>::get(&stash), Some(ctlr.clone()));
whitelist_account!(stash);
#[extrinsic_call]
_(RawOrigin::Signed(stash.clone()));
assert!(Ledger::<T>::contains_key(&stash));
Ok(())
}
#[benchmark]
fn set_validator_count() {
let validator_count = T::MaxValidatorSet::get() - 1;
#[extrinsic_call]
_(RawOrigin::Root, validator_count);
assert_eq!(ValidatorCount::<T>::get(), validator_count);
}
#[benchmark]
fn force_no_eras() {
#[extrinsic_call]
_(RawOrigin::Root);
assert_eq!(ForceEra::<T>::get(), Forcing::ForceNone);
}
#[benchmark]
fn force_new_era() {
#[extrinsic_call]
_(RawOrigin::Root);
assert_eq!(ForceEra::<T>::get(), Forcing::ForceNew);
}
#[benchmark]
fn force_new_era_always() {
#[extrinsic_call]
_(RawOrigin::Root);
assert_eq!(ForceEra::<T>::get(), Forcing::ForceAlways);
}
#[benchmark]
fn deprecate_controller_batch(
u: Linear<0, { T::MaxControllersInDeprecationBatch::get() }>,
) -> Result<(), BenchmarkError> {
let mut controllers: Vec<_> = vec![];
let mut stashes: Vec<_> = vec![];
for i in 0..u as u32 {
let (stash, controller) =
create_unique_stash_controller::<T>(i, 100, RewardDestination::Staked, false)?;
controllers.push(controller);
stashes.push(stash);
}
let bounded_controllers: BoundedVec<_, T::MaxControllersInDeprecationBatch> =
BoundedVec::try_from(controllers.clone()).unwrap();
#[extrinsic_call]
_(RawOrigin::Root, bounded_controllers);
for i in 0..u as u32 {
let stash = &stashes[i as usize];
let controller = &controllers[i as usize];
assert_eq!(Ledger::<T>::get(controller), None);
assert_eq!(Bonded::<T>::get(stash), Some(stash.clone()));
assert_eq!(Ledger::<T>::get(stash).unwrap().stash, *stash);
}
Ok(())
}
#[benchmark]
fn force_unstake() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let controller = scenario.origin_controller1.clone();
let stash = scenario.origin_stash1;
assert!(T::VoterList::contains(&stash));
#[extrinsic_call]
_(RawOrigin::Root, stash.clone(), 0);
assert!(!Ledger::<T>::contains_key(&controller));
assert!(!T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn cancel_deferred_slash(s: Linear<1, { T::MaxValidatorSet::get() }>) {
let era = EraIndex::one();
let validators: Vec<_> = (0..s)
.map(|i| {
let validator: T::AccountId = account("validator", i, SEED);
let slash_key = (validator.clone(), Perbill::from_percent(10), 0);
let unapplied_slash = UnappliedSlash::<T> {
validator: validator.clone(),
own: Zero::zero(),
others: WeakBoundedVec::default(),
reporter: Default::default(),
payout: Zero::zero(),
};
UnappliedSlashes::<T>::insert(era, slash_key, unapplied_slash);
validator
})
.collect();
let validator_slashes: Vec<_> =
validators.into_iter().map(|v| (v, Perbill::from_percent(10))).collect();
#[extrinsic_call]
_(RawOrigin::Root, era, validator_slashes.clone());
let cancelled_slashes = CancelledSlashes::<T>::get(era);
assert_eq!(cancelled_slashes.len(), s as usize);
}
#[benchmark]
fn payout_stakers_alive_staked(
n: Linear<0, { T::MaxExposurePageSize::get() as u32 }>,
) -> Result<(), BenchmarkError> {
let (validator, nominators, current_era) = create_validator_with_nominators::<T>(
n,
T::MaxExposurePageSize::get() as u32,
false,
true,
RewardDestination::Staked,
)?;
<ErasValidatorPrefs<T>>::insert(
current_era,
validator.clone(),
Validators::<T>::get(&validator),
);
let caller = whitelisted_caller();
let balance_before = asset::stakeable_balance::<T>(&validator);
let mut nominator_balances_before = Vec::new();
for (stash, _) in &nominators {
let balance = asset::stakeable_balance::<T>(stash);
nominator_balances_before.push(balance);
}
#[extrinsic_call]
payout_stakers(RawOrigin::Signed(caller), validator.clone(), current_era);
let balance_after = asset::stakeable_balance::<T>(&validator);
ensure!(
balance_before < balance_after,
"Balance of validator stash should have increased after payout.",
);
for ((stash, _), balance_before) in nominators.iter().zip(nominator_balances_before.iter())
{
let balance_after = asset::stakeable_balance::<T>(stash);
ensure!(
balance_before < &balance_after,
"Balance of nominator stash should have increased after payout.",
);
}
Ok(())
}
#[benchmark]
fn rebond(l: Linear<1, { T::MaxUnlockingChunks::get() as u32 }>) -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Pallet::<T>::min_nominator_bond()
.max(100u32.into());
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let dest_weight = scenario.dest_weight;
let rebond_amount = dest_weight - origin_weight;
let value = rebond_amount / l.into();
assert_ne!(value, Zero::zero());
assert!(value * l.into() + origin_weight > origin_weight);
assert!(value * l.into() + origin_weight <= dest_weight);
let unlock_chunk = UnlockChunk::<BalanceOf<T>> { value, era: EraIndex::zero() };
let controller = scenario.origin_controller1;
let mut staking_ledger = Ledger::<T>::get(controller.clone()).unwrap();
for _ in 0..l {
staking_ledger.unlocking.try_push(unlock_chunk.clone()).unwrap()
}
Ledger::<T>::insert(controller.clone(), staking_ledger.clone());
let original_bonded: BalanceOf<T> = staking_ledger.active;
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller.clone()), rebond_amount);
let ledger = Ledger::<T>::get(&controller).ok_or("ledger not created after")?;
let new_bonded: BalanceOf<T> = ledger.active;
assert!(original_bonded < new_bonded);
Ok(())
}
#[benchmark]
fn reap_stash() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let controller = scenario.origin_controller1.clone();
let stash = scenario.origin_stash1;
let l =
StakingLedger::<T>::new(stash.clone(), asset::existential_deposit::<T>() - One::one());
Ledger::<T>::insert(&controller, l);
assert!(Bonded::<T>::contains_key(&stash));
assert!(T::VoterList::contains(&stash));
whitelist_account!(controller);
#[extrinsic_call]
_(RawOrigin::Signed(controller), stash.clone(), 0);
assert!(!Bonded::<T>::contains_key(&stash));
assert!(!T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn set_staking_configs_all_set() {
#[extrinsic_call]
set_staking_configs(
RawOrigin::Root,
ConfigOp::Set(BalanceOf::<T>::max_value()),
ConfigOp::Set(BalanceOf::<T>::max_value()),
ConfigOp::Set(u32::MAX),
ConfigOp::Set(u32::MAX),
ConfigOp::Set(Percent::max_value()),
ConfigOp::Set(Perbill::max_value()),
ConfigOp::Set(Percent::max_value()),
ConfigOp::Set(false),
);
assert_eq!(MinNominatorBond::<T>::get(), BalanceOf::<T>::max_value());
assert_eq!(MinValidatorBond::<T>::get(), BalanceOf::<T>::max_value());
assert_eq!(MaxNominatorsCount::<T>::get(), Some(u32::MAX));
assert_eq!(MaxValidatorsCount::<T>::get(), Some(u32::MAX));
assert_eq!(ChillThreshold::<T>::get(), Some(Percent::from_percent(100)));
assert_eq!(MinCommission::<T>::get(), Perbill::from_percent(100));
assert_eq!(MaxStakedRewards::<T>::get(), Some(Percent::from_percent(100)));
assert_eq!(AreNominatorsSlashable::<T>::get(), false);
}
#[benchmark]
fn set_staking_configs_all_remove() {
#[extrinsic_call]
set_staking_configs(
RawOrigin::Root,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
ConfigOp::Remove,
);
assert!(!MinNominatorBond::<T>::exists());
assert!(!MinValidatorBond::<T>::exists());
assert!(!MaxNominatorsCount::<T>::exists());
assert!(!MaxValidatorsCount::<T>::exists());
assert!(!ChillThreshold::<T>::exists());
assert!(!MinCommission::<T>::exists());
assert!(!MaxStakedRewards::<T>::exists());
assert!(!AreNominatorsSlashable::<T>::exists());
}
#[benchmark]
fn chill_other() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let origin_weight = Staking::<T>::min_nominator_bond();
let scenario = ListScenario::<T>::new(origin_weight, true)?;
let stash = scenario.origin_stash1;
assert!(T::VoterList::contains(&stash));
Staking::<T>::set_staking_configs(
RawOrigin::Root.into(),
ConfigOp::Set(BalanceOf::<T>::max_value()),
ConfigOp::Set(BalanceOf::<T>::max_value()),
ConfigOp::Set(0),
ConfigOp::Set(0),
ConfigOp::Set(Percent::from_percent(0)),
ConfigOp::Set(Zero::zero()),
ConfigOp::Noop,
ConfigOp::Noop,
)?;
let caller = whitelisted_caller();
#[extrinsic_call]
_(RawOrigin::Signed(caller), stash.clone());
assert!(!T::VoterList::contains(&stash));
Ok(())
}
#[benchmark]
fn force_apply_min_commission() -> Result<(), BenchmarkError> {
clear_validators_and_nominators::<T>();
let (stash, controller) = create_stash_controller::<T>(1, 1, RewardDestination::Staked)?;
let validator_prefs =
ValidatorPrefs { commission: Perbill::from_percent(50), ..Default::default() };
Staking::<T>::validate(RawOrigin::Signed(controller).into(), validator_prefs)?;
assert_eq!(
Validators::<T>::get(&stash),
ValidatorPrefs { commission: Perbill::from_percent(50), ..Default::default() }
);
MinCommission::<T>::set(Perbill::from_percent(75));
let caller = whitelisted_caller();
#[extrinsic_call]
_(RawOrigin::Signed(caller), stash.clone());
assert_eq!(
Validators::<T>::get(&stash),
ValidatorPrefs { commission: Perbill::from_percent(75), ..Default::default() }
);
Ok(())
}
#[benchmark]
fn set_min_commission() {
let min_commission = Perbill::max_value();
#[extrinsic_call]
_(RawOrigin::Root, min_commission);
assert_eq!(MinCommission::<T>::get(), Perbill::from_percent(100));
}
#[benchmark]
fn restore_ledger() -> Result<(), BenchmarkError> {
let (stash, controller) = create_stash_controller::<T>(0, 100, RewardDestination::Staked)?;
Ledger::<T>::remove(controller);
#[extrinsic_call]
_(RawOrigin::Root, stash.clone(), None, None, None);
assert_eq!(Staking::<T>::inspect_bond_state(&stash), Ok(LedgerIntegrityState::Ok));
Ok(())
}
#[benchmark]
fn migrate_currency() -> Result<(), BenchmarkError> {
let (stash, _ctrl) =
create_stash_controller::<T>(USER_SEED, 100, RewardDestination::Staked)?;
let stake = asset::staked::<T>(&stash);
migrate_to_old_currency::<T>(stash.clone());
assert!(asset::staked::<T>(&stash).is_zero());
whitelist_account!(stash);
#[extrinsic_call]
_(RawOrigin::Signed(stash.clone()), stash.clone());
assert_eq!(asset::staked::<T>(&stash), stake);
Ok(())
}
#[benchmark]
fn apply_slash(
n: Linear<0, { T::MaxExposurePageSize::get() as u32 }>,
) -> Result<(), BenchmarkError> {
let era = EraIndex::one();
ActiveEra::<T>::put(ActiveEraInfo { index: era, start: None });
let (validator, nominators, _current_era) = create_validator_with_nominators::<T>(
T::MaxExposurePageSize::get() as u32,
T::MaxExposurePageSize::get() as u32,
false,
true,
RewardDestination::Staked,
)?;
let slash_fraction = Perbill::from_percent(10);
let page_index = 0;
let slashed_balance = BalanceOf::<T>::from(10u32);
let slash_key = (validator.clone(), slash_fraction, page_index);
let slashed_nominators = nominators
.iter()
.take(n as usize)
.map(|(nom, _)| (nom.clone(), slashed_balance))
.collect::<Vec<_>>();
let unapplied_slash = UnappliedSlash::<T> {
validator: validator.clone(),
own: slashed_balance,
others: WeakBoundedVec::force_from(slashed_nominators, None),
reporter: Default::default(),
payout: Zero::zero(),
};
UnappliedSlashes::<T>::insert(era, slash_key.clone(), unapplied_slash);
#[extrinsic_call]
_(RawOrigin::Signed(validator.clone()), era, slash_key.clone());
assert!(UnappliedSlashes::<T>::get(era, &slash_key).is_none());
Ok(())
}
#[benchmark]
fn process_offence_queue() -> Result<(), BenchmarkError> {
#[cfg(test)]
crate::mock::SlashDeferDuration::set(77);
let all_validators = crate::testing_utils::create_validators_with_nominators_for_era::<T>(
ValidatorCount::<T>::get().max(1),
2 * T::MaxExposurePageSize::get(),
16,
false,
Some(1),
)?;
let offender =
T::Lookup::lookup(all_validators.first().cloned().expect("must exist")).unwrap();
let _new_validators = Rotator::<T>::legacy_insta_plan_era();
Rotator::<T>::start_era(
crate::ActiveEraInfo { index: Rotator::<T>::planned_era() - 1, start: Some(1) },
42, 2, );
let offender_exposure =
Eras::<T>::get_full_exposure(Rotator::<T>::planned_era(), &offender);
ensure!(
offender_exposure.others.len() as u32 >= T::MaxExposurePageSize::get(),
"exposure not created"
);
let slash_session = 42;
let offences = vec![rc_client::Offence {
offender: offender.clone(),
reporters: Default::default(),
slash_fraction: Perbill::from_percent(50),
}];
<crate::Pallet<T> as rc_client::AHStakingInterface>::on_new_offences(
slash_session,
offences,
);
ensure!(
ValidatorSlashInEra::<T>::contains_key(Rotator::<T>::active_era(), offender),
"offence not submitted"
);
ensure!(
OffenceQueueEras::<T>::get().unwrap_or_default() == vec![Rotator::<T>::active_era()],
"offence should be queued"
);
#[block]
{
slashing::process_offence::<T>();
}
ensure!(OffenceQueueEras::<T>::get().is_none(), "offence should not be queued");
Ok(())
}
#[benchmark]
fn rc_on_offence(
v: Linear<2, { T::MaxValidatorSet::get() / 2 }>,
) -> Result<(), BenchmarkError> {
let initial_era = Rotator::<T>::planned_era();
let _ = crate::testing_utils::create_validators_with_nominators_for_era::<T>(
2 * v,
1000,
16,
false,
None,
)?;
let new_validators = Rotator::<T>::legacy_insta_plan_era();
ensure!(Rotator::<T>::planned_era() == initial_era + 1, "era should be incremented");
Rotator::<T>::start_era(
crate::ActiveEraInfo { index: initial_era, start: Some(1) },
42, 2, );
ensure!(Rotator::<T>::active_era_start_session_index() == 42, "BondedEra not set");
let to_slash_count = new_validators.len() / 2;
let to_slash = new_validators.into_iter().take(to_slash_count).collect::<Vec<_>>();
let one_slashed = to_slash.first().cloned().unwrap();
let offences = to_slash
.into_iter()
.map(|offender| rc_client::Offence {
offender,
reporters: Default::default(),
slash_fraction: Perbill::from_percent(50),
})
.collect::<Vec<_>>();
let slash_session = 42;
ensure!(
!ValidatorSlashInEra::<T>::contains_key(initial_era + 1, &one_slashed),
"offence submitted???"
);
#[block]
{
<crate::Pallet<T> as rc_client::AHStakingInterface>::on_new_offences(
slash_session,
offences,
);
}
ensure!(
ValidatorSlashInEra::<T>::contains_key(initial_era + 1, one_slashed),
"offence not submitted"
);
Ok(())
}
#[benchmark]
fn rc_on_session_report() -> Result<(), BenchmarkError> {
let initial_planned_era = Rotator::<T>::planned_era();
let initial_active_era = Rotator::<T>::active_era();
crate::testing_utils::create_validators_with_nominators_for_era::<T>(
10, 50, 2, false, None,
)?;
let _new_validators = Rotator::<T>::legacy_insta_plan_era();
ensure!(
CurrentEra::<T>::get().unwrap() == initial_planned_era + 1,
"era should be incremented"
);
let validator_points = (0..T::MaxValidatorSet::get())
.map(|v| (account::<T::AccountId>("random", v, SEED), v))
.collect::<Vec<_>>();
let activation_timestamp = Some((1u64, initial_planned_era + 1));
let report = rc_client::SessionReport {
end_index: 42,
leftover: false,
validator_points,
activation_timestamp,
};
#[block]
{
<crate::Pallet<T> as rc_client::AHStakingInterface>::on_relay_session_report(report);
}
ensure!(Rotator::<T>::active_era() == initial_active_era + 1, "active era not bumped");
Ok(())
}
fn setup_era_for_pruning<T: Config>(v: u32) -> EraIndex {
let validators = v;
let era = 7;
let history_depth = T::HistoryDepth::get();
let active_era = era + history_depth + 1;
crate::ActiveEra::<T>::put(crate::ActiveEraInfo { index: active_era, start: Some(0) });
let max_total_nominators_per_validator =
<T::ElectionProvider as ElectionProvider>::MaxBackersPerWinnerFinal::get();
let exposed_nominators_per_validator = max_total_nominators_per_validator / validators;
for i in 0..validators {
let validator = account::<T::AccountId>("validator", i, SEED);
ErasValidatorPrefs::<T>::insert(era, validator.clone(), ValidatorPrefs::default())
}
let pages: WeakBoundedVec<_, _> = (0..crate::ClaimedRewardsBound::<T>::get())
.collect::<Vec<_>>()
.try_into()
.unwrap();
for i in 0..validators {
let validator = account::<T::AccountId>("validator", i, SEED);
ClaimedRewards::<T>::insert(era, validator.clone(), pages.clone())
}
(0..validators)
.map(|validator_index| account::<T::AccountId>("validator", validator_index, SEED))
.for_each(|validator| {
let exposure = sp_staking::Exposure::<T::AccountId, BalanceOf<T>> {
own: T::Currency::minimum_balance(),
total: T::Currency::minimum_balance() *
(exposed_nominators_per_validator + 1).into(),
others: (0..exposed_nominators_per_validator)
.map(|n| {
let nominator = account::<T::AccountId>("nominator", n, SEED);
IndividualExposure {
who: nominator,
value: T::Currency::minimum_balance(),
}
})
.collect::<Vec<_>>(),
};
Eras::<T>::upsert_exposure(era, &validator, exposure);
});
ErasValidatorReward::<T>::insert(era, BalanceOf::<T>::max_value());
let reward_points = crate::EraRewardPoints::<T> {
total: 77777,
individual: (0..validators)
.map(|v| account::<T::AccountId>("validator", v, SEED))
.map(|v| (v, 7))
.collect::<BTreeMap<_, _>>()
.try_into()
.unwrap(),
};
ErasRewardPoints::<T>::insert(era, reward_points);
ErasTotalStake::<T>::insert(era, BalanceOf::<T>::max_value());
let slashed_validators = validators / 3;
for i in 0..slashed_validators {
let validator = account::<T::AccountId>("validator", i, SEED);
crate::ValidatorSlashInEra::<T>::insert(
era,
validator,
(Perbill::from_percent(10), BalanceOf::<T>::max_value() / 10u32.into()),
);
}
ErasNominatorsSlashable::<T>::insert(era, true);
era
}
fn validate_pruning_weight<T: Config>(
result: &frame_support::dispatch::DispatchResultWithPostInfo,
step_name: &str,
validator_count: u32,
) {
assert!(
result.is_ok(),
"Benchmark {} should succeed with v={}",
step_name,
validator_count
);
let post_info = result.unwrap();
let actual_ref_time = post_info
.actual_weight
.expect(&format!(
"Should report actual weight for {} with v={}",
step_name, validator_count
))
.ref_time();
assert!(
actual_ref_time > 0,
"Should report non-zero ref_time for {} with v={}",
step_name,
validator_count
);
}
#[benchmark(pov_mode = Measured)]
fn prune_era_stakers_paged(
v: Linear<1, { T::MaxValidatorSet::get() }>,
) -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(v);
EraPruningState::<T>::insert(era, PruningStep::ErasStakersPaged);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ErasStakersPaged", v);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_stakers_overview(
v: Linear<1, { T::MaxValidatorSet::get() }>,
) -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(v);
EraPruningState::<T>::insert(era, PruningStep::ErasStakersOverview);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ErasStakersOverview", v);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_validator_prefs(
v: Linear<1, { T::MaxValidatorSet::get() }>,
) -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(v);
EraPruningState::<T>::insert(era, PruningStep::ErasValidatorPrefs);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ErasValidatorPrefs", v);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_claimed_rewards(
v: Linear<1, { T::MaxValidatorSet::get() }>,
) -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(v);
EraPruningState::<T>::insert(era, PruningStep::ClaimedRewards);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ClaimedRewards", v);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_validator_reward() -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(1);
EraPruningState::<T>::insert(era, PruningStep::ErasValidatorReward);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ErasValidatorReward", 1);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_reward_points() -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(1);
EraPruningState::<T>::insert(era, PruningStep::ErasRewardPoints);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ErasRewardPoints", 1);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_single_entry_cleanups() -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(1);
EraPruningState::<T>::insert(era, PruningStep::SingleEntryCleanups);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "SingleEntryCleanups", 1);
Ok(())
}
#[benchmark(pov_mode = Measured)]
fn prune_era_validator_slash_in_era(
v: Linear<1, { T::MaxValidatorSet::get() }>,
) -> Result<(), BenchmarkError> {
let era = setup_era_for_pruning::<T>(v);
EraPruningState::<T>::insert(era, PruningStep::ValidatorSlashInEra);
let caller: T::AccountId = whitelisted_caller();
let result;
#[block]
{
result = Pallet::<T>::prune_era_step(RawOrigin::Signed(caller).into(), era);
}
validate_pruning_weight::<T>(&result, "ValidatorSlashInEra", v);
Ok(())
}
impl_benchmark_test_suite!(
Staking,
crate::mock::ExtBuilder::default().has_stakers(true),
crate::mock::Test,
exec_name = build_and_execute
);
}
#[cfg(test)]
mod tests {
use super::*;
use crate::mock::{ExtBuilder, RuntimeOrigin, Staking, Test};
use frame_support::assert_ok;
#[test]
fn create_validators_with_nominators_for_era_works() {
ExtBuilder::default().build_and_execute(|| {
let v = 10;
let n = 100;
create_validators_with_nominators_for_era::<Test>(
v,
n,
MaxNominationsOf::<Test>::get() as usize,
false,
None,
)
.unwrap();
let count_validators = Validators::<Test>::iter().count();
let count_nominators = Nominators::<Test>::iter().count();
assert_eq!(count_validators, Validators::<Test>::count() as usize);
assert_eq!(count_nominators, Nominators::<Test>::count() as usize);
assert_eq!(count_validators, v as usize);
assert_eq!(count_nominators, n as usize);
});
}
#[test]
fn create_validator_with_nominators_works() {
ExtBuilder::default().build_and_execute(|| {
let n = 10;
let (validator_stash, nominators, current_era) =
create_validator_with_nominators::<Test>(
n,
<<Test as Config>::MaxExposurePageSize as Get<_>>::get(),
false,
false,
RewardDestination::Staked,
)
.unwrap();
assert_eq!(nominators.len() as u32, n);
let original_stakeable_balance = asset::stakeable_balance::<Test>(&validator_stash);
assert_ok!(Staking::payout_stakers_by_page(
RuntimeOrigin::signed(1337),
validator_stash,
current_era,
0
));
let new_stakeable_balance = asset::stakeable_balance::<Test>(&validator_stash);
assert!(original_stakeable_balance < new_stakeable_balance);
});
}
}