// Copyright 2018-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 . //! Test utilities #![cfg(test)] use primitives::{traits::IdentityLookup, BuildStorage, Perbill}; use primitives::testing::{Digest, DigestItem, Header, UintAuthorityId, ConvertUintAuthorityId}; use substrate_primitives::{H256, Blake2Hasher}; use runtime_io; use srml_support::impl_outer_origin; use crate::{GenesisConfig, Module, Trait, StakerStatus}; // The AccountId alias in this test module. pub type AccountIdType = u64; impl_outer_origin!{ pub enum Origin for Test {} } // Workaround for https://github.com/rust-lang/rust/issues/26925 . Remove when sorted. #[derive(Clone, PartialEq, Eq, Debug)] pub struct Test; impl consensus::Trait for Test { type Log = DigestItem; type SessionKey = UintAuthorityId; type InherentOfflineReport = (); } impl system::Trait for Test { type Origin = Origin; type Index = u64; type BlockNumber = u64; type Hash = H256; type Hashing = ::primitives::traits::BlakeTwo256; type Digest = Digest; type AccountId = AccountIdType; type Lookup = IdentityLookup; type Header = Header; type Event = (); type Log = DigestItem; } impl balances::Trait for Test { type Balance = u64; type OnFreeBalanceZero = Staking; type OnNewAccount = (); type Event = (); } impl session::Trait for Test { type ConvertAccountIdToSessionKey = ConvertUintAuthorityId; type OnSessionChange = Staking; type Event = (); } impl timestamp::Trait for Test { type Moment = u64; type OnTimestampSet = (); } impl Trait for Test { type Currency = balances::Module; type OnRewardMinted = (); type Event = (); } pub struct ExtBuilder { existential_deposit: u64, session_length: u64, sessions_per_era: u64, current_era: u64, monied: bool, reward: u64, validator_pool: bool, nominate: bool, validator_count: u32, minimum_validator_count: u32, } impl Default for ExtBuilder { fn default() -> Self { Self { existential_deposit: 0, session_length: 1, sessions_per_era: 1, current_era: 0, monied: true, reward: 10, validator_pool: false, nominate: true, validator_count: 2, minimum_validator_count: 0, } } } impl ExtBuilder { pub fn existential_deposit(mut self, existential_deposit: u64) -> Self { self.existential_deposit = existential_deposit; self } pub fn session_length(mut self, session_length: u64) -> Self { self.session_length = session_length; self } pub fn sessions_per_era(mut self, sessions_per_era: u64) -> Self { self.sessions_per_era = sessions_per_era; self } pub fn _current_era(mut self, current_era: u64) -> Self { self.current_era = current_era; self } pub fn _monied(mut self, monied: bool) -> Self { self.monied = monied; self } pub fn reward(mut self, reward: u64) -> Self { self.reward = reward; self } pub fn validator_pool(mut self, validator_pool: bool) -> Self { // NOTE: this should only be set to true with monied = false. self.validator_pool = validator_pool; self } pub fn nominate(mut self, nominate: bool) -> Self { // NOTE: this only sets a dummy nominator for tests that want 10 and 20 (default validators) to be chosen by default. self.nominate = nominate; self } pub fn validator_count(mut self, count: u32) -> Self { self.validator_count = count; self } pub fn minimum_validator_count(mut self, count: u32) -> Self { self.minimum_validator_count = count; self } pub fn build(self) -> runtime_io::TestExternalities { let (mut t, mut c) = system::GenesisConfig::::default().build_storage().unwrap(); let balance_factor = if self.existential_deposit > 0 { 256 } else { 1 }; let _ = consensus::GenesisConfig::{ code: vec![], authorities: vec![], }.assimilate_storage(&mut t, &mut c); let _ = session::GenesisConfig::{ session_length: self.session_length, // NOTE: if config.nominate == false then 100 is also selected in the initial round. validators: if self.validator_pool { vec![10, 20, 30, 40] } else { vec![10, 20] }, keys: vec![], }.assimilate_storage(&mut t, &mut c); let _ = balances::GenesisConfig::{ balances: if self.monied { if self.reward > 0 { vec![ (1, 10 * balance_factor), (2, 20 * balance_factor), (3, 300 * balance_factor), (4, 400 * balance_factor), (10, balance_factor), (11, balance_factor * 1000), (20, balance_factor), (21, balance_factor * 2000), (100, 2000 * balance_factor), (101, 2000 * balance_factor), ] } else { vec![ (1, 10 * balance_factor), (2, 20 * balance_factor), (3, 300 * balance_factor), (4, 400 * balance_factor) ] } } else { vec![ (10, balance_factor), (11, balance_factor * 10), (20, balance_factor), (21, balance_factor * 20), (30, balance_factor), (31, balance_factor * 30), (40, balance_factor), (41, balance_factor * 40) ] }, existential_deposit: self.existential_deposit, transfer_fee: 0, creation_fee: 0, vesting: vec![], }.assimilate_storage(&mut t, &mut c); let _ = GenesisConfig::{ sessions_per_era: self.sessions_per_era, current_era: self.current_era, stakers: if self.validator_pool { vec![ (11, 10, balance_factor * 1000, StakerStatus::::Validator), (21, 20, balance_factor * 2000, StakerStatus::::Validator), (31, 30, balance_factor * 3000, if self.validator_pool { StakerStatus::::Validator } else { StakerStatus::::Idle }), (41, 40, balance_factor * 4000, if self.validator_pool { StakerStatus::::Validator } else { StakerStatus::::Idle }), // nominator (101, 100, balance_factor * 500, if self.nominate { StakerStatus::::Nominator(vec![10, 20]) } else { StakerStatus::::Nominator(vec![]) }) ] } else { vec![ (11, 10, balance_factor * 1000, StakerStatus::::Validator), (21, 20, balance_factor * 2000, StakerStatus::::Validator), // nominator (101, 100, balance_factor * 500, if self.nominate { StakerStatus::::Nominator(vec![10, 20]) } else { StakerStatus::::Nominator(vec![]) }) ] }, validator_count: self.validator_count, minimum_validator_count: self.minimum_validator_count, bonding_duration: self.sessions_per_era * self.session_length * 3, session_reward: Perbill::from_millionths((1000000 * self.reward / balance_factor) as u32), offline_slash: if self.monied { Perbill::from_percent(40) } else { Perbill::zero() }, current_session_reward: self.reward, current_offline_slash: 20, offline_slash_grace: 0, invulnerables: vec![], }.assimilate_storage(&mut t, &mut c); let _ = timestamp::GenesisConfig::{ period: 5, }.assimilate_storage(&mut t, &mut c); t.into() } } pub type System = system::Module; pub type Balances = balances::Module; pub type Session = session::Module; pub type Timestamp = timestamp::Module; pub type Staking = Module;