1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
/* * Copyright 2020 Nuclei Studio OÜ * * 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. */ //! A set of common traits to voting systems. use codec::{Decode, Encode}; use frame_support::Parameter; #[cfg(feature = "std")] use serde::{Deserialize, Serialize}; use sp_runtime::{DispatchError, DispatchResult, RuntimeDebug}; use sp_std::result; /// End result of a proposal being closed. #[derive(Encode, Decode, Clone, PartialEq, Eq, RuntimeDebug)] #[cfg_attr(feature = "std", derive(Serialize, Deserialize))] pub enum ProposalResult { Passing, Failing, } /// A common trait accross all voting implementations to make it easy to change /// between voting models or implementations. /// A pallet implementing this trait is not necessarily in charge of storing /// proposals but could stick to only to support the actual decison making /// code while proposal storage is delegated to another pallet. pub trait StandardizedVoting { /// How we represent the a proposal as passed to the underlying functions. /// This can be used to fetch any state associated to the proposal. type ProposalId; /// How the parameters of a voting system are represented and set at the /// organization level. type Parameters; /// How voting data is passed to the underlying pallet. type VoteData; /// How accounts are represented, used to identify voters. type AccountId; /// A proposal is being created. Handle any eventual registration and trigger /// an error if any preconditions are not met. Shall be called before any other /// state changes so that it is safe to fail here. It is the caller's responsibility /// to try and prevent overwrites or duplicated proposals. fn initiate(proposal: Self::ProposalId, parameters: Self::Parameters) -> DispatchResult; /// Special function to handle the case when a proposal is being vetoed. This /// should clean any storage or state associated to the given proposal. fn veto(proposal: Self::ProposalId) -> DispatchResult; /// Handle the reception of a new vote for the given proposal. This should mutate any /// state linked to the proposal accordingly. fn vote( proposal: Self::ProposalId, voter: &Self::AccountId, data: Self::VoteData, ) -> DispatchResult; /// Handle the closure of a proposal or return an error if it cannot be closed because /// some conditions are not met. Shall return an indicator on wether the proposal is /// passing (should be executed) or not (should be discarded). fn close(proposal: Self::ProposalId) -> result::Result<ProposalResult, DispatchError>; } /// Used to route votes and related actions between different voting system implementations. pub trait VotingRouter { /// How accounts are represented, used to identify voters. type AccountId; /// How the runtime defines a voting system. And how users can select it. Typically this /// would be a rust `enum`. type VotingSystemId: Parameter; /// How the parameters of a voting system are represented and set at the /// organization level. Typically an `enum` to account for all the different /// parameters for the different voting systems. type Parameters: Parameter; /// How we represent a proposal. Typically a `Hash`. type ProposalId: Parameter; /// How the runtime represents the different vote data of the different voting systems. /// Typically an `enum` to account for all the different voting systems. type VoteData: Parameter; /// Route the `initiate` call to the right `StandardizedVoting` implementation based /// on the value of `voting_systems`. fn initiate( voting_system: Self::VotingSystemId, proposal: Self::ProposalId, parameters: Self::Parameters, ) -> DispatchResult; /// Route the `veto` call to the right `StandardizedVoting` implementation based /// on the value of `voting_systems`. fn veto(voting_system: Self::VotingSystemId, proposal: Self::ProposalId) -> DispatchResult; /// Route the `vote` call to the right `StandardizedVoting` implementation based /// on the value of `voting_systems`. fn vote( voting_system: Self::VotingSystemId, proposal: Self::ProposalId, voter: &Self::AccountId, data: Self::VoteData, ) -> DispatchResult; /// Route the `close` call to the right `StandardizedVoting` implementation based /// on the value of `voting_systems`. fn close( voting_system: Self::VotingSystemId, proposal: Self::ProposalId, ) -> result::Result<ProposalResult, DispatchError>; }