mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-07 10:48:04 +00:00
1b5f4243d1
This PR removes sp-std crate from substrate/primitives sub-directories. For now crates that have `pub use` of sp-std or export macros that would necessitate users of the macros to `extern crate alloc` have been excluded from this PR. There should be no breaking changes in this PR. --------- Co-authored-by: Koute <koute@users.noreply.github.com>
426 lines
15 KiB
Rust
426 lines
15 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 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.
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
//! A crate which contains primitives that are useful for implementation that uses staking
|
|
//! approaches in general. Definitions related to sessions, slashing, etc go here.
|
|
|
|
extern crate alloc;
|
|
|
|
use crate::currency_to_vote::CurrencyToVote;
|
|
use alloc::{collections::btree_map::BTreeMap, vec, vec::Vec};
|
|
use codec::{Decode, Encode, FullCodec, HasCompact, MaxEncodedLen};
|
|
use core::ops::Sub;
|
|
use scale_info::TypeInfo;
|
|
use sp_runtime::{
|
|
traits::{AtLeast32BitUnsigned, Zero},
|
|
DispatchError, DispatchResult, RuntimeDebug, Saturating,
|
|
};
|
|
|
|
pub mod offence;
|
|
|
|
pub mod currency_to_vote;
|
|
|
|
/// Simple index type with which we can count sessions.
|
|
pub type SessionIndex = u32;
|
|
|
|
/// Counter for the number of eras that have passed.
|
|
pub type EraIndex = u32;
|
|
|
|
/// Type for identifying a page.
|
|
pub type Page = u32;
|
|
/// Representation of a staking account, which may be a stash or controller account.
|
|
///
|
|
/// Note: once the controller is completely deprecated, this enum can also be deprecated in favor of
|
|
/// the stash account. Tracking issue: <https://github.com/paritytech/substrate/issues/6927>.
|
|
#[derive(Clone, Debug)]
|
|
pub enum StakingAccount<AccountId> {
|
|
Stash(AccountId),
|
|
Controller(AccountId),
|
|
}
|
|
|
|
#[cfg(feature = "std")]
|
|
impl<AccountId> From<AccountId> for StakingAccount<AccountId> {
|
|
fn from(account: AccountId) -> Self {
|
|
StakingAccount::Stash(account)
|
|
}
|
|
}
|
|
|
|
/// Representation of the status of a staker.
|
|
#[derive(RuntimeDebug, TypeInfo)]
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize, PartialEq, Eq, Clone))]
|
|
pub enum StakerStatus<AccountId> {
|
|
/// Chilling.
|
|
Idle,
|
|
/// Declaring desire in validate, i.e author blocks.
|
|
Validator,
|
|
/// Declaring desire to nominate, delegate, or generally approve of the given set of others.
|
|
Nominator(Vec<AccountId>),
|
|
}
|
|
|
|
/// A struct that reflects stake that an account has in the staking system. Provides a set of
|
|
/// methods to operate on it's properties. Aimed at making `StakingInterface` more concise.
|
|
#[derive(RuntimeDebug, Clone, Copy, Eq, PartialEq, Default)]
|
|
pub struct Stake<Balance> {
|
|
/// The total stake that `stash` has in the staking system. This includes the
|
|
/// `active` stake, and any funds currently in the process of unbonding via
|
|
/// [`StakingInterface::unbond`].
|
|
///
|
|
/// # Note
|
|
///
|
|
/// This is only guaranteed to reflect the amount locked by the staking system. If there are
|
|
/// non-staking locks on the bonded pair's balance this amount is going to be larger in
|
|
/// reality.
|
|
pub total: Balance,
|
|
/// The total amount of the stash's balance that will be at stake in any forthcoming
|
|
/// rounds.
|
|
pub active: Balance,
|
|
}
|
|
|
|
/// A generic staking event listener.
|
|
///
|
|
/// Note that the interface is designed in a way that the events are fired post-action, so any
|
|
/// pre-action data that is needed needs to be passed to interface methods. The rest of the data can
|
|
/// be retrieved by using `StakingInterface`.
|
|
#[impl_trait_for_tuples::impl_for_tuples(10)]
|
|
pub trait OnStakingUpdate<AccountId, Balance> {
|
|
/// Fired when the stake amount of someone updates.
|
|
///
|
|
/// This is effectively any changes to the bond amount, such as bonding more funds, and
|
|
/// unbonding.
|
|
fn on_stake_update(_who: &AccountId, _prev_stake: Option<Stake<Balance>>) {}
|
|
|
|
/// Fired when someone sets their intention to nominate.
|
|
///
|
|
/// This should never be fired for existing nominators.
|
|
fn on_nominator_add(_who: &AccountId) {}
|
|
|
|
/// Fired when an existing nominator updates their nominations.
|
|
///
|
|
/// Note that this is not fired when a nominator changes their stake. For that,
|
|
/// `on_stake_update` should be used, followed by querying whether `who` was a validator or a
|
|
/// nominator.
|
|
fn on_nominator_update(_who: &AccountId, _prev_nominations: Vec<AccountId>) {}
|
|
|
|
/// Fired when someone removes their intention to nominate, either due to chill or validating.
|
|
///
|
|
/// The set of nominations at the time of removal is provided as it can no longer be fetched in
|
|
/// any way.
|
|
fn on_nominator_remove(_who: &AccountId, _nominations: Vec<AccountId>) {}
|
|
|
|
/// Fired when someone sets their intention to validate.
|
|
///
|
|
/// Note validator preference changes are not communicated, but could be added if needed.
|
|
fn on_validator_add(_who: &AccountId) {}
|
|
|
|
/// Fired when an existing validator updates their preferences.
|
|
///
|
|
/// Note validator preference changes are not communicated, but could be added if needed.
|
|
fn on_validator_update(_who: &AccountId) {}
|
|
|
|
/// Fired when someone removes their intention to validate, either due to chill or nominating.
|
|
fn on_validator_remove(_who: &AccountId) {}
|
|
|
|
/// Fired when someone is fully unstaked.
|
|
fn on_unstake(_who: &AccountId) {}
|
|
|
|
/// Fired when a staker is slashed.
|
|
///
|
|
/// * `stash` - The stash of the staker whom the slash was applied to.
|
|
/// * `slashed_active` - The new bonded balance of the staker after the slash was applied.
|
|
/// * `slashed_unlocking` - A map of slashed eras, and the balance of that unlocking chunk after
|
|
/// the slash is applied. Any era not present in the map is not affected at all.
|
|
/// * `slashed_total` - The aggregated balance that was lost due to the slash.
|
|
fn on_slash(
|
|
_stash: &AccountId,
|
|
_slashed_active: Balance,
|
|
_slashed_unlocking: &BTreeMap<EraIndex, Balance>,
|
|
_slashed_total: Balance,
|
|
) {
|
|
}
|
|
|
|
/// Fired when a portion of a staker's balance has been withdrawn.
|
|
fn on_withdraw(_stash: &AccountId, _amount: Balance) {}
|
|
}
|
|
|
|
/// A generic representation of a staking implementation.
|
|
///
|
|
/// This interface uses the terminology of NPoS, but it is aims to be generic enough to cover other
|
|
/// implementations as well.
|
|
pub trait StakingInterface {
|
|
/// Balance type used by the staking system.
|
|
type Balance: Sub<Output = Self::Balance>
|
|
+ Ord
|
|
+ PartialEq
|
|
+ Default
|
|
+ Copy
|
|
+ MaxEncodedLen
|
|
+ FullCodec
|
|
+ TypeInfo
|
|
+ Saturating;
|
|
|
|
/// AccountId type used by the staking system.
|
|
type AccountId: Clone + core::fmt::Debug;
|
|
|
|
/// Means of converting Currency to VoteWeight.
|
|
type CurrencyToVote: CurrencyToVote<Self::Balance>;
|
|
|
|
/// The minimum amount required to bond in order to set nomination intentions. This does not
|
|
/// necessarily mean the nomination will be counted in an election, but instead just enough to
|
|
/// be stored as a nominator. In other words, this is the minimum amount to register the
|
|
/// intention to nominate.
|
|
fn minimum_nominator_bond() -> Self::Balance;
|
|
|
|
/// The minimum amount required to bond in order to set validation intentions.
|
|
fn minimum_validator_bond() -> Self::Balance;
|
|
|
|
/// Return a stash account that is controlled by a `controller`.
|
|
///
|
|
/// ## Note
|
|
///
|
|
/// The controller abstraction is not permanent and might go away. Avoid using this as much as
|
|
/// possible.
|
|
fn stash_by_ctrl(controller: &Self::AccountId) -> Result<Self::AccountId, DispatchError>;
|
|
|
|
/// Number of eras that staked funds must remain bonded for.
|
|
fn bonding_duration() -> EraIndex;
|
|
|
|
/// The current era index.
|
|
///
|
|
/// This should be the latest planned era that the staking system knows about.
|
|
fn current_era() -> EraIndex;
|
|
|
|
/// Returns the [`Stake`] of `who`.
|
|
fn stake(who: &Self::AccountId) -> Result<Stake<Self::Balance>, DispatchError>;
|
|
|
|
/// Total stake of a staker, `Err` if not a staker.
|
|
fn total_stake(who: &Self::AccountId) -> Result<Self::Balance, DispatchError> {
|
|
Self::stake(who).map(|s| s.total)
|
|
}
|
|
|
|
/// Total active portion of a staker's [`Stake`], `Err` if not a staker.
|
|
fn active_stake(who: &Self::AccountId) -> Result<Self::Balance, DispatchError> {
|
|
Self::stake(who).map(|s| s.active)
|
|
}
|
|
|
|
/// Returns whether a staker is unbonding, `Err` if not a staker at all.
|
|
fn is_unbonding(who: &Self::AccountId) -> Result<bool, DispatchError> {
|
|
Self::stake(who).map(|s| s.active != s.total)
|
|
}
|
|
|
|
/// Returns whether a staker is FULLY unbonding, `Err` if not a staker at all.
|
|
fn fully_unbond(who: &Self::AccountId) -> DispatchResult {
|
|
Self::unbond(who, Self::stake(who)?.active)
|
|
}
|
|
|
|
/// Bond (lock) `value` of `who`'s balance, while forwarding any rewards to `payee`.
|
|
fn bond(who: &Self::AccountId, value: Self::Balance, payee: &Self::AccountId)
|
|
-> DispatchResult;
|
|
|
|
/// Have `who` nominate `validators`.
|
|
fn nominate(who: &Self::AccountId, validators: Vec<Self::AccountId>) -> DispatchResult;
|
|
|
|
/// Chill `who`.
|
|
fn chill(who: &Self::AccountId) -> DispatchResult;
|
|
|
|
/// Bond some extra amount in `who`'s free balance against the active bonded balance of
|
|
/// the account. The amount extra actually bonded will never be more than `who`'s free
|
|
/// balance.
|
|
fn bond_extra(who: &Self::AccountId, extra: Self::Balance) -> DispatchResult;
|
|
|
|
/// Schedule a portion of the active bonded balance to be unlocked at era
|
|
/// [Self::current_era] + [`Self::bonding_duration`].
|
|
///
|
|
/// Once the unlock era has been reached, [`Self::withdraw_unbonded`] can be called to unlock
|
|
/// the funds.
|
|
///
|
|
/// The amount of times this can be successfully called is limited based on how many distinct
|
|
/// eras funds are schedule to unlock in. Calling [`Self::withdraw_unbonded`] after some unlock
|
|
/// schedules have reached their unlocking era should allow more calls to this function.
|
|
fn unbond(stash: &Self::AccountId, value: Self::Balance) -> DispatchResult;
|
|
|
|
/// Unlock any funds schedule to unlock before or at the current era.
|
|
///
|
|
/// Returns whether the stash was killed because of this withdraw or not.
|
|
fn withdraw_unbonded(
|
|
stash: Self::AccountId,
|
|
num_slashing_spans: u32,
|
|
) -> Result<bool, DispatchError>;
|
|
|
|
/// The ideal number of active validators.
|
|
fn desired_validator_count() -> u32;
|
|
|
|
/// Whether or not there is an ongoing election.
|
|
fn election_ongoing() -> bool;
|
|
|
|
/// Force a current staker to become completely unstaked, immediately.
|
|
fn force_unstake(who: Self::AccountId) -> DispatchResult;
|
|
|
|
/// Checks whether an account `staker` has been exposed in an era.
|
|
fn is_exposed_in_era(who: &Self::AccountId, era: &EraIndex) -> bool;
|
|
|
|
/// Return the status of the given staker, `None` if not staked at all.
|
|
fn status(who: &Self::AccountId) -> Result<StakerStatus<Self::AccountId>, DispatchError>;
|
|
|
|
/// Checks whether or not this is a validator account.
|
|
fn is_validator(who: &Self::AccountId) -> bool {
|
|
Self::status(who).map(|s| matches!(s, StakerStatus::Validator)).unwrap_or(false)
|
|
}
|
|
|
|
/// Get the nominations of a stash, if they are a nominator, `None` otherwise.
|
|
fn nominations(who: &Self::AccountId) -> Option<Vec<Self::AccountId>> {
|
|
match Self::status(who) {
|
|
Ok(StakerStatus::Nominator(t)) => Some(t),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
fn max_exposure_page_size() -> Page;
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
fn add_era_stakers(
|
|
current_era: &EraIndex,
|
|
stash: &Self::AccountId,
|
|
exposures: Vec<(Self::AccountId, Self::Balance)>,
|
|
);
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
fn set_current_era(era: EraIndex);
|
|
}
|
|
|
|
/// The amount of exposure for an era that an individual nominator has (susceptible to slashing).
|
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Encode, Decode, RuntimeDebug, TypeInfo)]
|
|
pub struct IndividualExposure<AccountId, Balance: HasCompact> {
|
|
/// The stash account of the nominator in question.
|
|
pub who: AccountId,
|
|
/// Amount of funds exposed.
|
|
#[codec(compact)]
|
|
pub value: Balance,
|
|
}
|
|
|
|
/// A snapshot of the stake backing a single validator in the system.
|
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Encode, Decode, RuntimeDebug, TypeInfo)]
|
|
pub struct Exposure<AccountId, Balance: HasCompact> {
|
|
/// The total balance backing this validator.
|
|
#[codec(compact)]
|
|
pub total: Balance,
|
|
/// The validator's own stash that is exposed.
|
|
#[codec(compact)]
|
|
pub own: Balance,
|
|
/// The portions of nominators stashes that are exposed.
|
|
pub others: Vec<IndividualExposure<AccountId, Balance>>,
|
|
}
|
|
|
|
impl<AccountId, Balance: Default + HasCompact> Default for Exposure<AccountId, Balance> {
|
|
fn default() -> Self {
|
|
Self { total: Default::default(), own: Default::default(), others: vec![] }
|
|
}
|
|
}
|
|
|
|
impl<
|
|
AccountId: Clone,
|
|
Balance: HasCompact + AtLeast32BitUnsigned + Copy + codec::MaxEncodedLen,
|
|
> Exposure<AccountId, Balance>
|
|
{
|
|
/// Splits an `Exposure` into `PagedExposureMetadata` and multiple chunks of
|
|
/// `IndividualExposure` with each chunk having maximum of `page_size` elements.
|
|
pub fn into_pages(
|
|
self,
|
|
page_size: Page,
|
|
) -> (PagedExposureMetadata<Balance>, Vec<ExposurePage<AccountId, Balance>>) {
|
|
let individual_chunks = self.others.chunks(page_size as usize);
|
|
let mut exposure_pages: Vec<ExposurePage<AccountId, Balance>> =
|
|
Vec::with_capacity(individual_chunks.len());
|
|
|
|
for chunk in individual_chunks {
|
|
let mut page_total: Balance = Zero::zero();
|
|
let mut others: Vec<IndividualExposure<AccountId, Balance>> =
|
|
Vec::with_capacity(chunk.len());
|
|
for individual in chunk.iter() {
|
|
page_total.saturating_accrue(individual.value);
|
|
others.push(IndividualExposure {
|
|
who: individual.who.clone(),
|
|
value: individual.value,
|
|
})
|
|
}
|
|
|
|
exposure_pages.push(ExposurePage { page_total, others });
|
|
}
|
|
|
|
(
|
|
PagedExposureMetadata {
|
|
total: self.total,
|
|
own: self.own,
|
|
nominator_count: self.others.len() as u32,
|
|
page_count: exposure_pages.len() as Page,
|
|
},
|
|
exposure_pages,
|
|
)
|
|
}
|
|
}
|
|
|
|
/// A snapshot of the stake backing a single validator in the system.
|
|
#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Encode, Decode, RuntimeDebug, TypeInfo)]
|
|
pub struct ExposurePage<AccountId, Balance: HasCompact> {
|
|
/// The total balance of this chunk/page.
|
|
#[codec(compact)]
|
|
pub page_total: Balance,
|
|
/// The portions of nominators stashes that are exposed.
|
|
pub others: Vec<IndividualExposure<AccountId, Balance>>,
|
|
}
|
|
|
|
impl<A, B: Default + HasCompact> Default for ExposurePage<A, B> {
|
|
fn default() -> Self {
|
|
ExposurePage { page_total: Default::default(), others: vec![] }
|
|
}
|
|
}
|
|
|
|
/// Metadata for Paged Exposure of a validator such as total stake across pages and page count.
|
|
///
|
|
/// In combination with the associated `ExposurePage`s, it can be used to reconstruct a full
|
|
/// `Exposure` set of a validator. This is useful for cases where we want to query full set of
|
|
/// `Exposure` as one page (for backward compatibility).
|
|
#[derive(
|
|
PartialEq,
|
|
Eq,
|
|
PartialOrd,
|
|
Ord,
|
|
Clone,
|
|
Encode,
|
|
Decode,
|
|
RuntimeDebug,
|
|
TypeInfo,
|
|
Default,
|
|
MaxEncodedLen,
|
|
)]
|
|
pub struct PagedExposureMetadata<Balance: HasCompact + codec::MaxEncodedLen> {
|
|
/// The total balance backing this validator.
|
|
#[codec(compact)]
|
|
pub total: Balance,
|
|
/// The validator's own stash that is exposed.
|
|
#[codec(compact)]
|
|
pub own: Balance,
|
|
/// Number of nominators backing this validator.
|
|
pub nominator_count: u32,
|
|
/// Number of pages of nominators.
|
|
pub page_count: Page,
|
|
}
|
|
|
|
sp_core::generate_feature_enabled_macro!(runtime_benchmarks_enabled, feature = "runtime-benchmarks", $);
|