// Copyright 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 . //! Mock helpers for Session. use super::*; use std::cell::RefCell; use srml_support::{impl_outer_origin, parameter_types}; use primitives::H256; use sr_primitives::{ Perbill, traits::{BlakeTwo256, IdentityLookup, ConvertInto}, testing::{Header, UintAuthorityId} }; impl_outer_origin! { pub enum Origin for Test {} } thread_local! { pub static NEXT_VALIDATORS: RefCell> = RefCell::new(vec![1, 2, 3]); pub static AUTHORITIES: RefCell> = RefCell::new(vec![UintAuthorityId(1), UintAuthorityId(2), UintAuthorityId(3)]); pub static FORCE_SESSION_END: RefCell = RefCell::new(false); pub static SESSION_LENGTH: RefCell = RefCell::new(2); pub static SESSION_CHANGED: RefCell = RefCell::new(false); pub static TEST_SESSION_CHANGED: RefCell = RefCell::new(false); } pub struct TestShouldEndSession; impl ShouldEndSession for TestShouldEndSession { fn should_end_session(now: u64) -> bool { let l = SESSION_LENGTH.with(|l| *l.borrow()); now % l == 0 || FORCE_SESSION_END.with(|l| { let r = *l.borrow(); *l.borrow_mut() = false; r }) } } pub struct TestSessionHandler; impl SessionHandler for TestSessionHandler { fn on_new_session( changed: bool, validators: &[(u64, T)], _queued_validators: &[(u64, T)], ) { SESSION_CHANGED.with(|l| *l.borrow_mut() = changed); AUTHORITIES.with(|l| *l.borrow_mut() = validators.iter().map(|(_, id)| id.get::(0).unwrap_or_default()).collect() ); } fn on_disabled(_validator_index: usize) {} } pub struct TestOnSessionEnding; impl OnSessionEnding for TestOnSessionEnding { fn on_session_ending(_: SessionIndex, _: SessionIndex) -> Option> { if !TEST_SESSION_CHANGED.with(|l| *l.borrow()) { Some(NEXT_VALIDATORS.with(|l| l.borrow().clone())) } else { None } } } #[cfg(feature = "historical")] impl crate::historical::OnSessionEnding for TestOnSessionEnding { fn on_session_ending(_: SessionIndex, _: SessionIndex) -> Option<(Vec, Vec<(u64, u64)>)> { if !TEST_SESSION_CHANGED.with(|l| *l.borrow()) { let last_validators = Session::validators(); let last_identifications = last_validators.into_iter().map(|v| (v, v)).collect(); Some((NEXT_VALIDATORS.with(|l| l.borrow().clone()), last_identifications)) } else { None } } } pub fn authorities() -> Vec { AUTHORITIES.with(|l| l.borrow().to_vec()) } pub fn force_new_session() { FORCE_SESSION_END.with(|l| *l.borrow_mut() = true ) } pub fn set_session_length(x: u64) { SESSION_LENGTH.with(|l| *l.borrow_mut() = x ) } pub fn session_changed() -> bool { SESSION_CHANGED.with(|l| *l.borrow()) } pub fn set_next_validators(next: Vec) { NEXT_VALIDATORS.with(|v| *v.borrow_mut() = next); } #[derive(Clone, Eq, PartialEq)] pub struct Test; parameter_types! { pub const BlockHashCount: u64 = 250; pub const MaximumBlockWeight: u32 = 1024; pub const MaximumBlockLength: u32 = 2 * 1024; pub const MinimumPeriod: u64 = 5; pub const AvailableBlockRatio: Perbill = Perbill::one(); } impl system::Trait for Test { type Origin = Origin; type Index = u64; type BlockNumber = u64; type Hash = H256; type Hashing = BlakeTwo256; type AccountId = u64; type Lookup = IdentityLookup; type Header = Header; type WeightMultiplierUpdate = (); type Event = (); type BlockHashCount = BlockHashCount; type MaximumBlockWeight = MaximumBlockWeight; type AvailableBlockRatio = AvailableBlockRatio; type MaximumBlockLength = MaximumBlockLength; } impl timestamp::Trait for Test { type Moment = u64; type OnTimestampSet = (); type MinimumPeriod = MinimumPeriod; } impl Trait for Test { type ShouldEndSession = TestShouldEndSession; #[cfg(feature = "historical")] type OnSessionEnding = crate::historical::NoteHistoricalRoot; #[cfg(not(feature = "historical"))] type OnSessionEnding = TestOnSessionEnding; type SessionHandler = TestSessionHandler; type ValidatorId = u64; type ValidatorIdOf = ConvertInto; type Keys = UintAuthorityId; type Event = (); type SelectInitialValidators = (); } #[cfg(feature = "historical")] impl crate::historical::Trait for Test { type FullIdentification = u64; type FullIdentificationOf = sr_primitives::traits::ConvertInto; } pub type System = system::Module; pub type Session = Module;