// 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. //! Mock helpers for Session. use super::*; use crate as pallet_session; #[cfg(feature = "historical")] use crate::historical as pallet_session_historical; use std::collections::BTreeMap; use sp_core::{crypto::key_types::DUMMY, H256}; use sp_runtime::{ impl_opaque_keys, testing::{Header, UintAuthorityId}, traits::{BlakeTwo256, IdentityLookup}, }; use sp_staking::SessionIndex; use frame_support::{ parameter_types, traits::{ConstU32, ConstU64, GenesisBuild}, BasicExternalities, }; impl_opaque_keys! { pub struct MockSessionKeys { pub dummy: UintAuthorityId, } } impl From for MockSessionKeys { fn from(dummy: UintAuthorityId) -> Self { Self { dummy } } } pub const KEY_ID_A: KeyTypeId = KeyTypeId([4; 4]); pub const KEY_ID_B: KeyTypeId = KeyTypeId([9; 4]); #[derive(Debug, Clone, codec::Encode, codec::Decode, PartialEq, Eq)] pub struct PreUpgradeMockSessionKeys { pub a: [u8; 32], pub b: [u8; 64], } impl OpaqueKeys for PreUpgradeMockSessionKeys { type KeyTypeIdProviders = (); fn key_ids() -> &'static [KeyTypeId] { &[KEY_ID_A, KEY_ID_B] } fn get_raw(&self, i: KeyTypeId) -> &[u8] { match i { i if i == KEY_ID_A => &self.a[..], i if i == KEY_ID_B => &self.b[..], _ => &[], } } } type UncheckedExtrinsic = frame_system::mocking::MockUncheckedExtrinsic; type Block = frame_system::mocking::MockBlock; #[cfg(feature = "historical")] frame_support::construct_runtime!( pub enum Test where Block = Block, NodeBlock = Block, UncheckedExtrinsic = UncheckedExtrinsic, { System: frame_system::{Pallet, Call, Config, Storage, Event}, Session: pallet_session::{Pallet, Call, Storage, Event, Config}, Historical: pallet_session_historical::{Pallet}, } ); #[cfg(not(feature = "historical"))] frame_support::construct_runtime!( pub enum Test where Block = Block, NodeBlock = Block, UncheckedExtrinsic = UncheckedExtrinsic, { System: frame_system::{Pallet, Call, Config, Storage, Event}, Session: pallet_session::{Pallet, Call, Storage, Event, Config}, } ); parameter_types! { pub static Validators: Vec = vec![1, 2, 3]; pub static NextValidators: Vec = vec![1, 2, 3]; pub static Authorities: Vec = vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]; pub static ForceSessionEnd: bool = false; pub static SessionLength: u64 = 2; pub static SessionChanged: bool = false; pub static TestSessionChanged: bool = false; pub static Disabled: bool = false; // Stores if `on_before_session_end` was called pub static BeforeSessionEndCalled: bool = false; pub static ValidatorAccounts: BTreeMap = BTreeMap::new(); } pub struct TestShouldEndSession; impl ShouldEndSession for TestShouldEndSession { fn should_end_session(now: u64) -> bool { let l = SessionLength::get(); now % l == 0 || ForceSessionEnd::mutate(|l| { let r = *l; *l = false; r }) } } pub struct TestSessionHandler; impl SessionHandler for TestSessionHandler { const KEY_TYPE_IDS: &'static [sp_runtime::KeyTypeId] = &[UintAuthorityId::ID]; fn on_genesis_session(_validators: &[(u64, T)]) {} fn on_new_session( changed: bool, validators: &[(u64, T)], _queued_validators: &[(u64, T)], ) { SessionChanged::mutate(|l| *l = changed); Authorities::mutate(|l| { *l = validators .iter() .map(|(_, id)| id.get::(DUMMY).unwrap_or_default()) .collect() }); } fn on_disabled(_validator_index: u32) { Disabled::mutate(|l| *l = true) } fn on_before_session_ending() { BeforeSessionEndCalled::mutate(|b| *b = true); } } pub struct TestSessionManager; impl SessionManager for TestSessionManager { fn end_session(_: SessionIndex) {} fn start_session(_: SessionIndex) {} fn new_session(_: SessionIndex) -> Option> { if !TestSessionChanged::get() { Validators::mutate(|v| { *v = NextValidators::get().clone(); Some(v.clone()) }) } else if Disabled::mutate(|l| std::mem::replace(&mut *l, false)) { // If there was a disabled validator, underlying conditions have changed // so we return `Some`. Some(Validators::get().clone()) } else { None } } } #[cfg(feature = "historical")] impl crate::historical::SessionManager for TestSessionManager { fn end_session(_: SessionIndex) {} fn start_session(_: SessionIndex) {} fn new_session(new_index: SessionIndex) -> Option> { >::new_session(new_index) .map(|vals| vals.into_iter().map(|val| (val, val)).collect()) } } pub fn authorities() -> Vec { Authorities::get().to_vec() } pub fn force_new_session() { ForceSessionEnd::mutate(|l| *l = true) } pub fn set_session_length(x: u64) { SessionLength::mutate(|l| *l = x) } pub fn session_changed() -> bool { SessionChanged::get() } pub fn set_next_validators(next: Vec) { NextValidators::mutate(|v| *v = next); } pub fn before_session_end_called() -> bool { BeforeSessionEndCalled::get() } pub fn reset_before_session_end_called() { BeforeSessionEndCalled::mutate(|b| *b = false); } pub fn new_test_ext() -> sp_io::TestExternalities { let mut t = frame_system::GenesisConfig::default().build_storage::().unwrap(); let keys: Vec<_> = NextValidators::get() .iter() .cloned() .map(|i| (i, i, UintAuthorityId(i).into())) .collect(); BasicExternalities::execute_with_storage(&mut t, || { for (ref k, ..) in &keys { frame_system::Pallet::::inc_providers(k); } frame_system::Pallet::::inc_providers(&4); // An additional identity that we use. frame_system::Pallet::::inc_providers(&69); }); pallet_session::GenesisConfig:: { keys } .assimilate_storage(&mut t) .unwrap(); let v = NextValidators::get().iter().map(|&i| (i, i)).collect(); ValidatorAccounts::mutate(|m| *m = v); sp_io::TestExternalities::new(t) } impl frame_system::Config for Test { type BaseCallFilter = frame_support::traits::Everything; type BlockWeights = (); type BlockLength = (); type DbWeight = (); type RuntimeOrigin = RuntimeOrigin; type Index = u64; type BlockNumber = u64; type RuntimeCall = RuntimeCall; type Hash = H256; type Hashing = BlakeTwo256; type AccountId = u64; type Lookup = IdentityLookup; type Header = Header; type RuntimeEvent = RuntimeEvent; type BlockHashCount = ConstU64<250>; type Version = (); type PalletInfo = PalletInfo; type AccountData = (); type OnNewAccount = (); type OnKilledAccount = (); type SystemWeightInfo = (); type SS58Prefix = (); type OnSetCode = (); type MaxConsumers = ConstU32<16>; } impl pallet_timestamp::Config for Test { type Moment = u64; type OnTimestampSet = (); type MinimumPeriod = ConstU64<5>; type WeightInfo = (); } pub struct TestValidatorIdOf; impl TestValidatorIdOf { pub fn set(v: BTreeMap) { ValidatorAccounts::mutate(|m| *m = v); } } impl Convert> for TestValidatorIdOf { fn convert(x: u64) -> Option { ValidatorAccounts::get().get(&x).cloned() } } impl Config for Test { type ShouldEndSession = TestShouldEndSession; #[cfg(feature = "historical")] type SessionManager = crate::historical::NoteHistoricalRoot; #[cfg(not(feature = "historical"))] type SessionManager = TestSessionManager; type SessionHandler = TestSessionHandler; type ValidatorId = u64; type ValidatorIdOf = TestValidatorIdOf; type Keys = MockSessionKeys; type RuntimeEvent = RuntimeEvent; type NextSessionRotation = (); type WeightInfo = (); } #[cfg(feature = "historical")] impl crate::historical::Config for Test { type FullIdentification = u64; type FullIdentificationOf = sp_runtime::traits::ConvertInto; }