#![cfg_attr(not(feature = "std"), no_std)]
use frame_support::{
decl_error, decl_event, decl_module, decl_storage, ensure, traits::LockIdentifier,
};
use governance_os_support::traits::{
Currencies, LockableCurrencies, ProposalResult, StandardizedVoting,
};
use sp_runtime::{
traits::{Hash, Saturating, Zero},
DispatchError, DispatchResult, Perbill,
};
use sp_std::prelude::*;
use types::ProposalState;
#[cfg(test)]
mod tests;
mod types;
pub use types::{VoteData, VotingParameters};
pub const PLCR_VOTING_LOCK_ID: LockIdentifier = *b"plcrvote";
pub trait Config: frame_system::Config {
type Event: From<Event<Self>> + Into<<Self as frame_system::Config>::Event>;
type Currencies: LockableCurrencies<Self::AccountId>;
}
type BalanceOf<T> =
<<T as Config>::Currencies as Currencies<<T as frame_system::Config>::AccountId>>::Balance;
type CurrencyIdOf<T> =
<<T as Config>::Currencies as Currencies<<T as frame_system::Config>::AccountId>>::CurrencyId;
type PlcrProposalStateOf<T> =
ProposalState<BalanceOf<T>, <T as frame_system::Config>::BlockNumber, CurrencyIdOf<T>>;
type PlcrVoteData<Balance, Hash> = VoteData<Balance, Hash>;
decl_storage! {
trait Store for Module<T: Config> as PlcrVoting {
pub Proposals get(fn proposals): map hasher(blake2_128_concat) T::Hash => PlcrProposalStateOf<T>;
pub Locks get(fn locks): map hasher(blake2_128_concat) (CurrencyIdOf<T>, T::AccountId) => Vec<(T::Hash, BalanceOf<T>)>;
pub Votes get(fn votes): double_map hasher(blake2_128_concat) T::Hash, hasher(blake2_128_concat) T::AccountId => PlcrVoteData<BalanceOf<T>, T::Hash>;
}
}
decl_error! {
pub enum Error for Module<T: Config> {
RevealCommitMismatch,
NoCommitFound,
Revealed,
NotEnoughBalance,
Phase,
TooEarly,
}
}
decl_event!(
pub enum Event<T>
where
Hash = <T as frame_system::Config>::Hash,
AccountId = <T as frame_system::Config>::AccountId,
{
VoteCommited(AccountId, Hash, Hash),
VoteRevealed(AccountId, Hash, Hash),
}
);
decl_module! {
pub struct Module<T: Config> for enum Call where origin: T::Origin {
fn deposit_event() = default;
}
}
impl<T: Config> StandardizedVoting for Module<T> {
type ProposalId = T::Hash;
type Parameters = VotingParameters<T::BlockNumber, CurrencyIdOf<T>>;
type VoteData = VoteData<BalanceOf<T>, T::Hash>;
type AccountId = T::AccountId;
fn initiate(proposal: Self::ProposalId, parameters: Self::Parameters) -> DispatchResult {
Proposals::<T>::try_mutate_exists(proposal, |maybe_existing_state| -> DispatchResult {
*maybe_existing_state = Some(ProposalState {
parameters,
created_on: Self::now(),
revealed_against: Zero::zero(),
revealed_favorable: Zero::zero(),
});
Ok(())
})?;
Ok(())
}
fn veto(proposal: Self::ProposalId) -> DispatchResult {
Self::finalize_proposal(proposal, Self::proposals(proposal))
}
fn vote(
proposal: Self::ProposalId,
voter: &Self::AccountId,
data: Self::VoteData,
) -> DispatchResult {
let mut state = Self::proposals(proposal);
let commit_phase_ends_on = state
.created_on
.saturating_add(state.parameters.commit_duration);
let reveal_phase_ends_on =
commit_phase_ends_on.saturating_add(state.parameters.reveal_duration);
match data {
VoteData::Commit(hash) => {
if let VoteData::Reveal(_, _, _) = Self::votes(proposal, voter) {
return Err(Error::<T>::Revealed.into());
}
ensure!(Self::now() < commit_phase_ends_on, Error::<T>::Phase);
Self::deposit_event(RawEvent::VoteCommited(voter.clone(), proposal, hash));
}
VoteData::Reveal(balance, support, salt) => {
if let VoteData::Commit(hash) = Self::votes(proposal, voter) {
ensure!(
Self::now() > commit_phase_ends_on && Self::now() < reveal_phase_ends_on,
Error::<T>::Phase
);
ensure!(
T::Currencies::free_balance(state.parameters.voting_currency, voter)
>= balance,
Error::<T>::NotEnoughBalance
);
let hashed_reveal = T::Hashing::hash_of(&(balance, support, salt));
ensure!(hashed_reveal == hash, Error::<T>::RevealCommitMismatch);
Self::lock(proposal, state.parameters.voting_currency, voter, balance)?;
state.add_support(support, balance);
Proposals::<T>::insert(proposal, state);
Self::deposit_event(RawEvent::VoteRevealed(voter.clone(), proposal, hash));
} else {
return Err(Error::<T>::NoCommitFound.into());
}
}
};
Votes::<T>::insert(proposal, voter, data);
Ok(())
}
fn close(proposal: Self::ProposalId) -> Result<ProposalResult, DispatchError> {
let state = Self::proposals(proposal);
let proposal_expired = Self::now()
> state
.created_on
.saturating_add(state.parameters.commit_duration)
.saturating_add(state.parameters.reveal_duration);
let total_supply = T::Currencies::total_issuance(state.parameters.voting_currency);
let total_participation = state
.revealed_against
.saturating_add(state.revealed_favorable);
let participation_met = total_participation
> Perbill::from_percent(state.parameters.min_participation) * total_supply;
let quorum_met = state.revealed_favorable
> Perbill::from_percent(state.parameters.min_quorum) * total_participation;
let proposal_passing = quorum_met && participation_met;
ensure!(proposal_expired || proposal_passing, Error::<T>::TooEarly);
Self::finalize_proposal(proposal, state)?;
Ok(if proposal_passing {
ProposalResult::Passing
} else {
ProposalResult::Failing
})
}
}
impl<T: Config> Module<T> {
fn now() -> T::BlockNumber {
frame_system::Module::<T>::block_number()
}
fn lock(
proposal: T::Hash,
currency: CurrencyIdOf<T>,
who: &T::AccountId,
amount: BalanceOf<T>,
) -> DispatchResult {
Locks::<T>::try_mutate((currency, who), |locks| -> DispatchResult {
let locks_addition = vec![(proposal, amount)];
*locks = locks
.iter()
.cloned()
.filter(|&(maybe_duplicate_proposal, _locked_amount)| {
maybe_duplicate_proposal != proposal
})
.chain(locks_addition.iter().cloned())
.collect();
Self::rejig_locks(locks.to_vec(), currency, who)?;
Ok(())
})?;
Ok(())
}
fn unlock(proposal: T::Hash, currency: CurrencyIdOf<T>, who: &T::AccountId) -> DispatchResult {
let mut lock_data = Locks::<T>::get((currency, who));
lock_data = lock_data
.iter()
.cloned()
.filter(|&(maybe_duplicate_proposal, _locked_amount)| {
maybe_duplicate_proposal != proposal
})
.collect();
if lock_data.is_empty() {
Locks::<T>::remove((currency, who));
} else {
Locks::<T>::insert((currency, who), lock_data.clone());
}
Self::rejig_locks(lock_data, currency, who)?;
Ok(())
}
fn rejig_locks(
locks: Vec<(T::Hash, BalanceOf<T>)>,
currency: CurrencyIdOf<T>,
who: &T::AccountId,
) -> DispatchResult {
let max_to_lock =
locks
.into_iter()
.fold(Zero::zero(), |acc, (_proposal, locked_amount)| {
if acc < locked_amount {
locked_amount
} else {
acc
}
});
if max_to_lock == Zero::zero() {
T::Currencies::remove_lock(currency, PLCR_VOTING_LOCK_ID, who)?;
} else {
T::Currencies::set_lock(currency, PLCR_VOTING_LOCK_ID, who, max_to_lock)?;
}
Ok(())
}
fn finalize_proposal(proposal: T::Hash, state: PlcrProposalStateOf<T>) -> DispatchResult {
Votes::<T>::iter_prefix(proposal).try_for_each(|(account, _vote)| -> DispatchResult {
Self::unlock(proposal, state.parameters.voting_currency, &account)?;
Ok(())
})?;
Proposals::<T>::remove(proposal);
Votes::<T>::remove_prefix(proposal);
Ok(())
}
}