// This file is part of Substrate. // Copyright (C) 2017-2022 Parity Technologies (UK) Ltd. // SPDX-License-Identifier: Apache-2.0 // 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. //! Implementations for the Staking FRAME Pallet. use frame_election_provider_support::{ data_provider, ElectionDataProvider, ElectionProvider, SortedListProvider, Supports, VoteWeight, VoteWeightProvider, VoterOf, }; use frame_support::{ pallet_prelude::*, traits::{ Currency, CurrencyToVote, Defensive, EstimateNextNewSession, Get, Imbalance, LockableCurrency, OnUnbalanced, UnixTime, WithdrawReasons, }, weights::{Weight, WithPostDispatchInfo}, }; use frame_system::pallet_prelude::BlockNumberFor; use pallet_session::historical; use sp_runtime::{ traits::{Bounded, Convert, SaturatedConversion, Saturating, Zero}, Perbill, }; use sp_staking::{ offence::{DisableStrategy, OffenceDetails, OnOffenceHandler}, EraIndex, SessionIndex, }; use sp_std::{collections::btree_map::BTreeMap, prelude::*}; use crate::{ log, slashing, weights::WeightInfo, ActiveEraInfo, BalanceOf, EraPayout, Exposure, ExposureOf, Forcing, IndividualExposure, Nominations, PositiveImbalanceOf, RewardDestination, SessionInterface, StakingLedger, ValidatorPrefs, }; use super::{pallet::*, STAKING_ID}; impl Pallet { /// The total balance that can be slashed from a stash account as of right now. pub fn slashable_balance_of(stash: &T::AccountId) -> BalanceOf { // Weight note: consider making the stake accessible through stash. Self::bonded(stash).and_then(Self::ledger).map(|l| l.active).unwrap_or_default() } /// Internal impl of [`Self::slashable_balance_of`] that returns [`VoteWeight`]. pub fn slashable_balance_of_vote_weight( stash: &T::AccountId, issuance: BalanceOf, ) -> VoteWeight { T::CurrencyToVote::to_vote(Self::slashable_balance_of(stash), issuance) } /// Returns a closure around `slashable_balance_of_vote_weight` that can be passed around. /// /// This prevents call sites from repeatedly requesting `total_issuance` from backend. But it is /// important to be only used while the total issuance is not changing. pub fn weight_of_fn() -> Box VoteWeight> { // NOTE: changing this to unboxed `impl Fn(..)` return type and the pallet will still // compile, while some types in mock fail to resolve. let issuance = T::Currency::total_issuance(); Box::new(move |who: &T::AccountId| -> VoteWeight { Self::slashable_balance_of_vote_weight(who, issuance) }) } /// Same as `weight_of_fn`, but made for one time use. pub fn weight_of(who: &T::AccountId) -> VoteWeight { let issuance = T::Currency::total_issuance(); Self::slashable_balance_of_vote_weight(who, issuance) } pub(super) fn do_payout_stakers( validator_stash: T::AccountId, era: EraIndex, ) -> DispatchResultWithPostInfo { // Validate input data let current_era = CurrentEra::::get().ok_or_else(|| { Error::::InvalidEraToReward .with_weight(T::WeightInfo::payout_stakers_alive_staked(0)) })?; let history_depth = Self::history_depth(); ensure!( era <= current_era && era >= current_era.saturating_sub(history_depth), Error::::InvalidEraToReward .with_weight(T::WeightInfo::payout_stakers_alive_staked(0)) ); // Note: if era has no reward to be claimed, era may be future. better not to update // `ledger.claimed_rewards` in this case. let era_payout = >::get(&era).ok_or_else(|| { Error::::InvalidEraToReward .with_weight(T::WeightInfo::payout_stakers_alive_staked(0)) })?; let controller = Self::bonded(&validator_stash).ok_or_else(|| { Error::::NotStash.with_weight(T::WeightInfo::payout_stakers_alive_staked(0)) })?; let mut ledger = >::get(&controller).ok_or(Error::::NotController)?; ledger .claimed_rewards .retain(|&x| x >= current_era.saturating_sub(history_depth)); match ledger.claimed_rewards.binary_search(&era) { Ok(_) => Err(Error::::AlreadyClaimed .with_weight(T::WeightInfo::payout_stakers_alive_staked(0)))?, Err(pos) => ledger.claimed_rewards.insert(pos, era), } let exposure = >::get(&era, &ledger.stash); // Input data seems good, no errors allowed after this point >::insert(&controller, &ledger); // Get Era reward points. It has TOTAL and INDIVIDUAL // Find the fraction of the era reward that belongs to the validator // Take that fraction of the eras rewards to split to nominator and validator // // Then look at the validator, figure out the proportion of their reward // which goes to them and each of their nominators. let era_reward_points = >::get(&era); let total_reward_points = era_reward_points.total; let validator_reward_points = era_reward_points .individual .get(&ledger.stash) .map(|points| *points) .unwrap_or_else(|| Zero::zero()); // Nothing to do if they have no reward points. if validator_reward_points.is_zero() { return Ok(Some(T::WeightInfo::payout_stakers_alive_staked(0)).into()) } // This is the fraction of the total reward that the validator and the // nominators will get. let validator_total_reward_part = Perbill::from_rational(validator_reward_points, total_reward_points); // This is how much validator + nominators are entitled to. let validator_total_payout = validator_total_reward_part * era_payout; let validator_prefs = Self::eras_validator_prefs(&era, &validator_stash); // Validator first gets a cut off the top. let validator_commission = validator_prefs.commission; let validator_commission_payout = validator_commission * validator_total_payout; let validator_leftover_payout = validator_total_payout - validator_commission_payout; // Now let's calculate how this is split to the validator. let validator_exposure_part = Perbill::from_rational(exposure.own, exposure.total); let validator_staking_payout = validator_exposure_part * validator_leftover_payout; Self::deposit_event(Event::::PayoutStarted(era, ledger.stash.clone())); // We can now make total validator payout: if let Some(imbalance) = Self::make_payout(&ledger.stash, validator_staking_payout + validator_commission_payout) { Self::deposit_event(Event::::Rewarded(ledger.stash, imbalance.peek())); } // Track the number of payout ops to nominators. Note: // `WeightInfo::payout_stakers_alive_staked` always assumes at least a validator is paid // out, so we do not need to count their payout op. let mut nominator_payout_count: u32 = 0; // Lets now calculate how this is split to the nominators. // Reward only the clipped exposures. Note this is not necessarily sorted. for nominator in exposure.others.iter() { let nominator_exposure_part = Perbill::from_rational(nominator.value, exposure.total); let nominator_reward: BalanceOf = nominator_exposure_part * validator_leftover_payout; // We can now make nominator payout: if let Some(imbalance) = Self::make_payout(&nominator.who, nominator_reward) { // Note: this logic does not count payouts for `RewardDestination::None`. nominator_payout_count += 1; let e = Event::::Rewarded(nominator.who.clone(), imbalance.peek()); Self::deposit_event(e); } } debug_assert!(nominator_payout_count <= T::MaxNominatorRewardedPerValidator::get()); Ok(Some(T::WeightInfo::payout_stakers_alive_staked(nominator_payout_count)).into()) } /// Update the ledger for a controller. /// /// This will also update the stash lock. pub(crate) fn update_ledger( controller: &T::AccountId, ledger: &StakingLedger>, ) { T::Currency::set_lock(STAKING_ID, &ledger.stash, ledger.total, WithdrawReasons::all()); >::insert(controller, ledger); } /// Chill a stash account. pub(crate) fn chill_stash(stash: &T::AccountId) { let chilled_as_validator = Self::do_remove_validator(stash); let chilled_as_nominator = Self::do_remove_nominator(stash); if chilled_as_validator || chilled_as_nominator { Self::deposit_event(Event::::Chilled(stash.clone())); } } /// Actually make a payment to a staker. This uses the currency's reward function /// to pay the right payee for the given staker account. fn make_payout(stash: &T::AccountId, amount: BalanceOf) -> Option> { let dest = Self::payee(stash); match dest { RewardDestination::Controller => Self::bonded(stash) .map(|controller| T::Currency::deposit_creating(&controller, amount)), RewardDestination::Stash => T::Currency::deposit_into_existing(stash, amount).ok(), RewardDestination::Staked => Self::bonded(stash) .and_then(|c| Self::ledger(&c).map(|l| (c, l))) .and_then(|(controller, mut l)| { l.active += amount; l.total += amount; let r = T::Currency::deposit_into_existing(stash, amount).ok(); Self::update_ledger(&controller, &l); r }), RewardDestination::Account(dest_account) => Some(T::Currency::deposit_creating(&dest_account, amount)), RewardDestination::None => None, } } /// Plan a new session potentially trigger a new era. fn new_session(session_index: SessionIndex, is_genesis: bool) -> Option> { if let Some(current_era) = Self::current_era() { // Initial era has been set. let current_era_start_session_index = Self::eras_start_session_index(current_era) .unwrap_or_else(|| { frame_support::print("Error: start_session_index must be set for current_era"); 0 }); let era_length = session_index.checked_sub(current_era_start_session_index).unwrap_or(0); // Must never happen. match ForceEra::::get() { // Will be set to `NotForcing` again if a new era has been triggered. Forcing::ForceNew => (), // Short circuit to `try_trigger_new_era`. Forcing::ForceAlways => (), // Only go to `try_trigger_new_era` if deadline reached. Forcing::NotForcing if era_length >= T::SessionsPerEra::get() => (), _ => { // Either `Forcing::ForceNone`, // or `Forcing::NotForcing if era_length >= T::SessionsPerEra::get()`. return None }, } // New era. let maybe_new_era_validators = Self::try_trigger_new_era(session_index, is_genesis); if maybe_new_era_validators.is_some() && matches!(ForceEra::::get(), Forcing::ForceNew) { ForceEra::::put(Forcing::NotForcing); } maybe_new_era_validators } else { // Set initial era. log!(debug, "Starting the first era."); Self::try_trigger_new_era(session_index, is_genesis) } } /// Start a session potentially starting an era. fn start_session(start_session: SessionIndex) { let next_active_era = Self::active_era().map(|e| e.index + 1).unwrap_or(0); // This is only `Some` when current era has already progressed to the next era, while the // active era is one behind (i.e. in the *last session of the active era*, or *first session // of the new current era*, depending on how you look at it). if let Some(next_active_era_start_session_index) = Self::eras_start_session_index(next_active_era) { if next_active_era_start_session_index == start_session { Self::start_era(start_session); } else if next_active_era_start_session_index < start_session { // This arm should never happen, but better handle it than to stall the staking // pallet. frame_support::print("Warning: A session appears to have been skipped."); Self::start_era(start_session); } } // disable all offending validators that have been disabled for the whole era for (index, disabled) in >::get() { if disabled { T::SessionInterface::disable_validator(index); } } } /// End a session potentially ending an era. fn end_session(session_index: SessionIndex) { if let Some(active_era) = Self::active_era() { if let Some(next_active_era_start_session_index) = Self::eras_start_session_index(active_era.index + 1) { if next_active_era_start_session_index == session_index + 1 { Self::end_era(active_era, session_index); } } } } /// /// * Increment `active_era.index`, /// * reset `active_era.start`, /// * update `BondedEras` and apply slashes. fn start_era(start_session: SessionIndex) { let active_era = ActiveEra::::mutate(|active_era| { let new_index = active_era.as_ref().map(|info| info.index + 1).unwrap_or(0); *active_era = Some(ActiveEraInfo { index: new_index, // Set new active era start in next `on_finalize`. To guarantee usage of `Time` start: None, }); new_index }); let bonding_duration = T::BondingDuration::get(); BondedEras::::mutate(|bonded| { bonded.push((active_era, start_session)); if active_era > bonding_duration { let first_kept = active_era - bonding_duration; // Prune out everything that's from before the first-kept index. let n_to_prune = bonded.iter().take_while(|&&(era_idx, _)| era_idx < first_kept).count(); // Kill slashing metadata. for (pruned_era, _) in bonded.drain(..n_to_prune) { slashing::clear_era_metadata::(pruned_era); } if let Some(&(_, first_session)) = bonded.first() { T::SessionInterface::prune_historical_up_to(first_session); } } }); Self::apply_unapplied_slashes(active_era); } /// Compute payout for era. fn end_era(active_era: ActiveEraInfo, _session_index: SessionIndex) { // Note: active_era_start can be None if end era is called during genesis config. if let Some(active_era_start) = active_era.start { let now_as_millis_u64 = T::UnixTime::now().as_millis().saturated_into::(); let era_duration = (now_as_millis_u64 - active_era_start).saturated_into::(); let staked = Self::eras_total_stake(&active_era.index); let issuance = T::Currency::total_issuance(); let (validator_payout, rest) = T::EraPayout::era_payout(staked, issuance, era_duration); Self::deposit_event(Event::::EraPaid(active_era.index, validator_payout, rest)); // Set ending era reward. >::insert(&active_era.index, validator_payout); T::RewardRemainder::on_unbalanced(T::Currency::issue(rest)); // Clear offending validators. >::kill(); } } /// Plan a new era. /// /// * Bump the current era storage (which holds the latest planned era). /// * Store start session index for the new planned era. /// * Clean old era information. /// * Store staking information for the new planned era /// /// Returns the new validator set. pub fn trigger_new_era( start_session_index: SessionIndex, exposures: Vec<(T::AccountId, Exposure>)>, ) -> Vec { // Increment or set current era. let new_planned_era = CurrentEra::::mutate(|s| { *s = Some(s.map(|s| s + 1).unwrap_or(0)); s.unwrap() }); ErasStartSessionIndex::::insert(&new_planned_era, &start_session_index); // Clean old era information. if let Some(old_era) = new_planned_era.checked_sub(Self::history_depth() + 1) { Self::clear_era_information(old_era); } // Set staking information for the new era. Self::store_stakers_info(exposures, new_planned_era) } /// Potentially plan a new era. /// /// Get election result from `T::ElectionProvider`. /// In case election result has more than [`MinimumValidatorCount`] validator trigger a new era. /// /// In case a new era is planned, the new validator set is returned. pub(crate) fn try_trigger_new_era( start_session_index: SessionIndex, is_genesis: bool, ) -> Option> { let election_result = if is_genesis { T::GenesisElectionProvider::elect().map_err(|e| { log!(warn, "genesis election provider failed due to {:?}", e); Self::deposit_event(Event::StakingElectionFailed); }) } else { T::ElectionProvider::elect().map_err(|e| { log!(warn, "election provider failed due to {:?}", e); Self::deposit_event(Event::StakingElectionFailed); }) } .ok()?; let exposures = Self::collect_exposures(election_result); if (exposures.len() as u32) < Self::minimum_validator_count().max(1) { // Session will panic if we ever return an empty validator set, thus max(1) ^^. match CurrentEra::::get() { Some(current_era) if current_era > 0 => log!( warn, "chain does not have enough staking candidates to operate for era {:?} ({} \ elected, minimum is {})", CurrentEra::::get().unwrap_or(0), exposures.len(), Self::minimum_validator_count(), ), None => { // The initial era is allowed to have no exposures. // In this case the SessionManager is expected to choose a sensible validator // set. // TODO: this should be simplified #8911 CurrentEra::::put(0); ErasStartSessionIndex::::insert(&0, &start_session_index); }, _ => (), } Self::deposit_event(Event::StakingElectionFailed); return None } Self::deposit_event(Event::StakersElected); Some(Self::trigger_new_era(start_session_index, exposures)) } /// Process the output of the election. /// /// Store staking information for the new planned era pub fn store_stakers_info( exposures: Vec<(T::AccountId, Exposure>)>, new_planned_era: EraIndex, ) -> Vec { let elected_stashes = exposures.iter().cloned().map(|(x, _)| x).collect::>(); // Populate stakers, exposures, and the snapshot of validator prefs. let mut total_stake: BalanceOf = Zero::zero(); exposures.into_iter().for_each(|(stash, exposure)| { total_stake = total_stake.saturating_add(exposure.total); >::insert(new_planned_era, &stash, &exposure); let mut exposure_clipped = exposure; let clipped_max_len = T::MaxNominatorRewardedPerValidator::get() as usize; if exposure_clipped.others.len() > clipped_max_len { exposure_clipped.others.sort_by(|a, b| a.value.cmp(&b.value).reverse()); exposure_clipped.others.truncate(clipped_max_len); } >::insert(&new_planned_era, &stash, exposure_clipped); }); // Insert current era staking information >::insert(&new_planned_era, total_stake); // Collect the pref of all winners. for stash in &elected_stashes { let pref = Self::validators(stash); >::insert(&new_planned_era, stash, pref); } if new_planned_era > 0 { log!( info, "new validator set of size {:?} has been processed for era {:?}", elected_stashes.len(), new_planned_era, ); } elected_stashes } /// Consume a set of [`Supports`] from [`sp_npos_elections`] and collect them into a /// [`Exposure`]. fn collect_exposures( supports: Supports, ) -> Vec<(T::AccountId, Exposure>)> { let total_issuance = T::Currency::total_issuance(); let to_currency = |e: frame_election_provider_support::ExtendedBalance| { T::CurrencyToVote::to_currency(e, total_issuance) }; supports .into_iter() .map(|(validator, support)| { // Build `struct exposure` from `support`. let mut others = Vec::with_capacity(support.voters.len()); let mut own: BalanceOf = Zero::zero(); let mut total: BalanceOf = Zero::zero(); support .voters .into_iter() .map(|(nominator, weight)| (nominator, to_currency(weight))) .for_each(|(nominator, stake)| { if nominator == validator { own = own.saturating_add(stake); } else { others.push(IndividualExposure { who: nominator, value: stake }); } total = total.saturating_add(stake); }); let exposure = Exposure { own, others, total }; (validator, exposure) }) .collect::)>>() } /// Remove all associated data of a stash account from the staking system. /// /// Assumes storage is upgraded before calling. /// /// This is called: /// - after a `withdraw_unbonded()` call that frees all of a stash's bonded balance. /// - through `reap_stash()` if the balance has fallen to zero (through slashing). pub(crate) fn kill_stash(stash: &T::AccountId, num_slashing_spans: u32) -> DispatchResult { let controller = >::get(stash).ok_or(Error::::NotStash)?; slashing::clear_stash_metadata::(stash, num_slashing_spans)?; >::remove(stash); >::remove(&controller); >::remove(stash); Self::do_remove_validator(stash); Self::do_remove_nominator(stash); frame_system::Pallet::::dec_consumers(stash); Ok(()) } /// Clear all era information for given era. pub(crate) fn clear_era_information(era_index: EraIndex) { >::remove_prefix(era_index, None); >::remove_prefix(era_index, None); >::remove_prefix(era_index, None); >::remove(era_index); >::remove(era_index); >::remove(era_index); ErasStartSessionIndex::::remove(era_index); } /// Apply previously-unapplied slashes on the beginning of a new era, after a delay. fn apply_unapplied_slashes(active_era: EraIndex) { let slash_defer_duration = T::SlashDeferDuration::get(); ::EarliestUnappliedSlash::mutate(|earliest| { if let Some(ref mut earliest) = earliest { let keep_from = active_era.saturating_sub(slash_defer_duration); for era in (*earliest)..keep_from { let era_slashes = ::UnappliedSlashes::take(&era); for slash in era_slashes { slashing::apply_slash::(slash); } } *earliest = (*earliest).max(keep_from) } }) } /// Add reward points to validators using their stash account ID. /// /// Validators are keyed by stash account ID and must be in the current elected set. /// /// For each element in the iterator the given number of points in u32 is added to the /// validator, thus duplicates are handled. /// /// At the end of the era each the total payout will be distributed among validator /// relatively to their points. /// /// COMPLEXITY: Complexity is `number_of_validator_to_reward x current_elected_len`. pub fn reward_by_ids(validators_points: impl IntoIterator) { if let Some(active_era) = Self::active_era() { >::mutate(active_era.index, |era_rewards| { for (validator, points) in validators_points.into_iter() { *era_rewards.individual.entry(validator).or_default() += points; era_rewards.total += points; } }); } } /// Ensures that at the end of the current session there will be a new era. pub(crate) fn ensure_new_era() { match ForceEra::::get() { Forcing::ForceAlways | Forcing::ForceNew => (), _ => ForceEra::::put(Forcing::ForceNew), } } #[cfg(feature = "runtime-benchmarks")] pub fn add_era_stakers( current_era: EraIndex, controller: T::AccountId, exposure: Exposure>, ) { >::insert(¤t_era, &controller, &exposure); } #[cfg(feature = "runtime-benchmarks")] pub fn set_slash_reward_fraction(fraction: Perbill) { SlashRewardFraction::::put(fraction); } /// Get all of the voters that are eligible for the npos election. /// /// `maybe_max_len` can imposes a cap on the number of voters returned; First all the validator /// are included in no particular order, then remainder is taken from the nominators, as /// returned by [`Config::SortedListProvider`]. /// /// This will use nominators, and all the validators will inject a self vote. /// /// This function is self-weighing as [`DispatchClass::Mandatory`]. /// /// ### Slashing /// /// All nominations that have been submitted before the last non-zero slash of the validator are /// auto-chilled, but still count towards the limit imposed by `maybe_max_len`. pub fn get_npos_voters(maybe_max_len: Option) -> Vec> { let max_allowed_len = { let nominator_count = Nominators::::count() as usize; let validator_count = Validators::::count() as usize; let all_voter_count = validator_count.saturating_add(nominator_count); maybe_max_len.unwrap_or(all_voter_count).min(all_voter_count) }; let mut all_voters = Vec::<_>::with_capacity(max_allowed_len); // first, grab all validators in no particular order, capped by the maximum allowed length. let mut validators_taken = 0u32; for (validator, _) in >::iter().take(max_allowed_len) { // Append self vote. let self_vote = ( validator.clone(), Self::weight_of(&validator), vec![validator.clone()] .try_into() .expect("`MaxVotesPerVoter` must be greater than or equal to 1"), ); all_voters.push(self_vote); validators_taken.saturating_inc(); } // .. and grab whatever we have left from nominators. let nominators_quota = (max_allowed_len as u32).saturating_sub(validators_taken); let slashing_spans = >::iter().collect::>(); // track the count of nominators added to `all_voters let mut nominators_taken = 0u32; // track every nominator iterated over, but not necessarily added to `all_voters` let mut nominators_seen = 0u32; // cache the total-issuance once in this function let weight_of = Self::weight_of_fn(); let mut nominators_iter = T::SortedListProvider::iter(); while nominators_taken < nominators_quota && nominators_seen < nominators_quota * 2 { let nominator = match nominators_iter.next() { Some(nominator) => { nominators_seen.saturating_inc(); nominator }, None => break, }; if let Some(Nominations { submitted_in, mut targets, suppressed: _ }) = >::get(&nominator) { log!( trace, "fetched nominator {:?} with weight {:?}", nominator, weight_of(&nominator) ); targets.retain(|stash| { slashing_spans .get(stash) .map_or(true, |spans| submitted_in >= spans.last_nonzero_slash()) }); if !targets.len().is_zero() { all_voters.push((nominator.clone(), weight_of(&nominator), targets)); nominators_taken.saturating_inc(); } } else { // this can only happen if: 1. there a pretty bad bug in the bags-list (or whatever // is the sorted list) logic and the state of the two pallets is no longer // compatible, or because the nominators is not decodable since they have more // nomination than `T::MaxNominations`. This can rarely happen, and is not really an // emergency or bug if it does. log!(warn, "DEFENSIVE: invalid item in `SortedListProvider`: {:?}, this nominator probably has too many nominations now", nominator) } } // all_voters should have not re-allocated. debug_assert!(all_voters.capacity() == max_allowed_len); Self::register_weight(T::WeightInfo::get_npos_voters( validators_taken, nominators_taken, slashing_spans.len() as u32, )); log!( info, "generated {} npos voters, {} from validators and {} nominators", all_voters.len(), validators_taken, nominators_taken ); all_voters } /// Get the targets for an upcoming npos election. /// /// This function is self-weighing as [`DispatchClass::Mandatory`]. pub fn get_npos_targets() -> Vec { let mut validator_count = 0u32; let targets = Validators::::iter() .map(|(v, _)| { validator_count.saturating_inc(); v }) .collect::>(); Self::register_weight(T::WeightInfo::get_npos_targets(validator_count)); targets } /// This function will add a nominator to the `Nominators` storage map, /// and [`SortedListProvider`]. /// /// If the nominator already exists, their nominations will be updated. /// /// NOTE: you must ALWAYS use this function to add nominator or update their targets. Any access /// to `Nominators` or `VoterList` outside of this function is almost certainly /// wrong. pub fn do_add_nominator(who: &T::AccountId, nominations: Nominations) { if !Nominators::::contains_key(who) { // maybe update sorted list. Error checking is defensive-only - this should never fail. let _ = T::SortedListProvider::on_insert(who.clone(), Self::weight_of(who)) .defensive_unwrap_or_default(); debug_assert_eq!(T::SortedListProvider::sanity_check(), Ok(())); } Nominators::::insert(who, nominations); } /// This function will remove a nominator from the `Nominators` storage map, /// and [`SortedListProvider`]. /// /// Returns true if `who` was removed from `Nominators`, otherwise false. /// /// NOTE: you must ALWAYS use this function to remove a nominator from the system. Any access to /// `Nominators` or `VoterList` outside of this function is almost certainly /// wrong. pub fn do_remove_nominator(who: &T::AccountId) -> bool { if Nominators::::contains_key(who) { Nominators::::remove(who); T::SortedListProvider::on_remove(who); debug_assert_eq!(T::SortedListProvider::sanity_check(), Ok(())); debug_assert_eq!(Nominators::::count(), T::SortedListProvider::count()); true } else { false } } /// This function will add a validator to the `Validators` storage map. /// /// If the validator already exists, their preferences will be updated. /// /// NOTE: you must ALWAYS use this function to add a validator to the system. Any access to /// `Validators` or `VoterList` outside of this function is almost certainly /// wrong. pub fn do_add_validator(who: &T::AccountId, prefs: ValidatorPrefs) { Validators::::insert(who, prefs); } /// This function will remove a validator from the `Validators` storage map. /// /// Returns true if `who` was removed from `Validators`, otherwise false. /// /// NOTE: you must ALWAYS use this function to remove a validator from the system. Any access to /// `Validators` or `VoterList` outside of this function is almost certainly /// wrong. pub fn do_remove_validator(who: &T::AccountId) -> bool { if Validators::::contains_key(who) { Validators::::remove(who); true } else { false } } /// Register some amount of weight directly with the system pallet. /// /// This is always mandatory weight. fn register_weight(weight: Weight) { >::register_extra_weight_unchecked( weight, DispatchClass::Mandatory, ); } } impl ElectionDataProvider for Pallet { type AccountId = T::AccountId; type BlockNumber = BlockNumberFor; type MaxVotesPerVoter = T::MaxNominations; fn desired_targets() -> data_provider::Result { Self::register_weight(T::DbWeight::get().reads(1)); Ok(Self::validator_count()) } fn voters(maybe_max_len: Option) -> data_provider::Result>> { // This can never fail -- if `maybe_max_len` is `Some(_)` we handle it. let voters = Self::get_npos_voters(maybe_max_len); debug_assert!(maybe_max_len.map_or(true, |max| voters.len() <= max)); Ok(voters) } fn targets(maybe_max_len: Option) -> data_provider::Result> { let target_count = Validators::::count(); // We can't handle this case yet -- return an error. if maybe_max_len.map_or(false, |max_len| target_count > max_len as u32) { return Err("Target snapshot too big") } Ok(Self::get_npos_targets()) } fn next_election_prediction(now: T::BlockNumber) -> T::BlockNumber { let current_era = Self::current_era().unwrap_or(0); let current_session = Self::current_planned_session(); let current_era_start_session_index = Self::eras_start_session_index(current_era).unwrap_or(0); // Number of session in the current era or the maximum session per era if reached. let era_progress = current_session .saturating_sub(current_era_start_session_index) .min(T::SessionsPerEra::get()); let until_this_session_end = T::NextNewSession::estimate_next_new_session(now) .0 .unwrap_or_default() .saturating_sub(now); let session_length = T::NextNewSession::average_session_length(); let sessions_left: T::BlockNumber = match ForceEra::::get() { Forcing::ForceNone => Bounded::max_value(), Forcing::ForceNew | Forcing::ForceAlways => Zero::zero(), Forcing::NotForcing if era_progress >= T::SessionsPerEra::get() => Zero::zero(), Forcing::NotForcing => T::SessionsPerEra::get() .saturating_sub(era_progress) // One session is computed in this_session_end. .saturating_sub(1) .into(), }; now.saturating_add( until_this_session_end.saturating_add(sessions_left.saturating_mul(session_length)), ) } #[cfg(feature = "runtime-benchmarks")] fn add_voter( voter: T::AccountId, weight: VoteWeight, targets: BoundedVec, ) { let stake = >::try_from(weight).unwrap_or_else(|_| { panic!("cannot convert a VoteWeight into BalanceOf, benchmark needs reconfiguring.") }); >::insert(voter.clone(), voter.clone()); >::insert( voter.clone(), StakingLedger { stash: voter.clone(), active: stake, total: stake, unlocking: Default::default(), claimed_rewards: vec![], }, ); Self::do_add_nominator(&voter, Nominations { targets, submitted_in: 0, suppressed: false }); } #[cfg(feature = "runtime-benchmarks")] fn add_target(target: T::AccountId) { let stake = MinValidatorBond::::get() * 100u32.into(); >::insert(target.clone(), target.clone()); >::insert( target.clone(), StakingLedger { stash: target.clone(), active: stake, total: stake, unlocking: Default::default(), claimed_rewards: vec![], }, ); Self::do_add_validator( &target, ValidatorPrefs { commission: Perbill::zero(), blocked: false }, ); } #[cfg(feature = "runtime-benchmarks")] fn clear() { >::remove_all(None); >::remove_all(None); >::remove_all(); >::remove_all(); T::SortedListProvider::unsafe_clear(); } #[cfg(feature = "runtime-benchmarks")] fn put_snapshot( voters: Vec>, targets: Vec, target_stake: Option, ) { targets.into_iter().for_each(|v| { let stake: BalanceOf = target_stake .and_then(|w| >::try_from(w).ok()) .unwrap_or_else(|| MinNominatorBond::::get() * 100u32.into()); >::insert(v.clone(), v.clone()); >::insert( v.clone(), StakingLedger { stash: v.clone(), active: stake, total: stake, unlocking: Default::default(), claimed_rewards: vec![], }, ); Self::do_add_validator( &v, ValidatorPrefs { commission: Perbill::zero(), blocked: false }, ); }); voters.into_iter().for_each(|(v, s, t)| { let stake = >::try_from(s).unwrap_or_else(|_| { panic!("cannot convert a VoteWeight into BalanceOf, benchmark needs reconfiguring.") }); >::insert(v.clone(), v.clone()); >::insert( v.clone(), StakingLedger { stash: v.clone(), active: stake, total: stake, unlocking: Default::default(), claimed_rewards: vec![], }, ); Self::do_add_nominator( &v, Nominations { targets: t.try_into().unwrap(), submitted_in: 0, suppressed: false }, ); }); } } /// In this implementation `new_session(session)` must be called before `end_session(session-1)` /// i.e. the new session must be planned before the ending of the previous session. /// /// Once the first new_session is planned, all session must start and then end in order, though /// some session can lag in between the newest session planned and the latest session started. impl pallet_session::SessionManager for Pallet { fn new_session(new_index: SessionIndex) -> Option> { log!(trace, "planning new session {}", new_index); CurrentPlannedSession::::put(new_index); Self::new_session(new_index, false) } fn new_session_genesis(new_index: SessionIndex) -> Option> { log!(trace, "planning new session {} at genesis", new_index); CurrentPlannedSession::::put(new_index); Self::new_session(new_index, true) } fn start_session(start_index: SessionIndex) { log!(trace, "starting session {}", start_index); Self::start_session(start_index) } fn end_session(end_index: SessionIndex) { log!(trace, "ending session {}", end_index); Self::end_session(end_index) } } impl historical::SessionManager>> for Pallet { fn new_session( new_index: SessionIndex, ) -> Option>)>> { >::new_session(new_index).map(|validators| { let current_era = Self::current_era() // Must be some as a new era has been created. .unwrap_or(0); validators .into_iter() .map(|v| { let exposure = Self::eras_stakers(current_era, &v); (v, exposure) }) .collect() }) } fn new_session_genesis( new_index: SessionIndex, ) -> Option>)>> { >::new_session_genesis(new_index).map( |validators| { let current_era = Self::current_era() // Must be some as a new era has been created. .unwrap_or(0); validators .into_iter() .map(|v| { let exposure = Self::eras_stakers(current_era, &v); (v, exposure) }) .collect() }, ) } fn start_session(start_index: SessionIndex) { >::start_session(start_index) } fn end_session(end_index: SessionIndex) { >::end_session(end_index) } } /// Add reward points to block authors: /// * 20 points to the block producer for producing a (non-uncle) block in the relay chain, /// * 2 points to the block producer for each reference to a previously unreferenced uncle, and /// * 1 point to the producer of each referenced uncle block. impl pallet_authorship::EventHandler for Pallet where T: Config + pallet_authorship::Config + pallet_session::Config, { fn note_author(author: T::AccountId) { Self::reward_by_ids(vec![(author, 20)]) } fn note_uncle(uncle_author: T::AccountId, _age: T::BlockNumber) { // defensive-only: block author must exist. if let Some(block_author) = >::author() { Self::reward_by_ids(vec![(block_author, 2), (uncle_author, 1)]) } else { crate::log!(warn, "block author not set, this should never happen"); } } } /// This is intended to be used with `FilterHistoricalOffences`. impl OnOffenceHandler, Weight> for Pallet where T: pallet_session::Config::AccountId>, T: pallet_session::historical::Config< FullIdentification = Exposure<::AccountId, BalanceOf>, FullIdentificationOf = ExposureOf, >, T::SessionHandler: pallet_session::SessionHandler<::AccountId>, T::SessionManager: pallet_session::SessionManager<::AccountId>, T::ValidatorIdOf: Convert< ::AccountId, Option<::AccountId>, >, { fn on_offence( offenders: &[OffenceDetails< T::AccountId, pallet_session::historical::IdentificationTuple, >], slash_fraction: &[Perbill], slash_session: SessionIndex, disable_strategy: DisableStrategy, ) -> Weight { let reward_proportion = SlashRewardFraction::::get(); let mut consumed_weight: Weight = 0; let mut add_db_reads_writes = |reads, writes| { consumed_weight += T::DbWeight::get().reads_writes(reads, writes); }; let active_era = { let active_era = Self::active_era(); add_db_reads_writes(1, 0); if active_era.is_none() { // This offence need not be re-submitted. return consumed_weight } active_era.expect("value checked not to be `None`; qed").index }; let active_era_start_session_index = Self::eras_start_session_index(active_era) .unwrap_or_else(|| { frame_support::print("Error: start_session_index must be set for current_era"); 0 }); add_db_reads_writes(1, 0); let window_start = active_era.saturating_sub(T::BondingDuration::get()); // Fast path for active-era report - most likely. // `slash_session` cannot be in a future active era. It must be in `active_era` or before. let slash_era = if slash_session >= active_era_start_session_index { active_era } else { let eras = BondedEras::::get(); add_db_reads_writes(1, 0); // Reverse because it's more likely to find reports from recent eras. match eras.iter().rev().find(|&&(_, ref sesh)| sesh <= &slash_session) { Some(&(ref slash_era, _)) => *slash_era, // Before bonding period. defensive - should be filtered out. None => return consumed_weight, } }; ::EarliestUnappliedSlash::mutate(|earliest| { if earliest.is_none() { *earliest = Some(active_era) } }); add_db_reads_writes(1, 1); let slash_defer_duration = T::SlashDeferDuration::get(); let invulnerables = Self::invulnerables(); add_db_reads_writes(1, 0); for (details, slash_fraction) in offenders.iter().zip(slash_fraction) { let (stash, exposure) = &details.offender; // Skip if the validator is invulnerable. if invulnerables.contains(stash) { continue } let unapplied = slashing::compute_slash::(slashing::SlashParams { stash, slash: *slash_fraction, exposure, slash_era, window_start, now: active_era, reward_proportion, disable_strategy, }); if let Some(mut unapplied) = unapplied { let nominators_len = unapplied.others.len() as u64; let reporters_len = details.reporters.len() as u64; { let upper_bound = 1 /* Validator/NominatorSlashInEra */ + 2 /* fetch_spans */; let rw = upper_bound + nominators_len * upper_bound; add_db_reads_writes(rw, rw); } unapplied.reporters = details.reporters.clone(); if slash_defer_duration == 0 { // Apply right away. slashing::apply_slash::(unapplied); { let slash_cost = (6, 5); let reward_cost = (2, 2); add_db_reads_writes( (1 + nominators_len) * slash_cost.0 + reward_cost.0 * reporters_len, (1 + nominators_len) * slash_cost.1 + reward_cost.1 * reporters_len, ); } } else { // Defer to end of some `slash_defer_duration` from now. ::UnappliedSlashes::mutate(active_era, move |for_later| { for_later.push(unapplied) }); add_db_reads_writes(1, 1); } } else { add_db_reads_writes(4 /* fetch_spans */, 5 /* kick_out_if_recent */) } } consumed_weight } } impl VoteWeightProvider for Pallet { fn vote_weight(who: &T::AccountId) -> VoteWeight { Self::weight_of(who) } #[cfg(feature = "runtime-benchmarks")] fn set_vote_weight_of(who: &T::AccountId, weight: VoteWeight) { // this will clearly results in an inconsistent state, but it should not matter for a // benchmark. let active: BalanceOf = weight.try_into().map_err(|_| ()).unwrap(); let mut ledger = match Self::ledger(who) { None => StakingLedger::default_from(who.clone()), Some(l) => l, }; ledger.active = active; >::insert(who, ledger); >::insert(who, who); // also, we play a trick to make sure that a issuance based-`CurrencyToVote` behaves well: // This will make sure that total issuance is zero, thus the currency to vote will be a 1-1 // conversion. let imbalance = T::Currency::burn(T::Currency::total_issuance()); // kinda ugly, but gets the job done. The fact that this works here is a HUGE exception. // Don't try this pattern in other places. sp_std::mem::forget(imbalance); } } /// A simple voter list implementation that does not require any additional pallets. Note, this /// does not provided nominators in sorted ordered. If you desire nominators in a sorted order take /// a look at [`pallet-bags-list]. pub struct UseNominatorsMap(sp_std::marker::PhantomData); impl SortedListProvider for UseNominatorsMap { type Error = (); /// Returns iterator over voter list, which can have `take` called on it. fn iter() -> Box> { Box::new(Nominators::::iter().map(|(n, _)| n)) } fn count() -> u32 { Nominators::::count() } fn contains(id: &T::AccountId) -> bool { Nominators::::contains_key(id) } fn on_insert(_: T::AccountId, _weight: VoteWeight) -> Result<(), Self::Error> { // nothing to do on insert. Ok(()) } fn on_update(_: &T::AccountId, _weight: VoteWeight) { // nothing to do on update. } fn on_remove(_: &T::AccountId) { // nothing to do on remove. } fn unsafe_regenerate( _: impl IntoIterator, _: Box VoteWeight>, ) -> u32 { // nothing to do upon regenerate. 0 } fn sanity_check() -> Result<(), &'static str> { Ok(()) } fn unsafe_clear() { // NOTE: Caller must ensure this doesn't lead to too many storage accesses. This is a // condition of SortedListProvider::unsafe_clear. Nominators::::remove_all(); } }