#![cfg_attr(not(feature = "std"), no_std)]
use frame_support::{decl_error, decl_module, decl_storage, ensure, traits::LockIdentifier};
use governance_os_support::traits::{
Currencies, LockableCurrencies, ProposalResult, StandardizedVoting,
};
use sp_runtime::{
traits::{Saturating, Zero},
DispatchError, DispatchResult, Perbill,
};
use sp_std::{prelude::*, vec::Vec};
use types::ProposalState;
#[cfg(test)]
mod tests;
mod types;
pub use types::{VoteCountingStrategy, VoteData, VotingParameters};
pub const COIN_VOTING_LOCK_ID: LockIdentifier = *b"coinvote";
pub trait Config: frame_system::Config {
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 LockDataOf<T> = (
<T as frame_system::Config>::Hash,
bool,
BalanceOf<T>,
VoteCountingStrategy,
);
type LockIdentifierOf<T> = (CurrencyIdOf<T>, <T as frame_system::Config>::AccountId);
type CoinProposalStateOf<T> = ProposalState<
BalanceOf<T>,
<T as frame_system::Config>::BlockNumber,
CurrencyIdOf<T>,
LockIdentifierOf<T>,
>;
decl_storage! {
trait Store for Module<T: Config> as CoinVoting {
pub Proposals get(fn proposals): map hasher(blake2_128_concat) T::Hash => CoinProposalStateOf<T>;
pub Locks get(fn locks): map hasher(blake2_128_concat) LockIdentifierOf<T> => Vec<LockDataOf<T>>;
}
}
decl_error! {
pub enum Error for Module<T: Config> {
NotEnoughBalance,
CannotClose,
}
}
decl_module! {
pub struct Module<T: Config> for enum Call where origin: T::Origin {
}
}
impl<T: Config> StandardizedVoting for Module<T> {
type ProposalId = T::Hash;
type Parameters = VotingParameters<T::BlockNumber, CurrencyIdOf<T>>;
type VoteData = VoteData<BalanceOf<T>>;
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,
total_against: Zero::zero(),
total_favorable: Zero::zero(),
locks: vec![],
created_on: Self::now(),
});
Ok(())
})?;
Ok(())
}
fn vote(
proposal: Self::ProposalId,
voter: &Self::AccountId,
data: Self::VoteData,
) -> DispatchResult {
let mut state = Self::proposals(proposal);
ensure!(
T::Currencies::free_balance(state.parameters.voting_currency, voter) >= data.power,
Error::<T>::NotEnoughBalance
);
let mut this_vote_is_a_duplicate = false;
Self::update_locks(
state.parameters.voting_currency,
voter,
proposal,
data.in_support,
data.power,
state.parameters.vote_counting_strategy,
|_proposal, old_support, old_power, _strategy| {
state.unrecord_vote(old_support, old_power);
this_vote_is_a_duplicate = true;
},
)?;
if !this_vote_is_a_duplicate {
state
.locks
.push((state.parameters.voting_currency, voter.clone()));
}
state.record_vote(data.in_support, data.power);
Proposals::<T>::insert(proposal, state);
Ok(())
}
fn veto(proposal: Self::ProposalId) -> DispatchResult {
Self::unlock(Proposals::<T>::take(proposal).locks, proposal)
}
fn close(proposal: Self::ProposalId) -> Result<ProposalResult, DispatchError> {
let state = Proposals::<T>::get(proposal);
let total_supply = T::Currencies::total_issuance(state.parameters.voting_currency);
let total_participation = state.total_against.saturating_add(state.total_favorable);
let enough_participation = total_participation
> Perbill::from_percent(state.parameters.min_participation) * total_supply;
let enough_quorum = state.total_favorable
> Perbill::from_percent(state.parameters.min_quorum) * total_participation;
let result = if enough_participation && enough_quorum {
ProposalResult::Passing
} else {
ProposalResult::Failing
};
let can_close = state.created_on.saturating_add(state.parameters.ttl) < Self::now();
ensure!(
can_close || result == ProposalResult::Passing,
Error::<T>::CannotClose
);
Self::unlock(state.locks, proposal)?;
Proposals::<T>::remove(proposal);
Ok(result)
}
}
impl<T: Config> Module<T> {
fn update_locks<F>(
voting_currency: CurrencyIdOf<T>,
voter: &T::AccountId,
proposal: T::Hash,
support: bool,
power: BalanceOf<T>,
strategy: VoteCountingStrategy,
mut on_duplicate_vote_found: F,
) -> DispatchResult
where
F: FnMut(T::Hash, bool, BalanceOf<T>, VoteCountingStrategy),
{
Locks::<T>::try_mutate((voting_currency, voter), |locks| -> DispatchResult {
let locks_addition = vec![(proposal, support, power, strategy)];
*locks = locks
.iter()
.cloned()
.filter(|&(maybe_duplicate_proposal, support, power, strategy)| {
if maybe_duplicate_proposal != proposal {
true
} else {
on_duplicate_vote_found(maybe_duplicate_proposal, support, power, strategy);
false
}
})
.chain(locks_addition.iter().cloned())
.collect();
Self::rejig_locks(locks.to_vec(), voting_currency, voter)?;
Ok(())
})?;
Ok(())
}
fn unlock(locks: Vec<LockIdentifierOf<T>>, proposal: T::Hash) -> DispatchResult {
locks
.iter()
.try_for_each(|lock_identifier| -> DispatchResult {
let lock_data = Locks::<T>::get(lock_identifier);
let new_lock_data: Vec<LockDataOf<T>> = lock_data
.iter()
.cloned()
.filter(|(prop, _, _, _)| prop != &proposal)
.collect();
if !new_lock_data.is_empty() {
Locks::<T>::insert(lock_identifier, new_lock_data.clone());
} else {
Locks::<T>::remove(lock_identifier);
}
Self::rejig_locks(new_lock_data, lock_identifier.0, &lock_identifier.1)?;
Ok(())
})?;
Ok(())
}
fn rejig_locks(
locks: Vec<LockDataOf<T>>,
voting_currency: CurrencyIdOf<T>,
who: &T::AccountId,
) -> DispatchResult {
let max_to_lock_with_simple_voting: BalanceOf<T> = locks
.iter()
.cloned()
.filter(|(_proposal, _support, _power, strategy)| {
strategy == &VoteCountingStrategy::Simple
})
.fold(
Zero::zero(),
|acc, (_proposal, _support, power, _strategy)| {
if power > acc {
power
} else {
acc
}
},
);
let max_to_lock_with_quadratic_voting: BalanceOf<T> = locks
.iter()
.cloned()
.filter(|(_proposal, _support, _power, strategy)| {
strategy == &VoteCountingStrategy::Quadratic
})
.fold(
Zero::zero(),
|acc, (_proposal, _support, power, _strategy)| acc.saturating_add(power),
);
let max_to_lock =
max_to_lock_with_simple_voting.saturating_add(max_to_lock_with_quadratic_voting);
if max_to_lock == Zero::zero() {
T::Currencies::remove_lock(voting_currency, COIN_VOTING_LOCK_ID, who)?;
} else {
T::Currencies::set_lock(voting_currency, COIN_VOTING_LOCK_ID, who, max_to_lock)?;
}
Ok(())
}
fn now() -> T::BlockNumber {
frame_system::Module::<T>::block_number()
}
}