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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
/*
 * 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.
 */

//! This pallet can be used to deploy a PLCR voting system. With PLCR
//! voting, votes are hashed before being revealed and counted. This
//! allows people to hide their votes until everyone places theirs as
//! to prevent collusion between participants or typical human biases.

#![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 {
    /// Because this pallet emits events, it depends on the runtime's definition of an event.
    type Event: From<Event<Self>> + Into<<Self as frame_system::Config>::Event>;
    /// Pallet in charge of currencies. Used so that we can lock tokens etc...
    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 {
        /// Proposals actively opened and linked to this voting implementation. Erased when closed or vetoed.
        pub Proposals get(fn proposals): map hasher(blake2_128_concat) T::Hash => PlcrProposalStateOf<T>;
        /// Keeps track of locks set on user's balances and to which proposal they were linked to.
        pub Locks get(fn locks): map hasher(blake2_128_concat) (CurrencyIdOf<T>, T::AccountId) => Vec<(T::Hash, BalanceOf<T>)>;
        /// Keeps track of the votes casted in favor of or against the different proposals.
        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> {
        /// The reveal vote does not match our saved commit.
        RevealCommitMismatch,
        /// We were not able to find a commit for the given reveal vote.
        NoCommitFound,
        /// The vote we are trying to commit for was already revealed.
        Revealed,
        /// The caller's balance is too small to lock the coins they want
        /// to lock.
        NotEnoughBalance,
        /// The vote is being pushed for the wrong phase, either you are
        /// trying to commit too late, either you are trying to reveal too
        /// too early or late.
        Phase,
        /// We cannot close the vote now and have to wait
        TooEarly,
    }
}

decl_event!(
    pub enum Event<T>
    where
        Hash = <T as frame_system::Config>::Hash,
        AccountId = <T as frame_system::Config>::AccountId,
    {
        /// A commit vote was registered. \[voter, proposal, commit\]
        VoteCommited(AccountId, Hash, Hash),
        /// A commited vote was revealed. \[voter, proposal, commit\]
        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;

    /// Register the proposal in our storage. Does not make any attempt at preventing duplicates
    /// as we assume this is handled by the calling pallet.
    fn initiate(proposal: Self::ProposalId, parameters: Self::Parameters) -> DispatchResult {
        Proposals::<T>::try_mutate_exists(proposal, |maybe_existing_state| -> DispatchResult {
            // no duplicates, we can create a new state
            *maybe_existing_state = Some(ProposalState {
                parameters,
                created_on: Self::now(),

                revealed_against: Zero::zero(),
                revealed_favorable: Zero::zero(),
            });

            Ok(())
        })?;

        Ok(())
    }

    /// Simply unlock voters' tokens and clear the storage.
    fn veto(proposal: Self::ProposalId) -> DispatchResult {
        Self::finalize_proposal(proposal, Self::proposals(proposal))
    }

    /// Handle votes depending on the different phases. We prevent revealing votes before the
    /// reveal phase and idem for commit votes.
    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
                    );
                    // We want to prevent votes for user with less coins than they'd like to lock.
                    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(())
    }

    /// Checks the wether the proposal is passing and then clear the storage plus unlock
    /// the voters' coins.
    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> {
    /// Just a helper function to return the current block number. Simply sexier
    /// than calling the actual `frame_system::Module::<T>::block_number()` function.
    fn now() -> T::BlockNumber {
        frame_system::Module::<T>::block_number()
    }

    /// Register a new lock of `amount` for `currency` linked to `proposal` for `who`.
    /// If a lock with similar parameters but a different `amount` exists we will
    /// replace it and lock or unlock the difference.
    fn lock(
        proposal: T::Hash,
        currency: CurrencyIdOf<T>,
        who: &T::AccountId,
        amount: BalanceOf<T>,
    ) -> DispatchResult {
        Locks::<T>::try_mutate((currency, who), |locks| -> DispatchResult {
            // because we use iterators we have to first create a vec for
            // a use with chain() later on
            let locks_addition = vec![(proposal, amount)];

            // Filter and remove any duplicate votes
            *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(())
    }

    /// Frees the locked coins of `who` for `currency` related to `proposal`.
    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(())
    }

    /// Read the locks in `locks` and lock the maximum amount of coins for `who` and `currency`.
    /// Useful to handle cases where people vote on multiple proposals for the same coins.
    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(())
    }

    /// Clear all the storage related to a proposal.
    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(())
    }
}