mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 16:57:58 +00:00
8e393aa5a8
* Make `decl_error!` errors usable This pr implements support for returning errors of different pallets in a pallet. These errors need to be declared with `decl_error!`. The pr changes the following: - Each dispatchable function now returns a `DispatchResult` which is an alias for `Result<(), DispatchError>`. - `DispatchError` is an enum that has 4 variants: - `Other`: For storing string error messages - `CannotLookup`: Variant that is returned when something returns a `sp_runtime::LookupError` - `BadOrigin`: Variant that is returned for any kind of bad origin - `Module`: The error of a specific module. Contains the `index`, `error` and the `message`. The index is the index of the module in `construct_runtime!`. `error` is the index of the error in the error enum declared by `decl_error!`. `message` is the message to the error variant (this will not be encoded). - `construct_runtime!` now creates a new struct `ModuleToIndex`. This struct implements the trait `ModuleToIndex`. - `frame_system::Trait` has a new associated type: `ModuleToIndex` that expects the `ModuleToIndex` generated by `construct_runtime!`. - All error strings returned in any module are being converted now to `DispatchError`. - `BadOrigin` is the default error returned by any type that implements `EnsureOrigin`. * Fix frame system benchmarks
1021 lines
35 KiB
Rust
1021 lines
35 KiB
Rust
// Copyright 2017-2019 Parity Technologies (UK) Ltd.
|
|
// This file is part of Substrate.
|
|
|
|
// Substrate is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Substrate is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Substrate. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! # Session Module
|
|
//!
|
|
//! The Session module allows validators to manage their session keys, provides a function for changing
|
|
//! the session length, and handles session rotation.
|
|
//!
|
|
//! - [`session::Trait`](./trait.Trait.html)
|
|
//! - [`Call`](./enum.Call.html)
|
|
//! - [`Module`](./struct.Module.html)
|
|
//!
|
|
//! ## Overview
|
|
//!
|
|
//! ### Terminology
|
|
//! <!-- Original author of paragraph: @gavofyork -->
|
|
//!
|
|
//! - **Session:** A session is a period of time that has a constant set of validators. Validators can only join
|
|
//! or exit the validator set at a session change. It is measured in block numbers. The block where a session is
|
|
//! ended is determined by the `ShouldSessionEnd` trait. When the session is ending, a new validator set
|
|
//! can be chosen by `OnSessionEnding` implementations.
|
|
//! - **Session key:** A session key is actually several keys kept together that provide the various signing
|
|
//! functions required by network authorities/validators in pursuit of their duties.
|
|
//! - **Validator ID:** Every account has an associated validator ID. For some simple staking systems, this
|
|
//! may just be the same as the account ID. For staking systems using a stash/controller model,
|
|
//! the validator ID would be the stash account ID of the controller.
|
|
//! - **Session key configuration process:** A session key is set using `set_key` for use in the
|
|
//! next session. It is stored in `NextKeyFor`, a mapping between the caller's `ValidatorId` and the session
|
|
//! keys provided. `set_key` allows users to set their session key prior to being selected as validator.
|
|
//! It is a public call since it uses `ensure_signed`, which checks that the origin is a signed account.
|
|
//! As such, the account ID of the origin stored in in `NextKeyFor` may not necessarily be associated with
|
|
//! a block author or a validator. The session keys of accounts are removed once their account balance is zero.
|
|
//! - **Validator set session key configuration process:** Each session we iterate through the current
|
|
//! set of validator account IDs to check if a session key was created for it in the previous session
|
|
//! using `set_key`. If it was then we call `set_authority` from the [Consensus module](../frame_consensus/index.html)
|
|
//! and pass it a set of session keys (each associated with an account ID) as the session keys for the new
|
|
//! validator set. Lastly, if the session key of the current authority does not match any session keys stored under
|
|
//! its validator index in the `AuthorityStorageVec` mapping, then we update the mapping with its session
|
|
//! key and update the saved list of original authorities if necessary
|
|
//! (see https://github.com/paritytech/substrate/issues/1290). Note: Authorities are stored in the Consensus module.
|
|
//! They are represented by a validator account ID index from the Session module and allocated with a session
|
|
//! key for the length of the session.
|
|
//! - **Session length change process:** At the start of the next session we allocate a session index and record the
|
|
//! timestamp when the session started. If a `NextSessionLength` was recorded in the previous session, we record
|
|
//! it as the new session length. Additionally, if the new session length differs from the length of the
|
|
//! next session then we record a `LastLengthChange`.
|
|
//! - **Session rotation configuration:** Configure as either a 'normal' (rewardable session where rewards are
|
|
//! applied) or 'exceptional' (slashable) session rotation.
|
|
//! - **Session rotation process:** The session is changed at the end of the final block of the current session
|
|
//! using the `on_finalize` method. It may be called by either an origin or internally from another runtime
|
|
//! module at the end of each block.
|
|
//!
|
|
//! ### Goals
|
|
//!
|
|
//! The Session module in Substrate is designed to make the following possible:
|
|
//!
|
|
//! - Set session keys of the validator set for the next session.
|
|
//! - Set the length of a session.
|
|
//! - Configure and switch between either normal or exceptional session rotations.
|
|
//!
|
|
//! ## Interface
|
|
//!
|
|
//! ### Dispatchable Functions
|
|
//!
|
|
//! - `set_key` - Set a validator's session key for the next session.
|
|
//! - `set_length` - Set a new session length to be applied upon the next session change.
|
|
//! - `force_new_session` - Force a new session that should be considered either a normal (rewardable)
|
|
//! or exceptional rotation.
|
|
//! - `on_finalize` - Called when a block is finalized. Will rotate session if it is the last
|
|
//! block of the current session.
|
|
//!
|
|
//! ### Public Functions
|
|
//!
|
|
//! - `validator_count` - Get the current number of validators.
|
|
//! - `last_length_change` - Get the block number when the session length last changed.
|
|
//! - `apply_force_new_session` - Force a new session. Can be called by other runtime modules.
|
|
//! - `set_validators` - Set the current set of validators. Can only be called by the Staking module.
|
|
//! - `check_rotate_session` - Rotate the session and apply rewards if necessary. Called after the Staking
|
|
//! module updates the authorities to the new validator set.
|
|
//! - `rotate_session` - Change to the next session. Register the new authority set. Update session keys.
|
|
//! Enact session length change if applicable.
|
|
//! - `ideal_session_duration` - Get the time of an ideal session.
|
|
//! - `blocks_remaining` - Get the number of blocks remaining in the current session,
|
|
//! excluding the current block.
|
|
//!
|
|
//! ## Usage
|
|
//!
|
|
//! ### Example from the SRML
|
|
//!
|
|
//! The [Staking module](../pallet_staking/index.html) uses the Session module to get the validator set.
|
|
//!
|
|
//! ```
|
|
//! use pallet_session as session;
|
|
//! # fn not_executed<T: session::Trait>() {
|
|
//!
|
|
//! let validators = <session::Module<T>>::validators();
|
|
//! # }
|
|
//! # fn main(){}
|
|
//! ```
|
|
//!
|
|
//! ## Related Modules
|
|
//!
|
|
//! - [Consensus](../frame_consensus/index.html)
|
|
//! - [Staking](../pallet_staking/index.html)
|
|
//! - [Timestamp](../pallet_timestamp/index.html)
|
|
|
|
#![cfg_attr(not(feature = "std"), no_std)]
|
|
|
|
use sp_std::{prelude::*, marker::PhantomData, ops::{Sub, Rem}};
|
|
use codec::Decode;
|
|
use sp_runtime::{KeyTypeId, Perbill, RuntimeAppPublic, BoundToRuntimeAppPublic};
|
|
use frame_support::weights::SimpleDispatchInfo;
|
|
use sp_runtime::traits::{Convert, Zero, Member, OpaqueKeys};
|
|
use sp_staking::SessionIndex;
|
|
use frame_support::{dispatch, ConsensusEngineId, decl_module, decl_event, decl_storage};
|
|
use frame_support::{ensure, traits::{OnFreeBalanceZero, Get, FindAuthor, ValidatorRegistration}, Parameter};
|
|
use frame_system::{self as system, ensure_signed};
|
|
|
|
#[cfg(test)]
|
|
mod mock;
|
|
|
|
#[cfg(feature = "historical")]
|
|
pub mod historical;
|
|
|
|
/// Decides whether the session should be ended.
|
|
pub trait ShouldEndSession<BlockNumber> {
|
|
/// Return `true` if the session should be ended.
|
|
fn should_end_session(now: BlockNumber) -> bool;
|
|
}
|
|
|
|
/// Ends the session after a fixed period of blocks.
|
|
///
|
|
/// The first session will have length of `Offset`, and
|
|
/// the following sessions will have length of `Period`.
|
|
/// This may prove nonsensical if `Offset` >= `Period`.
|
|
pub struct PeriodicSessions<
|
|
Period,
|
|
Offset,
|
|
>(PhantomData<(Period, Offset)>);
|
|
|
|
impl<
|
|
BlockNumber: Rem<Output=BlockNumber> + Sub<Output=BlockNumber> + Zero + PartialOrd,
|
|
Period: Get<BlockNumber>,
|
|
Offset: Get<BlockNumber>,
|
|
> ShouldEndSession<BlockNumber> for PeriodicSessions<Period, Offset> {
|
|
fn should_end_session(now: BlockNumber) -> bool {
|
|
let offset = Offset::get();
|
|
now >= offset && ((now - offset) % Period::get()).is_zero()
|
|
}
|
|
}
|
|
|
|
/// An event handler for when the session is ending.
|
|
/// TODO [slashing] consider renaming to OnSessionStarting
|
|
pub trait OnSessionEnding<ValidatorId> {
|
|
/// Handle the fact that the session is ending, and optionally provide the new validator set.
|
|
///
|
|
/// Even if the validator-set is the same as before, if any underlying economic
|
|
/// conditions have changed (i.e. stake-weights), the new validator set must be returned.
|
|
/// This is necessary for consensus engines making use of the session module to
|
|
/// issue a validator-set change so misbehavior can be provably associated with the new
|
|
/// economic conditions as opposed to the old.
|
|
///
|
|
/// `ending_index` is the index of the currently ending session.
|
|
/// The returned validator set, if any, will not be applied until `will_apply_at`.
|
|
/// `will_apply_at` is guaranteed to be at least `ending_index + 1`, since session indices don't
|
|
/// repeat, but it could be some time after in case we are staging authority set changes.
|
|
fn on_session_ending(
|
|
ending_index: SessionIndex,
|
|
will_apply_at: SessionIndex
|
|
) -> Option<Vec<ValidatorId>>;
|
|
}
|
|
|
|
impl<A> OnSessionEnding<A> for () {
|
|
fn on_session_ending(_: SessionIndex, _: SessionIndex) -> Option<Vec<A>> { None }
|
|
}
|
|
|
|
/// Handler for session lifecycle events.
|
|
pub trait SessionHandler<ValidatorId> {
|
|
/// All the key type ids this session handler can process.
|
|
///
|
|
/// The order must be the same as it expects them in
|
|
/// [`on_new_session`](Self::on_new_session) and [`on_genesis_session`](Self::on_genesis_session).
|
|
const KEY_TYPE_IDS: &'static [KeyTypeId];
|
|
|
|
/// The given validator set will be used for the genesis session.
|
|
/// It is guaranteed that the given validator set will also be used
|
|
/// for the second session, therefore the first call to `on_new_session`
|
|
/// should provide the same validator set.
|
|
fn on_genesis_session<Ks: OpaqueKeys>(validators: &[(ValidatorId, Ks)]);
|
|
|
|
/// Session set has changed; act appropriately. Note that this can be called
|
|
/// before initialization of your module.
|
|
///
|
|
/// `changed` is true whenever any of the session keys or underlying economic
|
|
/// identities or weightings behind those keys has changed.
|
|
fn on_new_session<Ks: OpaqueKeys>(
|
|
changed: bool,
|
|
validators: &[(ValidatorId, Ks)],
|
|
queued_validators: &[(ValidatorId, Ks)],
|
|
);
|
|
|
|
/// A notification for end of the session.
|
|
///
|
|
/// Note it is triggered before any `OnSessionEnding` handlers,
|
|
/// so we can still affect the validator set.
|
|
fn on_before_session_ending() {}
|
|
|
|
/// A validator got disabled. Act accordingly until a new session begins.
|
|
fn on_disabled(validator_index: usize);
|
|
}
|
|
|
|
/// A session handler for specific key type.
|
|
pub trait OneSessionHandler<ValidatorId>: BoundToRuntimeAppPublic {
|
|
/// The key type expected.
|
|
type Key: Decode + Default + RuntimeAppPublic;
|
|
|
|
fn on_genesis_session<'a, I: 'a>(validators: I)
|
|
where I: Iterator<Item=(&'a ValidatorId, Self::Key)>, ValidatorId: 'a;
|
|
|
|
/// Session set has changed; act appropriately. Note that this can be called
|
|
/// before initialization of your module.
|
|
///
|
|
/// `changed` is true when at least one of the session keys
|
|
/// or the underlying economic identities/distribution behind one the
|
|
/// session keys has changed, false otherwise.
|
|
///
|
|
/// The `validators` are the validators of the incoming session, and `queued_validators`
|
|
/// will follow.
|
|
fn on_new_session<'a, I: 'a>(
|
|
changed: bool,
|
|
validators: I,
|
|
queued_validators: I,
|
|
) where I: Iterator<Item=(&'a ValidatorId, Self::Key)>, ValidatorId: 'a;
|
|
|
|
|
|
/// A notification for end of the session.
|
|
///
|
|
/// Note it is triggered before any `OnSessionEnding` handlers,
|
|
/// so we can still affect the validator set.
|
|
fn on_before_session_ending() {}
|
|
|
|
/// A validator got disabled. Act accordingly until a new session begins.
|
|
fn on_disabled(_validator_index: usize);
|
|
}
|
|
|
|
#[impl_trait_for_tuples::impl_for_tuples(1, 30)]
|
|
#[tuple_types_no_default_trait_bound]
|
|
impl<AId> SessionHandler<AId> for Tuple {
|
|
for_tuples!( where #( Tuple: OneSessionHandler<AId> )* );
|
|
|
|
for_tuples!(
|
|
const KEY_TYPE_IDS: &'static [KeyTypeId] = &[ #( <Tuple::Key as RuntimeAppPublic>::ID ),* ];
|
|
);
|
|
|
|
fn on_genesis_session<Ks: OpaqueKeys>(validators: &[(AId, Ks)]) {
|
|
for_tuples!(
|
|
#(
|
|
let our_keys: Box<dyn Iterator<Item=_>> = Box::new(validators.iter()
|
|
.map(|k| (&k.0, k.1.get::<Tuple::Key>(<Tuple::Key as RuntimeAppPublic>::ID)
|
|
.unwrap_or_default())));
|
|
|
|
Tuple::on_genesis_session(our_keys);
|
|
)*
|
|
)
|
|
}
|
|
|
|
fn on_new_session<Ks: OpaqueKeys>(
|
|
changed: bool,
|
|
validators: &[(AId, Ks)],
|
|
queued_validators: &[(AId, Ks)],
|
|
) {
|
|
for_tuples!(
|
|
#(
|
|
let our_keys: Box<dyn Iterator<Item=_>> = Box::new(validators.iter()
|
|
.map(|k| (&k.0, k.1.get::<Tuple::Key>(<Tuple::Key as RuntimeAppPublic>::ID)
|
|
.unwrap_or_default())));
|
|
let queued_keys: Box<dyn Iterator<Item=_>> = Box::new(queued_validators.iter()
|
|
.map(|k| (&k.0, k.1.get::<Tuple::Key>(<Tuple::Key as RuntimeAppPublic>::ID)
|
|
.unwrap_or_default())));
|
|
Tuple::on_new_session(changed, our_keys, queued_keys);
|
|
)*
|
|
)
|
|
}
|
|
|
|
fn on_before_session_ending() {
|
|
for_tuples!( #( Tuple::on_before_session_ending(); )* )
|
|
}
|
|
|
|
fn on_disabled(i: usize) {
|
|
for_tuples!( #( Tuple::on_disabled(i); )* )
|
|
}
|
|
}
|
|
|
|
/// `SessionHandler` for tests that use `UintAuthorityId` as `Keys`.
|
|
pub struct TestSessionHandler;
|
|
impl<AId> SessionHandler<AId> for TestSessionHandler {
|
|
const KEY_TYPE_IDS: &'static [KeyTypeId] = &[sp_runtime::key_types::DUMMY];
|
|
|
|
fn on_genesis_session<Ks: OpaqueKeys>(_: &[(AId, Ks)]) {}
|
|
|
|
fn on_new_session<Ks: OpaqueKeys>(_: bool, _: &[(AId, Ks)], _: &[(AId, Ks)]) {}
|
|
|
|
fn on_before_session_ending() {}
|
|
|
|
fn on_disabled(_: usize) {}
|
|
}
|
|
|
|
/// Handler for selecting the genesis validator set.
|
|
pub trait SelectInitialValidators<ValidatorId> {
|
|
/// Returns the initial validator set. If `None` is returned
|
|
/// all accounts that have session keys set in the genesis block
|
|
/// will be validators.
|
|
fn select_initial_validators() -> Option<Vec<ValidatorId>>;
|
|
}
|
|
|
|
/// Implementation of `SelectInitialValidators` that does nothing.
|
|
impl<V> SelectInitialValidators<V> for () {
|
|
fn select_initial_validators() -> Option<Vec<V>> {
|
|
None
|
|
}
|
|
}
|
|
|
|
impl<T: Trait> ValidatorRegistration<T::ValidatorId> for Module<T> {
|
|
fn is_registered(id: &T::ValidatorId) -> bool {
|
|
Self::load_keys(id).is_some()
|
|
}
|
|
}
|
|
|
|
pub trait Trait: frame_system::Trait {
|
|
/// The overarching event type.
|
|
type Event: From<Event> + Into<<Self as frame_system::Trait>::Event>;
|
|
|
|
/// A stable ID for a validator.
|
|
type ValidatorId: Member + Parameter;
|
|
|
|
/// A conversion from account ID to validator ID.
|
|
type ValidatorIdOf: Convert<Self::AccountId, Option<Self::ValidatorId>>;
|
|
|
|
/// Indicator for when to end the session.
|
|
type ShouldEndSession: ShouldEndSession<Self::BlockNumber>;
|
|
|
|
/// Handler for when a session is about to end.
|
|
type OnSessionEnding: OnSessionEnding<Self::ValidatorId>;
|
|
|
|
/// Handler when a session has changed.
|
|
type SessionHandler: SessionHandler<Self::ValidatorId>;
|
|
|
|
/// The keys.
|
|
type Keys: OpaqueKeys + Member + Parameter + Default;
|
|
|
|
/// The fraction of validators set that is safe to be disabled.
|
|
///
|
|
/// After the threshold is reached `disabled` method starts to return true,
|
|
/// which in combination with `pallet_staking` forces a new era.
|
|
type DisabledValidatorsThreshold: Get<Perbill>;
|
|
|
|
/// Select initial validators.
|
|
type SelectInitialValidators: SelectInitialValidators<Self::ValidatorId>;
|
|
}
|
|
|
|
const DEDUP_KEY_PREFIX: &[u8] = b":session:keys";
|
|
|
|
decl_storage! {
|
|
trait Store for Module<T: Trait> as Session {
|
|
/// The current set of validators.
|
|
Validators get(fn validators): Vec<T::ValidatorId>;
|
|
|
|
/// Current index of the session.
|
|
CurrentIndex get(fn current_index): SessionIndex;
|
|
|
|
/// True if the underlying economic identities or weighting behind the validators
|
|
/// has changed in the queued validator set.
|
|
QueuedChanged: bool;
|
|
|
|
/// The queued keys for the next session. When the next session begins, these keys
|
|
/// will be used to determine the validator's session keys.
|
|
QueuedKeys get(fn queued_keys): Vec<(T::ValidatorId, T::Keys)>;
|
|
|
|
/// Indices of disabled validators.
|
|
///
|
|
/// The set is cleared when `on_session_ending` returns a new set of identities.
|
|
DisabledValidators get(fn disabled_validators): Vec<u32>;
|
|
|
|
/// The next session keys for a validator.
|
|
///
|
|
/// The first key is always `DEDUP_KEY_PREFIX` to have all the data in the same branch of
|
|
/// the trie. Having all data in the same branch should prevent slowing down other queries.
|
|
NextKeys: double_map hasher(twox_64_concat) Vec<u8>, blake2_256(T::ValidatorId) => Option<T::Keys>;
|
|
|
|
/// The owner of a key. The second key is the `KeyTypeId` + the encoded key.
|
|
///
|
|
/// The first key is always `DEDUP_KEY_PREFIX` to have all the data in the same branch of
|
|
/// the trie. Having all data in the same branch should prevent slowing down other queries.
|
|
KeyOwner: double_map hasher(twox_64_concat) Vec<u8>, blake2_256((KeyTypeId, Vec<u8>)) => Option<T::ValidatorId>;
|
|
}
|
|
add_extra_genesis {
|
|
config(keys): Vec<(T::ValidatorId, T::Keys)>;
|
|
build(|config: &GenesisConfig<T>| {
|
|
if T::SessionHandler::KEY_TYPE_IDS.len() != T::Keys::key_ids().len() {
|
|
panic!("Number of keys in session handler and session keys does not match");
|
|
}
|
|
|
|
T::SessionHandler::KEY_TYPE_IDS.iter().zip(T::Keys::key_ids()).enumerate()
|
|
.for_each(|(i, (sk, kk))| {
|
|
if sk != kk {
|
|
panic!(
|
|
"Session handler and session key expect different key type at index: {}",
|
|
i,
|
|
);
|
|
}
|
|
});
|
|
|
|
for (who, keys) in config.keys.iter().cloned() {
|
|
assert!(
|
|
<Module<T>>::load_keys(&who).is_none(),
|
|
"genesis config contained duplicate validator {:?}", who,
|
|
);
|
|
|
|
<Module<T>>::do_set_keys(&who, keys)
|
|
.expect("genesis config must not contain duplicates; qed");
|
|
}
|
|
|
|
let initial_validators = T::SelectInitialValidators::select_initial_validators()
|
|
.unwrap_or_else(|| config.keys.iter().map(|(ref v, _)| v.clone()).collect());
|
|
|
|
assert!(!initial_validators.is_empty(), "Empty validator set in genesis block!");
|
|
|
|
let queued_keys: Vec<_> = initial_validators
|
|
.iter()
|
|
.cloned()
|
|
.map(|v| (
|
|
v.clone(),
|
|
<Module<T>>::load_keys(&v).unwrap_or_default(),
|
|
))
|
|
.collect();
|
|
|
|
// Tell everyone about the genesis session keys
|
|
T::SessionHandler::on_genesis_session::<T::Keys>(&queued_keys);
|
|
|
|
<Validators<T>>::put(initial_validators);
|
|
<QueuedKeys<T>>::put(queued_keys);
|
|
});
|
|
}
|
|
}
|
|
|
|
decl_event!(
|
|
pub enum Event {
|
|
/// New session has happened. Note that the argument is the session index, not the block
|
|
/// number as the type might suggest.
|
|
NewSession(SessionIndex),
|
|
}
|
|
);
|
|
|
|
decl_module! {
|
|
pub struct Module<T: Trait> for enum Call where origin: T::Origin {
|
|
/// Used as first key for `NextKeys` and `KeyOwner` to put all the data into the same branch
|
|
/// of the trie.
|
|
const DEDUP_KEY_PREFIX: &[u8] = DEDUP_KEY_PREFIX;
|
|
|
|
fn deposit_event() = default;
|
|
|
|
/// Sets the session key(s) of the function caller to `key`.
|
|
/// Allows an account to set its session key prior to becoming a validator.
|
|
/// This doesn't take effect until the next session.
|
|
///
|
|
/// The dispatch origin of this function must be signed.
|
|
///
|
|
/// # <weight>
|
|
/// - O(log n) in number of accounts.
|
|
/// - One extra DB entry.
|
|
/// # </weight>
|
|
#[weight = SimpleDispatchInfo::FixedNormal(150_000)]
|
|
fn set_keys(origin, keys: T::Keys, proof: Vec<u8>) -> dispatch::DispatchResult {
|
|
let who = ensure_signed(origin)?;
|
|
|
|
ensure!(keys.ownership_proof_is_valid(&proof), "invalid ownership proof");
|
|
|
|
let who = match T::ValidatorIdOf::convert(who) {
|
|
Some(val_id) => val_id,
|
|
None => Err("no associated validator ID for account.")?,
|
|
};
|
|
|
|
Self::do_set_keys(&who, keys)?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Called when a block is initialized. Will rotate session if it is the last
|
|
/// block of the current session.
|
|
fn on_initialize(n: T::BlockNumber) {
|
|
if T::ShouldEndSession::should_end_session(n) {
|
|
Self::rotate_session();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: Trait> Module<T> {
|
|
/// Move on to next session. Register new validator set and session keys. Changes
|
|
/// to the validator set have a session of delay to take effect. This allows for
|
|
/// equivocation punishment after a fork.
|
|
pub fn rotate_session() {
|
|
let session_index = CurrentIndex::get();
|
|
|
|
let changed = QueuedChanged::get();
|
|
|
|
// Inform the session handlers that a session is going to end.
|
|
T::SessionHandler::on_before_session_ending();
|
|
|
|
// Get queued session keys and validators.
|
|
let session_keys = <QueuedKeys<T>>::get();
|
|
let validators = session_keys.iter()
|
|
.map(|(validator, _)| validator.clone())
|
|
.collect::<Vec<_>>();
|
|
<Validators<T>>::put(&validators);
|
|
|
|
if changed {
|
|
// reset disabled validators
|
|
DisabledValidators::take();
|
|
}
|
|
|
|
let applied_at = session_index + 2;
|
|
|
|
// Get next validator set.
|
|
let maybe_next_validators = T::OnSessionEnding::on_session_ending(session_index, applied_at);
|
|
let (next_validators, next_identities_changed)
|
|
= if let Some(validators) = maybe_next_validators
|
|
{
|
|
// NOTE: as per the documentation on `OnSessionEnding`, we consider
|
|
// the validator set as having changed even if the validators are the
|
|
// same as before, as underlying economic conditions may have changed.
|
|
(validators, true)
|
|
} else {
|
|
(<Validators<T>>::get(), false)
|
|
};
|
|
|
|
// Increment session index.
|
|
let session_index = session_index + 1;
|
|
CurrentIndex::put(session_index);
|
|
|
|
// Queue next session keys.
|
|
let (queued_amalgamated, next_changed) = {
|
|
// until we are certain there has been a change, iterate the prior
|
|
// validators along with the current and check for changes
|
|
let mut changed = next_identities_changed;
|
|
|
|
let mut now_session_keys = session_keys.iter();
|
|
let mut check_next_changed = |keys: &T::Keys| {
|
|
if changed { return }
|
|
// since a new validator set always leads to `changed` starting
|
|
// as true, we can ensure that `now_session_keys` and `next_validators`
|
|
// have the same length. this function is called once per iteration.
|
|
if let Some(&(_, ref old_keys)) = now_session_keys.next() {
|
|
if old_keys != keys {
|
|
changed = true;
|
|
return
|
|
}
|
|
}
|
|
};
|
|
let queued_amalgamated = next_validators.into_iter()
|
|
.map(|a| {
|
|
let k = Self::load_keys(&a).unwrap_or_default();
|
|
check_next_changed(&k);
|
|
(a, k)
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
(queued_amalgamated, changed)
|
|
};
|
|
|
|
<QueuedKeys<T>>::put(queued_amalgamated.clone());
|
|
QueuedChanged::put(next_changed);
|
|
|
|
// Record that this happened.
|
|
Self::deposit_event(Event::NewSession(session_index));
|
|
|
|
// Tell everyone about the new session keys.
|
|
T::SessionHandler::on_new_session::<T::Keys>(
|
|
changed,
|
|
&session_keys,
|
|
&queued_amalgamated,
|
|
);
|
|
}
|
|
|
|
/// Disable the validator of index `i`.
|
|
///
|
|
/// Returns `true` if this causes a `DisabledValidatorsThreshold` of validators
|
|
/// to be already disabled.
|
|
pub fn disable_index(i: usize) -> bool {
|
|
let (fire_event, threshold_reached) = DisabledValidators::mutate(|disabled| {
|
|
let i = i as u32;
|
|
if let Err(index) = disabled.binary_search(&i) {
|
|
let count = <Validators<T>>::decode_len().unwrap_or(0) as u32;
|
|
let threshold = T::DisabledValidatorsThreshold::get() * count;
|
|
disabled.insert(index, i);
|
|
(true, disabled.len() as u32 > threshold)
|
|
} else {
|
|
(false, false)
|
|
}
|
|
});
|
|
|
|
if fire_event {
|
|
T::SessionHandler::on_disabled(i);
|
|
}
|
|
|
|
threshold_reached
|
|
}
|
|
|
|
/// Disable the validator identified by `c`. (If using with the staking module,
|
|
/// this would be their *stash* account.)
|
|
///
|
|
/// Returns `Ok(true)` if more than `DisabledValidatorsThreshold` validators in current
|
|
/// session is already disabled.
|
|
/// If used with the staking module it allows to force a new era in such case.
|
|
pub fn disable(c: &T::ValidatorId) -> sp_std::result::Result<bool, ()> {
|
|
Self::validators().iter().position(|i| i == c).map(Self::disable_index).ok_or(())
|
|
}
|
|
|
|
// perform the set_key operation, checking for duplicates.
|
|
// does not set `Changed`.
|
|
fn do_set_keys(who: &T::ValidatorId, keys: T::Keys) -> dispatch::DispatchResult {
|
|
let old_keys = Self::load_keys(&who);
|
|
|
|
for id in T::Keys::key_ids() {
|
|
let key = keys.get_raw(*id);
|
|
|
|
// ensure keys are without duplication.
|
|
ensure!(
|
|
Self::key_owner(*id, key).map_or(true, |owner| &owner == who),
|
|
"registered duplicate key"
|
|
);
|
|
|
|
if let Some(old) = old_keys.as_ref().map(|k| k.get_raw(*id)) {
|
|
if key == old {
|
|
continue;
|
|
}
|
|
|
|
Self::clear_key_owner(*id, old);
|
|
}
|
|
|
|
Self::put_key_owner(*id, key, &who);
|
|
}
|
|
|
|
Self::put_keys(&who, &keys);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn prune_dead_keys(who: &T::ValidatorId) {
|
|
if let Some(old_keys) = Self::take_keys(who) {
|
|
for id in T::Keys::key_ids() {
|
|
let key_data = old_keys.get_raw(*id);
|
|
Self::clear_key_owner(*id, key_data);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn load_keys(v: &T::ValidatorId) -> Option<T::Keys> {
|
|
<NextKeys<T>>::get(DEDUP_KEY_PREFIX, v)
|
|
}
|
|
|
|
fn take_keys(v: &T::ValidatorId) -> Option<T::Keys> {
|
|
<NextKeys<T>>::take(DEDUP_KEY_PREFIX, v)
|
|
}
|
|
|
|
fn put_keys(v: &T::ValidatorId, keys: &T::Keys) {
|
|
<NextKeys<T>>::insert(DEDUP_KEY_PREFIX, v, keys);
|
|
}
|
|
|
|
fn key_owner(id: KeyTypeId, key_data: &[u8]) -> Option<T::ValidatorId> {
|
|
<KeyOwner<T>>::get(DEDUP_KEY_PREFIX, (id, key_data))
|
|
}
|
|
|
|
fn put_key_owner(id: KeyTypeId, key_data: &[u8], v: &T::ValidatorId) {
|
|
<KeyOwner<T>>::insert(DEDUP_KEY_PREFIX, (id, key_data), v)
|
|
}
|
|
|
|
fn clear_key_owner(id: KeyTypeId, key_data: &[u8]) {
|
|
<KeyOwner<T>>::remove(DEDUP_KEY_PREFIX, (id, key_data));
|
|
}
|
|
}
|
|
|
|
impl<T: Trait> OnFreeBalanceZero<T::ValidatorId> for Module<T> {
|
|
fn on_free_balance_zero(who: &T::ValidatorId) {
|
|
Self::prune_dead_keys(who);
|
|
}
|
|
}
|
|
|
|
/// Wraps the author-scraping logic for consensus engines that can recover
|
|
/// the canonical index of an author. This then transforms it into the
|
|
/// registering account-ID of that session key index.
|
|
pub struct FindAccountFromAuthorIndex<T, Inner>(sp_std::marker::PhantomData<(T, Inner)>);
|
|
|
|
impl<T: Trait, Inner: FindAuthor<u32>> FindAuthor<T::ValidatorId>
|
|
for FindAccountFromAuthorIndex<T, Inner>
|
|
{
|
|
fn find_author<'a, I>(digests: I) -> Option<T::ValidatorId>
|
|
where I: 'a + IntoIterator<Item=(ConsensusEngineId, &'a [u8])>
|
|
{
|
|
let i = Inner::find_author(digests)?;
|
|
|
|
let validators = <Module<T>>::validators();
|
|
validators.get(i as usize).map(|k| k.clone())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use frame_support::assert_ok;
|
|
use sp_core::crypto::key_types::DUMMY;
|
|
use sp_runtime::{traits::OnInitialize, testing::UintAuthorityId};
|
|
use mock::{
|
|
NEXT_VALIDATORS, SESSION_CHANGED, TEST_SESSION_CHANGED, authorities, force_new_session,
|
|
set_next_validators, set_session_length, session_changed, Test, Origin, System, Session,
|
|
reset_before_session_end_called, before_session_end_called,
|
|
};
|
|
|
|
fn new_test_ext() -> sp_io::TestExternalities {
|
|
let mut t = frame_system::GenesisConfig::default().build_storage::<Test>().unwrap();
|
|
GenesisConfig::<Test> {
|
|
keys: NEXT_VALIDATORS.with(|l|
|
|
l.borrow().iter().cloned().map(|i| (i, UintAuthorityId(i).into())).collect()
|
|
),
|
|
}.assimilate_storage(&mut t).unwrap();
|
|
sp_io::TestExternalities::new(t)
|
|
}
|
|
|
|
fn initialize_block(block: u64) {
|
|
SESSION_CHANGED.with(|l| *l.borrow_mut() = false);
|
|
System::set_block_number(block);
|
|
Session::on_initialize(block);
|
|
}
|
|
|
|
#[test]
|
|
fn simple_setup_should_work() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
assert_eq!(Session::validators(), vec![1, 2, 3]);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn put_get_keys() {
|
|
new_test_ext().execute_with(|| {
|
|
Session::put_keys(&10, &UintAuthorityId(10).into());
|
|
assert_eq!(Session::load_keys(&10), Some(UintAuthorityId(10).into()));
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn keys_cleared_on_kill() {
|
|
let mut ext = new_test_ext();
|
|
ext.execute_with(|| {
|
|
assert_eq!(Session::validators(), vec![1, 2, 3]);
|
|
assert_eq!(Session::load_keys(&1), Some(UintAuthorityId(1).into()));
|
|
|
|
let id = DUMMY;
|
|
assert_eq!(Session::key_owner(id, UintAuthorityId(1).get_raw(id)), Some(1));
|
|
|
|
Session::on_free_balance_zero(&1);
|
|
assert_eq!(Session::load_keys(&1), None);
|
|
assert_eq!(Session::key_owner(id, UintAuthorityId(1).get_raw(id)), None);
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn authorities_should_track_validators() {
|
|
reset_before_session_end_called();
|
|
|
|
new_test_ext().execute_with(|| {
|
|
set_next_validators(vec![1, 2]);
|
|
force_new_session();
|
|
initialize_block(1);
|
|
assert_eq!(Session::queued_keys(), vec![
|
|
(1, UintAuthorityId(1).into()),
|
|
(2, UintAuthorityId(2).into()),
|
|
]);
|
|
assert_eq!(Session::validators(), vec![1, 2, 3]);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
force_new_session();
|
|
initialize_block(2);
|
|
assert_eq!(Session::queued_keys(), vec![
|
|
(1, UintAuthorityId(1).into()),
|
|
(2, UintAuthorityId(2).into()),
|
|
]);
|
|
assert_eq!(Session::validators(), vec![1, 2]);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2)]);
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
set_next_validators(vec![1, 2, 4]);
|
|
assert_ok!(Session::set_keys(Origin::signed(4), UintAuthorityId(4).into(), vec![]));
|
|
force_new_session();
|
|
initialize_block(3);
|
|
assert_eq!(Session::queued_keys(), vec![
|
|
(1, UintAuthorityId(1).into()),
|
|
(2, UintAuthorityId(2).into()),
|
|
(4, UintAuthorityId(4).into()),
|
|
]);
|
|
assert_eq!(Session::validators(), vec![1, 2]);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2)]);
|
|
assert!(before_session_end_called());
|
|
|
|
force_new_session();
|
|
initialize_block(4);
|
|
assert_eq!(Session::queued_keys(), vec![
|
|
(1, UintAuthorityId(1).into()),
|
|
(2, UintAuthorityId(2).into()),
|
|
(4, UintAuthorityId(4).into()),
|
|
]);
|
|
assert_eq!(Session::validators(), vec![1, 2, 4]);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(4)]);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn should_work_with_early_exit() {
|
|
new_test_ext().execute_with(|| {
|
|
set_session_length(10);
|
|
|
|
initialize_block(1);
|
|
assert_eq!(Session::current_index(), 0);
|
|
|
|
initialize_block(2);
|
|
assert_eq!(Session::current_index(), 0);
|
|
|
|
force_new_session();
|
|
initialize_block(3);
|
|
assert_eq!(Session::current_index(), 1);
|
|
|
|
initialize_block(9);
|
|
assert_eq!(Session::current_index(), 1);
|
|
|
|
initialize_block(10);
|
|
assert_eq!(Session::current_index(), 2);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn session_change_should_work() {
|
|
new_test_ext().execute_with(|| {
|
|
// Block 1: No change
|
|
initialize_block(1);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
|
|
// Block 2: Session rollover, but no change.
|
|
initialize_block(2);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
|
|
// Block 3: Set new key for validator 2; no visible change.
|
|
initialize_block(3);
|
|
assert_ok!(Session::set_keys(Origin::signed(2), UintAuthorityId(5).into(), vec![]));
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
|
|
// Block 4: Session rollover; no visible change.
|
|
initialize_block(4);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
|
|
// Block 5: No change.
|
|
initialize_block(5);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]);
|
|
|
|
// Block 6: Session rollover; authority 2 changes.
|
|
initialize_block(6);
|
|
assert_eq!(authorities(), vec![UintAuthorityId(1), UintAuthorityId(5), UintAuthorityId(3)]);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn duplicates_are_not_allowed() {
|
|
new_test_ext().execute_with(|| {
|
|
System::set_block_number(1);
|
|
Session::on_initialize(1);
|
|
assert!(Session::set_keys(Origin::signed(4), UintAuthorityId(1).into(), vec![]).is_err());
|
|
assert!(Session::set_keys(Origin::signed(1), UintAuthorityId(10).into(), vec![]).is_ok());
|
|
|
|
// is fine now that 1 has migrated off.
|
|
assert!(Session::set_keys(Origin::signed(4), UintAuthorityId(1).into(), vec![]).is_ok());
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn session_changed_flag_works() {
|
|
reset_before_session_end_called();
|
|
|
|
new_test_ext().execute_with(|| {
|
|
TEST_SESSION_CHANGED.with(|l| *l.borrow_mut() = true);
|
|
|
|
force_new_session();
|
|
initialize_block(1);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
force_new_session();
|
|
initialize_block(2);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
Session::disable_index(0);
|
|
force_new_session();
|
|
initialize_block(3);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
force_new_session();
|
|
initialize_block(4);
|
|
assert!(session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
force_new_session();
|
|
initialize_block(5);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
assert_ok!(Session::set_keys(Origin::signed(2), UintAuthorityId(5).into(), vec![]));
|
|
force_new_session();
|
|
initialize_block(6);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
// changing the keys of a validator leads to change.
|
|
assert_ok!(Session::set_keys(Origin::signed(69), UintAuthorityId(69).into(), vec![]));
|
|
force_new_session();
|
|
initialize_block(7);
|
|
assert!(session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
|
|
// while changing the keys of a non-validator does not.
|
|
force_new_session();
|
|
initialize_block(7);
|
|
assert!(!session_changed());
|
|
assert!(before_session_end_called());
|
|
reset_before_session_end_called();
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn periodic_session_works() {
|
|
struct Period;
|
|
struct Offset;
|
|
|
|
impl Get<u64> for Period {
|
|
fn get() -> u64 { 10 }
|
|
}
|
|
|
|
impl Get<u64> for Offset {
|
|
fn get() -> u64 { 3 }
|
|
}
|
|
|
|
|
|
type P = PeriodicSessions<Period, Offset>;
|
|
|
|
for i in 0..3 {
|
|
assert!(!P::should_end_session(i));
|
|
}
|
|
|
|
assert!(P::should_end_session(3));
|
|
|
|
for i in (1..10).map(|i| 3 + i) {
|
|
assert!(!P::should_end_session(i));
|
|
}
|
|
|
|
assert!(P::should_end_session(13));
|
|
}
|
|
|
|
#[test]
|
|
fn session_keys_generate_output_works_as_set_keys_input() {
|
|
new_test_ext().execute_with(|| {
|
|
let new_keys = mock::MockSessionKeys::generate(None);
|
|
assert_ok!(
|
|
Session::set_keys(
|
|
Origin::signed(2),
|
|
<mock::Test as Trait>::Keys::decode(&mut &new_keys[..]).expect("Decode keys"),
|
|
vec![],
|
|
)
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn return_true_if_more_than_third_is_disabled() {
|
|
new_test_ext().execute_with(|| {
|
|
set_next_validators(vec![1, 2, 3, 4, 5, 6, 7]);
|
|
force_new_session();
|
|
initialize_block(1);
|
|
// apply the new validator set
|
|
force_new_session();
|
|
initialize_block(2);
|
|
|
|
assert_eq!(Session::disable_index(0), false);
|
|
assert_eq!(Session::disable_index(1), false);
|
|
assert_eq!(Session::disable_index(2), true);
|
|
assert_eq!(Session::disable_index(3), true);
|
|
});
|
|
}
|
|
}
|