// 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 . //! Council system: Handles the voting in and maintenance of council members. #![cfg_attr(not(feature = "std"), no_std)] pub mod motions; pub mod seats; pub use crate::seats::{Trait, Module, RawEvent, Event, VoteIndex}; /// Trait for type that can handle incremental changes to a set of account IDs. pub trait OnMembersChanged { /// A number of members `new` just joined the set and replaced some `old` ones. fn on_members_changed(new: &[AccountId], old: &[AccountId]); } impl OnMembersChanged for () { fn on_members_changed(_new: &[T], _old: &[T]) {} } #[cfg(test)] mod tests { // These re-exports are here for a reason, edit with care pub use super::*; pub use runtime_io::with_externalities; use srml_support::{impl_outer_origin, impl_outer_event, impl_outer_dispatch, parameter_types}; pub use substrate_primitives::{H256, Blake2Hasher, u32_trait::{_1, _2, _3, _4}}; pub use primitives::{ BuildStorage, traits::{BlakeTwo256, IdentityLookup}, testing::{Digest, DigestItem, Header} }; pub use {seats, motions}; impl_outer_origin! { pub enum Origin for Test { motions } } impl_outer_event! { pub enum Event for Test { balances, democracy, seats, motions, } } impl_outer_dispatch! { pub enum Call for Test where origin: Origin { balances::Balances, democracy::Democracy, } } // Workaround for https://github.com/rust-lang/rust/issues/26925 . Remove when sorted. #[derive(Clone, Eq, PartialEq, Debug)] pub struct Test; impl system::Trait for Test { type Origin = Origin; type Index = u64; type BlockNumber = u64; type Hash = H256; type Hashing = BlakeTwo256; type Digest = Digest; type AccountId = u64; type Lookup = IdentityLookup; type Header = Header; type Event = Event; type Log = DigestItem; } impl balances::Trait for Test { type Balance = u64; type OnFreeBalanceZero = (); type OnNewAccount = (); type Event = Event; type TransactionPayment = (); type TransferPayment = (); type DustRemoval = (); } parameter_types! { pub const LaunchPeriod: u64 = 1; pub const VotingPeriod: u64 = 3; pub const MinimumDeposit: u64 = 1; pub const EnactmentPeriod: u64 = 0; pub const CooloffPeriod: u64 = 2; } impl democracy::Trait for Test { type Proposal = Call; type Event = Event; type Currency = balances::Module; type EnactmentPeriod = EnactmentPeriod; type LaunchPeriod = LaunchPeriod; type EmergencyVotingPeriod = VotingPeriod; type VotingPeriod = VotingPeriod; type MinimumDeposit = MinimumDeposit; type ExternalOrigin = motions::EnsureProportionAtLeast<_1, _2, u64>; type ExternalMajorityOrigin = motions::EnsureProportionAtLeast<_2, _3, u64>; type EmergencyOrigin = motions::EnsureProportionAtLeast<_1, _1, u64>; type CancellationOrigin = motions::EnsureProportionAtLeast<_2, _3, u64>; type VetoOrigin = motions::EnsureMember; type CooloffPeriod = CooloffPeriod; } impl seats::Trait for Test { type Event = Event; type BadPresentation = (); type BadReaper = (); type BadVoterIndex = (); type LoserCandidate = (); type OnMembersChanged = CouncilMotions; } impl motions::Trait for Test { type Origin = Origin; type Proposal = Call; type Event = Event; } pub struct ExtBuilder { balance_factor: u64, decay_ratio: u32, voting_fee: u64, voter_bond: u64, bad_presentation_punishment: u64, with_council: bool, } impl Default for ExtBuilder { fn default() -> Self { Self { balance_factor: 1, decay_ratio: 24, voting_fee: 0, voter_bond: 0, bad_presentation_punishment: 1, with_council: false, } } } impl ExtBuilder { pub fn with_council(mut self, council: bool) -> Self { self.with_council = council; self } pub fn balance_factor(mut self, factor: u64) -> Self { self.balance_factor = factor; self } pub fn decay_ratio(mut self, ratio: u32) -> Self { self.decay_ratio = ratio; self } pub fn voting_fee(mut self, fee: u64) -> Self { self.voting_fee = fee; self } pub fn bad_presentation_punishment(mut self, fee: u64) -> Self { self.bad_presentation_punishment = fee; self } pub fn voter_bond(mut self, fee: u64) -> Self { self.voter_bond = fee; self } pub fn build(self) -> runtime_io::TestExternalities { let mut t = system::GenesisConfig::::default().build_storage().unwrap().0; t.extend(balances::GenesisConfig::{ transaction_base_fee: 0, transaction_byte_fee: 0, balances: vec![ (1, 10 * self.balance_factor), (2, 20 * self.balance_factor), (3, 30 * self.balance_factor), (4, 40 * self.balance_factor), (5, 50 * self.balance_factor), (6, 60 * self.balance_factor) ], existential_deposit: 0, transfer_fee: 0, creation_fee: 0, vesting: vec![], }.build_storage().unwrap().0); t.extend(seats::GenesisConfig:: { candidacy_bond: 3, voter_bond: self.voter_bond, present_slash_per_voter: self.bad_presentation_punishment, carry_count: 2, inactive_grace_period: 1, active_council: if self.with_council { vec![ (1, 10), (2, 10), (3, 10) ] } else { vec![] }, approval_voting_period: 4, presentation_duration: 2, desired_seats: 2, decay_ratio: self.decay_ratio, voting_fee: self.voting_fee, term_duration: 5, }.build_storage().unwrap().0); runtime_io::TestExternalities::new(t) } } pub type System = system::Module; pub type Balances = balances::Module; pub type Democracy = democracy::Module; pub type Council = seats::Module; pub type CouncilMotions = motions::Module; }