mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-28 01:38:04 +00:00
002d9260f9
**Update:** Pushed additional changes based on the review comments. **This pull request fixes various spelling mistakes in this repository.** Most of the changes are contained in the first **3** commits: - `Fix spelling mistakes in comments and docs` - `Fix spelling mistakes in test names` - `Fix spelling mistakes in error messages, panic messages, logs and tracing` Other source code spelling mistakes are separated into individual commits for easier reviewing: - `Fix the spelling of 'authority'` - `Fix the spelling of 'REASONABLE_HEADERS_IN_JUSTIFICATION_ANCESTRY'` - `Fix the spelling of 'prev_enqueud_messages'` - `Fix the spelling of 'endpoint'` - `Fix the spelling of 'children'` - `Fix the spelling of 'PenpalSiblingSovereignAccount'` - `Fix the spelling of 'PenpalSudoAccount'` - `Fix the spelling of 'insufficient'` - `Fix the spelling of 'PalletXcmExtrinsicsBenchmark'` - `Fix the spelling of 'subtracted'` - `Fix the spelling of 'CandidatePendingAvailability'` - `Fix the spelling of 'exclusive'` - `Fix the spelling of 'until'` - `Fix the spelling of 'discriminator'` - `Fix the spelling of 'nonexistent'` - `Fix the spelling of 'subsystem'` - `Fix the spelling of 'indices'` - `Fix the spelling of 'committed'` - `Fix the spelling of 'topology'` - `Fix the spelling of 'response'` - `Fix the spelling of 'beneficiary'` - `Fix the spelling of 'formatted'` - `Fix the spelling of 'UNKNOWN_PROOF_REQUEST'` - `Fix the spelling of 'succeeded'` - `Fix the spelling of 'reopened'` - `Fix the spelling of 'proposer'` - `Fix the spelling of 'InstantiationNonce'` - `Fix the spelling of 'depositor'` - `Fix the spelling of 'expiration'` - `Fix the spelling of 'phantom'` - `Fix the spelling of 'AggregatedKeyValue'` - `Fix the spelling of 'randomness'` - `Fix the spelling of 'defendant'` - `Fix the spelling of 'AquaticMammal'` - `Fix the spelling of 'transactions'` - `Fix the spelling of 'PassingTracingSubscriber'` - `Fix the spelling of 'TxSignaturePayload'` - `Fix the spelling of 'versioning'` - `Fix the spelling of 'descendant'` - `Fix the spelling of 'overridden'` - `Fix the spelling of 'network'` Let me know if this structure is adequate. **Note:** The usage of the words `Merkle`, `Merkelize`, `Merklization`, `Merkelization`, `Merkleization`, is somewhat inconsistent but I left it as it is. ~~**Note:** In some places the term `Receival` is used to refer to message reception, IMO `Reception` is the correct word here, but I left it as it is.~~ ~~**Note:** In some places the term `Overlayed` is used instead of the more acceptable version `Overlaid` but I also left it as it is.~~ ~~**Note:** In some places the term `Applyable` is used instead of the correct version `Applicable` but I also left it as it is.~~ **Note:** Some usage of British vs American english e.g. `judgement` vs `judgment`, `initialise` vs `initialize`, `optimise` vs `optimize` etc. are both present in different places, but I suppose that's understandable given the number of contributors. ~~**Note:** There is a spelling mistake in `.github/CODEOWNERS` but it triggers errors in CI when I make changes to it, so I left it as it is.~~
300 lines
11 KiB
Rust
300 lines
11 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.
|
|
|
|
//! A Ledger implementation for stakers.
|
|
//!
|
|
//! A [`StakingLedger`] encapsulates all the state and logic related to the stake of bonded
|
|
//! stakers, namely, it handles the following storage items:
|
|
//! * [`Bonded`]: mutates and reads the state of the controller <> stash bond map (to be deprecated
|
|
//! soon);
|
|
//! * [`Ledger`]: mutates and reads the state of all the stakers. The [`Ledger`] storage item stores
|
|
//! instances of [`StakingLedger`] keyed by the staker's controller account and should be mutated
|
|
//! and read through the [`StakingLedger`] API;
|
|
//! * [`Payee`]: mutates and reads the reward destination preferences for a bonded stash.
|
|
//! * Staking locks: mutates the locks for staking.
|
|
//!
|
|
//! NOTE: All the storage operations related to the staking ledger (both reads and writes) *MUST* be
|
|
//! performed through the methods exposed by the [`StakingLedger`] implementation in order to ensure
|
|
//! state consistency.
|
|
|
|
use frame_support::{
|
|
defensive, ensure,
|
|
traits::{Defensive, LockableCurrency, WithdrawReasons},
|
|
};
|
|
use sp_staking::StakingAccount;
|
|
use sp_std::prelude::*;
|
|
|
|
use crate::{
|
|
BalanceOf, Bonded, Config, Error, Ledger, Payee, RewardDestination, StakingLedger, STAKING_ID,
|
|
};
|
|
|
|
#[cfg(any(feature = "runtime-benchmarks", test))]
|
|
use sp_runtime::traits::Zero;
|
|
|
|
impl<T: Config> StakingLedger<T> {
|
|
#[cfg(any(feature = "runtime-benchmarks", test))]
|
|
pub fn default_from(stash: T::AccountId) -> Self {
|
|
Self {
|
|
stash: stash.clone(),
|
|
total: Zero::zero(),
|
|
active: Zero::zero(),
|
|
unlocking: Default::default(),
|
|
legacy_claimed_rewards: Default::default(),
|
|
controller: Some(stash),
|
|
}
|
|
}
|
|
|
|
/// Returns a new instance of a staking ledger.
|
|
///
|
|
/// The [`Ledger`] storage is not mutated. In order to store, `StakingLedger::update` must be
|
|
/// called on the returned staking ledger.
|
|
///
|
|
/// Note: as the controller accounts are being deprecated, the stash account is the same as the
|
|
/// controller account.
|
|
pub fn new(stash: T::AccountId, stake: BalanceOf<T>) -> Self {
|
|
Self {
|
|
stash: stash.clone(),
|
|
active: stake,
|
|
total: stake,
|
|
unlocking: Default::default(),
|
|
legacy_claimed_rewards: Default::default(),
|
|
// controllers are deprecated and mapped 1-1 to stashes.
|
|
controller: Some(stash),
|
|
}
|
|
}
|
|
|
|
/// Returns the paired account, if any.
|
|
///
|
|
/// A "pair" refers to the tuple (stash, controller). If the input is a
|
|
/// [`StakingAccount::Stash`] variant, its pair account will be of type
|
|
/// [`StakingAccount::Controller`] and vice-versa.
|
|
///
|
|
/// This method is meant to abstract from the runtime development the difference between stash
|
|
/// and controller. This will be deprecated once the controller is fully deprecated as well.
|
|
pub(crate) fn paired_account(account: StakingAccount<T::AccountId>) -> Option<T::AccountId> {
|
|
match account {
|
|
StakingAccount::Stash(stash) => <Bonded<T>>::get(stash),
|
|
StakingAccount::Controller(controller) =>
|
|
<Ledger<T>>::get(&controller).map(|ledger| ledger.stash),
|
|
}
|
|
}
|
|
|
|
/// Returns whether a given account is bonded.
|
|
pub(crate) fn is_bonded(account: StakingAccount<T::AccountId>) -> bool {
|
|
match account {
|
|
StakingAccount::Stash(stash) => <Bonded<T>>::contains_key(stash),
|
|
StakingAccount::Controller(controller) => <Ledger<T>>::contains_key(controller),
|
|
}
|
|
}
|
|
|
|
/// Returns a staking ledger, if it is bonded and it exists in storage.
|
|
///
|
|
/// This getter can be called with either a controller or stash account, provided that the
|
|
/// account is properly wrapped in the respective [`StakingAccount`] variant. This is meant to
|
|
/// abstract the concept of controller/stash accounts from the caller.
|
|
///
|
|
/// Returns [`Error::BadState`] when a bond is in "bad state". A bond is in a bad state when a
|
|
/// stash has a controller which is bonding a ledger associated with another stash.
|
|
pub(crate) fn get(account: StakingAccount<T::AccountId>) -> Result<StakingLedger<T>, Error<T>> {
|
|
let (stash, controller) = match account.clone() {
|
|
StakingAccount::Stash(stash) =>
|
|
(stash.clone(), <Bonded<T>>::get(&stash).ok_or(Error::<T>::NotStash)?),
|
|
StakingAccount::Controller(controller) => (
|
|
Ledger::<T>::get(&controller)
|
|
.map(|l| l.stash)
|
|
.ok_or(Error::<T>::NotController)?,
|
|
controller,
|
|
),
|
|
};
|
|
|
|
let ledger = <Ledger<T>>::get(&controller)
|
|
.map(|mut ledger| {
|
|
ledger.controller = Some(controller.clone());
|
|
ledger
|
|
})
|
|
.ok_or(Error::<T>::NotController)?;
|
|
|
|
// if ledger bond is in a bad state, return error to prevent applying operations that may
|
|
// further spoil the ledger's state. A bond is in bad state when the bonded controller is
|
|
// associated with a different ledger (i.e. a ledger with a different stash).
|
|
//
|
|
// See <https://github.com/paritytech/polkadot-sdk/issues/3245> for more details.
|
|
ensure!(
|
|
Bonded::<T>::get(&stash) == Some(controller) && ledger.stash == stash,
|
|
Error::<T>::BadState
|
|
);
|
|
|
|
Ok(ledger)
|
|
}
|
|
|
|
/// Returns the reward destination of a staking ledger, stored in [`Payee`].
|
|
///
|
|
/// Note: if the stash is not bonded and/or does not have an entry in [`Payee`], it returns the
|
|
/// default reward destination.
|
|
pub(crate) fn reward_destination(
|
|
account: StakingAccount<T::AccountId>,
|
|
) -> Option<RewardDestination<T::AccountId>> {
|
|
let stash = match account {
|
|
StakingAccount::Stash(stash) => Some(stash),
|
|
StakingAccount::Controller(controller) =>
|
|
Self::paired_account(StakingAccount::Controller(controller)),
|
|
};
|
|
|
|
if let Some(stash) = stash {
|
|
<Payee<T>>::get(stash)
|
|
} else {
|
|
defensive!("fetched reward destination from unbonded stash {}", stash);
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Returns the controller account of a staking ledger.
|
|
///
|
|
/// Note: it will fallback into querying the [`Bonded`] storage with the ledger stash if the
|
|
/// controller is not set in `self`, which most likely means that self was fetched directly from
|
|
/// [`Ledger`] instead of through the methods exposed in [`StakingLedger`]. If the ledger does
|
|
/// not exist in storage, it returns `None`.
|
|
pub(crate) fn controller(&self) -> Option<T::AccountId> {
|
|
self.controller.clone().or_else(|| {
|
|
defensive!("fetched a controller on a ledger instance without it.");
|
|
Self::paired_account(StakingAccount::Stash(self.stash.clone()))
|
|
})
|
|
}
|
|
|
|
/// Inserts/updates a staking ledger account.
|
|
///
|
|
/// Bonds the ledger if it is not bonded yet, signalling that this is a new ledger. The staking
|
|
/// locks of the stash account are updated accordingly.
|
|
///
|
|
/// Note: To ensure lock consistency, all the [`Ledger`] storage updates should be made through
|
|
/// this helper function.
|
|
pub(crate) fn update(self) -> Result<(), Error<T>> {
|
|
if !<Bonded<T>>::contains_key(&self.stash) {
|
|
return Err(Error::<T>::NotStash)
|
|
}
|
|
|
|
T::Currency::set_lock(STAKING_ID, &self.stash, self.total, WithdrawReasons::all());
|
|
Ledger::<T>::insert(
|
|
&self.controller().ok_or_else(|| {
|
|
defensive!("update called on a ledger that is not bonded.");
|
|
Error::<T>::NotController
|
|
})?,
|
|
&self,
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Bonds a ledger.
|
|
///
|
|
/// It sets the reward preferences for the bonded stash.
|
|
pub(crate) fn bond(self, payee: RewardDestination<T::AccountId>) -> Result<(), Error<T>> {
|
|
if <Bonded<T>>::contains_key(&self.stash) {
|
|
Err(Error::<T>::AlreadyBonded)
|
|
} else {
|
|
<Payee<T>>::insert(&self.stash, payee);
|
|
<Bonded<T>>::insert(&self.stash, &self.stash);
|
|
self.update()
|
|
}
|
|
}
|
|
|
|
/// Sets the ledger Payee.
|
|
pub(crate) fn set_payee(self, payee: RewardDestination<T::AccountId>) -> Result<(), Error<T>> {
|
|
if !<Bonded<T>>::contains_key(&self.stash) {
|
|
Err(Error::<T>::NotStash)
|
|
} else {
|
|
<Payee<T>>::insert(&self.stash, payee);
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Sets the ledger controller to its stash.
|
|
pub(crate) fn set_controller_to_stash(self) -> Result<(), Error<T>> {
|
|
let controller = self.controller.as_ref()
|
|
.defensive_proof("Ledger's controller field didn't exist. The controller should have been fetched using StakingLedger.")
|
|
.ok_or(Error::<T>::NotController)?;
|
|
|
|
ensure!(self.stash != *controller, Error::<T>::AlreadyPaired);
|
|
|
|
// check if the ledger's stash is a controller of another ledger.
|
|
if let Some(bonded_ledger) = Ledger::<T>::get(&self.stash) {
|
|
// there is a ledger bonded by the stash. In this case, the stash of the bonded ledger
|
|
// should be the same as the ledger's stash. Otherwise fail to prevent data
|
|
// inconsistencies. See <https://github.com/paritytech/polkadot-sdk/pull/3639> for more
|
|
// details.
|
|
ensure!(bonded_ledger.stash == self.stash, Error::<T>::BadState);
|
|
}
|
|
|
|
<Ledger<T>>::remove(&controller);
|
|
<Ledger<T>>::insert(&self.stash, &self);
|
|
<Bonded<T>>::insert(&self.stash, &self.stash);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Clears all data related to a staking ledger and its bond in both [`Ledger`] and [`Bonded`]
|
|
/// storage items and updates the stash staking lock.
|
|
pub(crate) fn kill(stash: &T::AccountId) -> Result<(), Error<T>> {
|
|
let controller = <Bonded<T>>::get(stash).ok_or(Error::<T>::NotStash)?;
|
|
|
|
<Ledger<T>>::get(&controller).ok_or(Error::<T>::NotController).map(|ledger| {
|
|
T::Currency::remove_lock(STAKING_ID, &ledger.stash);
|
|
Ledger::<T>::remove(controller);
|
|
|
|
<Bonded<T>>::remove(&stash);
|
|
<Payee<T>>::remove(&stash);
|
|
|
|
Ok(())
|
|
})?
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
use {
|
|
crate::UnlockChunk,
|
|
codec::{Decode, Encode, MaxEncodedLen},
|
|
scale_info::TypeInfo,
|
|
};
|
|
|
|
// This structs makes it easy to write tests to compare staking ledgers fetched from storage. This
|
|
// is required because the controller field is not stored in storage and it is private.
|
|
#[cfg(test)]
|
|
#[derive(frame_support::DebugNoBound, Clone, Encode, Decode, TypeInfo, MaxEncodedLen)]
|
|
pub struct StakingLedgerInspect<T: Config> {
|
|
pub stash: T::AccountId,
|
|
#[codec(compact)]
|
|
pub total: BalanceOf<T>,
|
|
#[codec(compact)]
|
|
pub active: BalanceOf<T>,
|
|
pub unlocking: frame_support::BoundedVec<UnlockChunk<BalanceOf<T>>, T::MaxUnlockingChunks>,
|
|
pub legacy_claimed_rewards: frame_support::BoundedVec<sp_staking::EraIndex, T::HistoryDepth>,
|
|
}
|
|
|
|
#[cfg(test)]
|
|
impl<T: Config> PartialEq<StakingLedgerInspect<T>> for StakingLedger<T> {
|
|
fn eq(&self, other: &StakingLedgerInspect<T>) -> bool {
|
|
self.stash == other.stash &&
|
|
self.total == other.total &&
|
|
self.active == other.active &&
|
|
self.unlocking == other.unlocking &&
|
|
self.legacy_claimed_rewards == other.legacy_claimed_rewards
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
impl<T: Config> codec::EncodeLike<StakingLedger<T>> for StakingLedgerInspect<T> {}
|