referrerpolicy=no-referrer-when-downgrade

pallet_society/
lib.rs

1// This file is part of Substrate.
2
3// Copyright (C) Parity Technologies (UK) Ltd.
4// SPDX-License-Identifier: Apache-2.0
5
6// Licensed under the Apache License, Version 2.0 (the "License");
7// you may not use this file except in compliance with the License.
8// You may obtain a copy of the License at
9//
10// 	http://www.apache.org/licenses/LICENSE-2.0
11//
12// Unless required by applicable law or agreed to in writing, software
13// distributed under the License is distributed on an "AS IS" BASIS,
14// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15// See the License for the specific language governing permissions and
16// limitations under the License.
17
18//! # Society Pallet
19//!
20//! - [`Config`]
21//! - [`Call`]
22//!
23//! ## Overview
24//!
25//! The Society pallet is an economic game which incentivizes users to participate
26//! and maintain a membership society.
27//!
28//! ### User Types
29//!
30//! At any point, a user in the society can be one of a:
31//! * Bidder - A user who has submitted intention of joining the society.
32//! * Candidate - A user who will be voted on to join the society.
33//! * Member - A user who is a member of the society.
34//! * Suspended Member - A member of the society who has accumulated too many strikes
35//! or failed their membership challenge.
36//!
37//! Of the non-suspended members, there is always a:
38//! * Head - A member who is exempt from suspension.
39//! * Defender - A member whose membership is under question and voted on again.
40//!
41//! Of the non-suspended members of the society, a random set of them are chosen as
42//! "skeptics". The mechanics of skeptics is explained in the
43//! [member phase](#member-phase) below.
44//!
45//! ### Mechanics
46//!
47//! #### Rewards
48//!
49//! Members are incentivized to participate in the society through rewards paid
50//! by the Society treasury. These payments have a maturity period that the user
51//! must wait before they are able to access the funds.
52//!
53//! #### Punishments
54//!
55//! Members can be punished by slashing the reward payouts that have not been
56//! collected. Additionally, members can accumulate "strikes", and when they
57//! reach a max strike limit, they become suspended.
58//!
59//! #### Skeptics
60//!
61//! During the voting period, a random set of members are selected as "skeptics".
62//! These skeptics are expected to vote on the current candidates. If they do not vote,
63//! their skeptic status is treated as a rejection vote, the member is deemed
64//! "lazy", and are given a strike per missing vote.
65//!
66//! #### Membership Challenges
67//!
68//! Every challenge rotation period, an existing member will be randomly selected
69//! to defend their membership into society. Then, other members can vote whether
70//! this defender should stay in society. A simple majority wins vote will determine
71//! the outcome of the user. Ties are treated as a failure of the challenge, but
72//! assuming no one else votes, the defender always get a free vote on their
73//! own challenge keeping them in the society. The Head member is exempt from the
74//! negative outcome of a membership challenge.
75//!
76//! #### Society Treasury
77//!
78//! The membership society is independently funded by a treasury managed by this
79//! pallet. Some subset of this treasury is placed in a Society Pot, which is used
80//! to determine the number of accepted bids.
81//!
82//! #### Rate of Growth
83//!
84//! The membership society can grow at a rate of 10 accepted candidates per rotation period up
85//! to the max membership threshold. Once this threshold is met, candidate selections
86//! are stalled until there is space for new members to join. This can be resolved by
87//! voting out existing members through the random challenges or by using governance
88//! to increase the maximum membership count.
89//!
90//! ### User Life Cycle
91//!
92//! A user can go through the following phases:
93//!
94//! ```ignore
95//!           +------->  User  <----------+
96//!           |           +               |
97//!           |           |               |
98//! +----------------------------------------------+
99//! |         |           |               |        |
100//! |         |           v               |        |
101//! |         |        Bidder <-----------+        |
102//! |         |           +               |        |
103//! |         |           |               +        |
104//! |         |           v            Suspended   |
105//! |         |       Candidate +----> Candidate   |
106//! |         |           +               +        |
107//! |         |           |               |        |
108//! |         +           |               |        |
109//! |   Suspended +------>|               |        |
110//! |      Member         |               |        |
111//! |         ^           |               |        |
112//! |         |           v               |        |
113//! |         +-------+ Member <----------+        |
114//! |                                              |
115//! |                                              |
116//! +------------------Society---------------------+
117//! ```
118//!
119//! #### Initialization
120//!
121//! The society is initialized with a single member who is automatically chosen as the Head.
122//!
123//! #### Bid Phase
124//!
125//! New users must have a bid to join the society.
126//!
127//! A user can make a bid by reserving a deposit. Alternatively, an already existing member
128//! can create a bid on a user's behalf by "vouching" for them.
129//!
130//! A bid includes reward information that the user would like to receive for joining
131//! the society. A vouching bid can additionally request some portion of that reward as a tip
132//! to the voucher for vouching for the prospective candidate.
133//!
134//! Every rotation period, Bids are ordered by reward amount, and the pallet
135//! selects as many bids the Society Pot can support for that period.
136//!
137//! These selected bids become candidates and move on to the Candidate phase.
138//! Bids that were not selected stay in the bidder pool until they are selected or
139//! a user chooses to "unbid".
140//!
141//! #### Candidate Phase
142//!
143//! Once a bidder becomes a candidate, members vote whether to approve or reject
144//! that candidate into society. This voting process also happens during a rotation period.
145//!
146//! The approval and rejection criteria for candidates are not set on chain,
147//! and may change for different societies.
148//!
149//! At the end of the rotation period, we collect the votes for a candidate
150//! and randomly select a vote as the final outcome.
151//!
152//! ```ignore
153//!  [ a-accept, r-reject, s-skeptic ]
154//! +----------------------------------+
155//! |                                  |
156//! |  Member   |0|1|2|3|4|5|6|7|8|9|  |
157//! |  -----------------------------   |
158//! |  Vote     |a|a|a|r|s|r|a|a|s|a|  |
159//! |  -----------------------------   |
160//! |  Selected | | | |x| | | | | | |  |
161//! |                                  |
162//! +----------------------------------+
163//!
164//! Result: Rejected
165//! ```
166//!
167//! Each member that voted opposite to this randomly selected vote is punished by
168//! slashing their unclaimed payouts and increasing the number of strikes they have.
169//!
170//! These slashed funds are given to a random user who voted the same as the
171//! selected vote as a reward for participating in the vote.
172//!
173//! If the candidate wins the vote, they receive their bid reward as a future payout.
174//! If the bid was placed by a voucher, they will receive their portion of the reward,
175//! before the rest is paid to the winning candidate.
176//!
177//! One winning candidate is selected as the Head of the members. This is randomly
178//! chosen, weighted by the number of approvals the winning candidates accumulated.
179//!
180//! If the candidate loses the vote, they are suspended and it is up to the Suspension
181//! Judgement origin to determine if the candidate should go through the bidding process
182//! again, should be accepted into the membership society, or rejected and their deposit
183//! slashed.
184//!
185//! #### Member Phase
186//!
187//! Once a candidate becomes a member, their role is to participate in society.
188//!
189//! Regular participation involves voting on candidates who want to join the membership
190//! society, and by voting in the right way, a member will accumulate future payouts.
191//! When a payout matures, members are able to claim those payouts.
192//!
193//! Members can also vouch for users to join the society, and request a "tip" from
194//! the fees the new member would collect by joining the society. This vouching
195//! process is useful in situations where a user may not have enough balance to
196//! satisfy the bid deposit. A member can only vouch one user at a time.
197//!
198//! During rotation periods, a random group of members are selected as "skeptics".
199//! These skeptics are expected to vote on the current candidates. If they do not vote,
200//! their skeptic status is treated as a rejection vote, the member is deemed
201//! "lazy", and are given a strike per missing vote.
202//!
203//! There is a challenge period in parallel to the rotation period. During a challenge period,
204//! a random member is selected to defend their membership to the society. Other members
205//! make a traditional majority-wins vote to determine if the member should stay in the society.
206//! Ties are treated as a failure of the challenge.
207//!
208//! If a member accumulates too many strikes or fails their membership challenge,
209//! they will become suspended. While a member is suspended, they are unable to
210//! claim matured payouts. It is up to the Suspension Judgement origin to determine
211//! if the member should re-enter society or be removed from society with all their
212//! future payouts slashed.
213//!
214//! ## Interface
215//!
216//! ### Dispatchable Functions
217//!
218//! #### For General Users
219//!
220//! * `bid` - A user can make a bid to join the membership society by reserving a deposit.
221//! * `unbid` - A user can withdraw their bid for entry, the deposit is returned.
222//!
223//! #### For Members
224//!
225//! * `vouch` - A member can place a bid on behalf of a user to join the membership society.
226//! * `unvouch` - A member can revoke their vouch for a user.
227//! * `vote` - A member can vote to approve or reject a candidate's request to join the society.
228//! * `defender_vote` - A member can vote to approve or reject a defender's continued membership
229//! to the society.
230//! * `payout` - A member can claim their first matured payment.
231//! * `unfound` - Allow the founder to unfound the society when they are the only member.
232//!
233//! #### For Super Users
234//!
235//! * `found` - The founder origin can initiate this society. Useful for bootstrapping the Society
236//! pallet on an already running chain.
237//! * `judge_suspended_member` - The suspension judgement origin is able to make
238//! judgement on a suspended member.
239//! * `judge_suspended_candidate` - The suspension judgement origin is able to
240//! make judgement on a suspended candidate.
241//! * `set_max_membership` - The ROOT origin can update the maximum member count for the society.
242//! The max membership count must be greater than 1.
243
244// Ensure we're `no_std` when compiling for Wasm.
245#![cfg_attr(not(feature = "std"), no_std)]
246
247#[cfg(test)]
248mod mock;
249
250#[cfg(test)]
251mod tests;
252
253#[cfg(feature = "runtime-benchmarks")]
254mod benchmarking;
255
256pub mod weights;
257
258pub mod migrations;
259
260extern crate alloc;
261
262use alloc::vec::Vec;
263use frame_support::{
264	impl_ensure_origin_with_arg_ignoring_arg,
265	pallet_prelude::*,
266	storage::KeyLenOf,
267	traits::{
268		BalanceStatus, Currency, EnsureOrigin, EnsureOriginWithArg,
269		ExistenceRequirement::AllowDeath, Imbalance, OnUnbalanced, Randomness, ReservableCurrency,
270		StorageVersion,
271	},
272	PalletId,
273};
274use frame_system::pallet_prelude::{
275	ensure_signed, BlockNumberFor as SystemBlockNumberFor, OriginFor,
276};
277use rand_chacha::{
278	rand_core::{RngCore, SeedableRng},
279	ChaChaRng,
280};
281use scale_info::TypeInfo;
282use sp_runtime::{
283	traits::{
284		AccountIdConversion, CheckedAdd, CheckedSub, Hash, Saturating, StaticLookup,
285		TrailingZeroInput, Zero,
286	},
287	ArithmeticError::Overflow,
288	Debug, Percent,
289};
290
291pub use weights::WeightInfo;
292
293pub use pallet::*;
294use sp_runtime::traits::BlockNumberProvider;
295
296pub type BlockNumberFor<T, I> =
297	<<T as Config<I>>::BlockNumberProvider as BlockNumberProvider>::BlockNumber;
298
299pub type BalanceOf<T, I> =
300	<<T as Config<I>>::Currency as Currency<<T as frame_system::Config>::AccountId>>::Balance;
301pub type NegativeImbalanceOf<T, I> = <<T as Config<I>>::Currency as Currency<
302	<T as frame_system::Config>::AccountId,
303>>::NegativeImbalance;
304pub type AccountIdLookupOf<T> = <<T as frame_system::Config>::Lookup as StaticLookup>::Source;
305
306#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
307pub struct Vote {
308	pub approve: bool,
309	pub weight: u32,
310}
311
312/// A judgement by the suspension judgement origin on a suspended candidate.
313#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
314pub enum Judgement {
315	/// The suspension judgement origin takes no direct judgment
316	/// and places the candidate back into the bid pool.
317	Rebid,
318	/// The suspension judgement origin has rejected the candidate's application.
319	Reject,
320	/// The suspension judgement origin approves of the candidate's application.
321	Approve,
322}
323
324/// Details of a payout given as a per-block linear "trickle".
325#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, Default, TypeInfo, MaxEncodedLen)]
326pub struct Payout<Balance, BlockNumber> {
327	/// Total value of the payout.
328	pub value: Balance,
329	/// Block number at which the payout begins.
330	pub begin: BlockNumber,
331	/// Total number of blocks over which the payout is spread.
332	pub duration: BlockNumber,
333	/// Total value paid out so far.
334	pub paid: Balance,
335}
336
337/// Status of a vouching member.
338#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
339pub enum VouchingStatus {
340	/// Member is currently vouching for a user.
341	Vouching,
342	/// Member is banned from vouching for other members.
343	Banned,
344}
345
346/// Number of strikes that a member has against them.
347pub type StrikeCount = u32;
348
349/// A bid for entry into society.
350#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
351pub struct Bid<AccountId, Balance> {
352	/// The bidder/candidate trying to enter society
353	pub who: AccountId,
354	/// The kind of bid placed for this bidder/candidate. See `BidKind`.
355	pub kind: BidKind<AccountId, Balance>,
356	/// The reward that the bidder has requested for successfully joining the society.
357	pub value: Balance,
358}
359
360/// The index of a round of candidates.
361pub type RoundIndex = u32;
362
363/// The rank of a member.
364pub type Rank = u32;
365
366/// The number of votes.
367pub type VoteCount = u32;
368
369/// Tally of votes.
370#[derive(Default, Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
371pub struct Tally {
372	/// The approval votes.
373	pub approvals: VoteCount,
374	/// The rejection votes.
375	pub rejections: VoteCount,
376}
377
378impl Tally {
379	fn more_approvals(&self) -> bool {
380		self.approvals > self.rejections
381	}
382
383	fn more_rejections(&self) -> bool {
384		self.rejections > self.approvals
385	}
386
387	fn clear_approval(&self) -> bool {
388		self.approvals >= (2 * self.rejections).max(1)
389	}
390
391	fn clear_rejection(&self) -> bool {
392		self.rejections >= (2 * self.approvals).max(1)
393	}
394}
395
396/// A bid for entry into society.
397#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
398pub struct Candidacy<AccountId, Balance> {
399	/// The index of the round where the candidacy began.
400	pub round: RoundIndex,
401	/// The kind of bid placed for this bidder/candidate. See `BidKind`.
402	pub kind: BidKind<AccountId, Balance>,
403	/// The reward that the bidder has requested for successfully joining the society.
404	pub bid: Balance,
405	/// The tally of votes so far.
406	pub tally: Tally,
407	/// True if the skeptic was already punished for note voting.
408	pub skeptic_struck: bool,
409}
410
411/// A vote by a member on a candidate application.
412#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
413pub enum BidKind<AccountId, Balance> {
414	/// The given deposit was paid for this bid.
415	Deposit(Balance),
416	/// A member vouched for this bid. The account should be reinstated into `Members` once the
417	/// bid is successful (or if it is rescinded prior to launch).
418	Vouch(AccountId, Balance),
419}
420
421impl<AccountId: PartialEq, Balance> BidKind<AccountId, Balance> {
422	fn is_vouch(&self, v: &AccountId) -> bool {
423		matches!(self, BidKind::Vouch(ref a, _) if a == v)
424	}
425}
426
427pub type PayoutsFor<T, I> =
428	BoundedVec<(BlockNumberFor<T, I>, BalanceOf<T, I>), <T as Config<I>>::MaxPayouts>;
429
430/// Information concerning a member.
431#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
432pub struct MemberRecord {
433	pub rank: Rank,
434	pub strikes: StrikeCount,
435	pub vouching: Option<VouchingStatus>,
436	pub index: u32,
437}
438
439/// Information concerning a member.
440#[derive(Encode, Decode, Clone, PartialEq, Eq, Debug, TypeInfo, Default, MaxEncodedLen)]
441pub struct PayoutRecord<Balance, PayoutsVec> {
442	pub paid: Balance,
443	pub payouts: PayoutsVec,
444}
445
446pub type PayoutRecordFor<T, I> = PayoutRecord<
447	BalanceOf<T, I>,
448	BoundedVec<(BlockNumberFor<T, I>, BalanceOf<T, I>), <T as Config<I>>::MaxPayouts>,
449>;
450
451/// Record for an individual new member who was elevated from a candidate recently.
452#[derive(Encode, Decode, Copy, Clone, PartialEq, Eq, Debug, TypeInfo, MaxEncodedLen)]
453pub struct IntakeRecord<AccountId, Balance> {
454	pub who: AccountId,
455	pub bid: Balance,
456	pub round: RoundIndex,
457}
458
459pub type IntakeRecordFor<T, I> =
460	IntakeRecord<<T as frame_system::Config>::AccountId, BalanceOf<T, I>>;
461
462#[derive(
463	Encode,
464	Decode,
465	DecodeWithMemTracking,
466	Copy,
467	Clone,
468	PartialEq,
469	Eq,
470	Debug,
471	TypeInfo,
472	MaxEncodedLen,
473)]
474pub struct GroupParams<Balance> {
475	pub max_members: u32,
476	pub max_intake: u32,
477	pub max_strikes: u32,
478	pub candidate_deposit: Balance,
479}
480
481pub type GroupParamsFor<T, I> = GroupParams<BalanceOf<T, I>>;
482
483pub(crate) const STORAGE_VERSION: StorageVersion = StorageVersion::new(2);
484
485#[frame_support::pallet]
486pub mod pallet {
487	use super::*;
488
489	#[pallet::pallet]
490	#[pallet::storage_version(STORAGE_VERSION)]
491	pub struct Pallet<T, I = ()>(_);
492
493	#[pallet::config]
494	pub trait Config<I: 'static = ()>: frame_system::Config {
495		/// The overarching event type.
496		#[allow(deprecated)]
497		type RuntimeEvent: From<Event<Self, I>>
498			+ IsType<<Self as frame_system::Config>::RuntimeEvent>;
499
500		/// The societies's pallet id
501		#[pallet::constant]
502		type PalletId: Get<PalletId>;
503
504		/// The currency type used for bidding.
505		type Currency: ReservableCurrency<Self::AccountId>;
506
507		/// Something that provides randomness in the runtime.
508		type Randomness: Randomness<Self::Hash, BlockNumberFor<Self, I>>;
509
510		/// The maximum number of strikes before a member gets funds slashed.
511		#[pallet::constant]
512		type GraceStrikes: Get<u32>;
513
514		/// The amount of incentive paid within each period. Doesn't include VoterTip.
515		#[pallet::constant]
516		type PeriodSpend: Get<BalanceOf<Self, I>>;
517
518		/// The number of [Config::BlockNumberProvider] blocks on which new candidates should be
519		/// voted on. Together with
520		/// `ClaimPeriod`, this sums to the number of blocks between candidate intake periods.
521		#[pallet::constant]
522		type VotingPeriod: Get<BlockNumberFor<Self, I>>;
523
524		/// The number of [Config::BlockNumberProvider] blocks on which new candidates can claim
525		/// their membership and be the named head.
526		#[pallet::constant]
527		type ClaimPeriod: Get<BlockNumberFor<Self, I>>;
528
529		/// The maximum duration of the payout lock.
530		#[pallet::constant]
531		type MaxLockDuration: Get<BlockNumberFor<Self, I>>;
532
533		/// The origin that is allowed to call `found`.
534		type FounderSetOrigin: EnsureOrigin<Self::RuntimeOrigin>;
535
536		/// The number of [Config::BlockNumberProvider] blocks between membership challenges.
537		#[pallet::constant]
538		type ChallengePeriod: Get<BlockNumberFor<Self, I>>;
539
540		/// The maximum number of payouts a member may have waiting unclaimed.
541		#[pallet::constant]
542		type MaxPayouts: Get<u32>;
543
544		/// The maximum number of bids at once.
545		#[pallet::constant]
546		type MaxBids: Get<u32>;
547
548		/// Weight information for extrinsics in this pallet.
549		type WeightInfo: WeightInfo;
550		/// Provider for the block number. Normally this is the `frame_system` pallet.
551		type BlockNumberProvider: BlockNumberProvider;
552	}
553
554	#[pallet::error]
555	pub enum Error<T, I = ()> {
556		/// User is not a member.
557		NotMember,
558		/// User is already a member.
559		AlreadyMember,
560		/// User is suspended.
561		Suspended,
562		/// User is not suspended.
563		NotSuspended,
564		/// Nothing to payout.
565		NoPayout,
566		/// Society already founded.
567		AlreadyFounded,
568		/// Not enough in pot to accept candidate.
569		InsufficientPot,
570		/// Member is already vouching or banned from vouching again.
571		AlreadyVouching,
572		/// Member is not vouching.
573		NotVouchingOnBidder,
574		/// Cannot remove the head of the chain.
575		Head,
576		/// Cannot remove the founder.
577		Founder,
578		/// User has already made a bid.
579		AlreadyBid,
580		/// User is already a candidate.
581		AlreadyCandidate,
582		/// User is not a candidate.
583		NotCandidate,
584		/// Too many members in the society.
585		MaxMembers,
586		/// The caller is not the founder.
587		NotFounder,
588		/// The caller is not the head.
589		NotHead,
590		/// The membership cannot be claimed as the candidate was not clearly approved.
591		NotApproved,
592		/// The candidate cannot be kicked as the candidate was not clearly rejected.
593		NotRejected,
594		/// The candidacy cannot be dropped as the candidate was clearly approved.
595		Approved,
596		/// The candidacy cannot be bestowed as the candidate was clearly rejected.
597		Rejected,
598		/// The candidacy cannot be concluded as the voting is still in progress.
599		InProgress,
600		/// The candidacy cannot be pruned until a full additional intake period has passed.
601		TooEarly,
602		/// The skeptic already voted.
603		Voted,
604		/// The skeptic need not vote on candidates from expired rounds.
605		Expired,
606		/// User is not a bidder.
607		NotBidder,
608		/// There is no defender currently.
609		NoDefender,
610		/// Group doesn't exist.
611		NotGroup,
612		/// The member is already elevated to this rank.
613		AlreadyElevated,
614		/// The skeptic has already been punished for this offence.
615		AlreadyPunished,
616		/// Funds are insufficient to pay off society debts.
617		InsufficientFunds,
618		/// The candidate/defender has no stale votes to remove.
619		NoVotes,
620		/// There is no deposit associated with a bid.
621		NoDeposit,
622	}
623
624	#[pallet::event]
625	#[pallet::generate_deposit(pub(super) fn deposit_event)]
626	pub enum Event<T: Config<I>, I: 'static = ()> {
627		/// The society is founded by the given identity.
628		Founded { founder: T::AccountId },
629		/// A membership bid just happened. The given account is the candidate's ID and their offer
630		/// is the second.
631		Bid { candidate_id: T::AccountId, offer: BalanceOf<T, I> },
632		/// A membership bid just happened by vouching. The given account is the candidate's ID and
633		/// their offer is the second. The vouching party is the third.
634		Vouch { candidate_id: T::AccountId, offer: BalanceOf<T, I>, vouching: T::AccountId },
635		/// A candidate was dropped (due to an excess of bids in the system).
636		AutoUnbid { candidate: T::AccountId },
637		/// A candidate was dropped (by their request).
638		Unbid { candidate: T::AccountId },
639		/// A candidate was dropped (by request of who vouched for them).
640		Unvouch { candidate: T::AccountId },
641		/// A group of candidates have been inducted. The batch's primary is the first value, the
642		/// batch in full is the second.
643		Inducted { primary: T::AccountId, candidates: Vec<T::AccountId> },
644		/// A suspended member has been judged.
645		SuspendedMemberJudgement { who: T::AccountId, judged: bool },
646		/// A candidate has been suspended
647		CandidateSuspended { candidate: T::AccountId },
648		/// A member has been suspended
649		MemberSuspended { member: T::AccountId },
650		/// A member has been challenged
651		Challenged { member: T::AccountId },
652		/// A vote has been placed
653		Vote { candidate: T::AccountId, voter: T::AccountId, vote: bool },
654		/// A vote has been placed for a defending member
655		DefenderVote { voter: T::AccountId, vote: bool },
656		/// A new set of \[params\] has been set for the group.
657		NewParams { params: GroupParamsFor<T, I> },
658		/// Society is unfounded.
659		Unfounded { founder: T::AccountId },
660		/// Some funds were deposited into the society account.
661		Deposit { value: BalanceOf<T, I> },
662		/// A \[member\] got elevated to \[rank\].
663		Elevated { member: T::AccountId, rank: Rank },
664		/// A deposit was poked / adjusted.
665		DepositPoked {
666			who: T::AccountId,
667			old_deposit: BalanceOf<T, I>,
668			new_deposit: BalanceOf<T, I>,
669		},
670	}
671
672	/// Old name generated by `decl_event`.
673	#[deprecated(note = "use `Event` instead")]
674	pub type RawEvent<T, I = ()> = Event<T, I>;
675
676	/// The max number of members for the society at one time.
677	#[pallet::storage]
678	pub type Parameters<T: Config<I>, I: 'static = ()> =
679		StorageValue<_, GroupParamsFor<T, I>, OptionQuery>;
680
681	/// Amount of our account balance that is specifically for the next round's bid(s).
682	#[pallet::storage]
683	pub type Pot<T: Config<I>, I: 'static = ()> = StorageValue<_, BalanceOf<T, I>, ValueQuery>;
684
685	/// The first member.
686	#[pallet::storage]
687	pub type Founder<T: Config<I>, I: 'static = ()> = StorageValue<_, T::AccountId>;
688
689	/// The most primary from the most recently approved rank 0 members in the society.
690	#[pallet::storage]
691	pub type Head<T: Config<I>, I: 'static = ()> = StorageValue<_, T::AccountId>;
692
693	/// A hash of the rules of this society concerning membership. Can only be set once and
694	/// only by the founder.
695	#[pallet::storage]
696	pub type Rules<T: Config<I>, I: 'static = ()> = StorageValue<_, T::Hash>;
697
698	/// The current members and their rank. Doesn't include `SuspendedMembers`.
699	#[pallet::storage]
700	pub type Members<T: Config<I>, I: 'static = ()> =
701		StorageMap<_, Twox64Concat, T::AccountId, MemberRecord, OptionQuery>;
702
703	/// Information regarding rank-0 payouts, past and future.
704	#[pallet::storage]
705	pub type Payouts<T: Config<I>, I: 'static = ()> =
706		StorageMap<_, Twox64Concat, T::AccountId, PayoutRecordFor<T, I>, ValueQuery>;
707
708	/// The number of items in `Members` currently. (Doesn't include `SuspendedMembers`.)
709	#[pallet::storage]
710	pub type MemberCount<T: Config<I>, I: 'static = ()> = StorageValue<_, u32, ValueQuery>;
711
712	/// The current items in `Members` keyed by their unique index. Keys are densely populated
713	/// `0..MemberCount` (does not include `MemberCount`).
714	#[pallet::storage]
715	pub type MemberByIndex<T: Config<I>, I: 'static = ()> =
716		StorageMap<_, Twox64Concat, u32, T::AccountId, OptionQuery>;
717
718	/// The set of suspended members, with their old membership record.
719	#[pallet::storage]
720	pub type SuspendedMembers<T: Config<I>, I: 'static = ()> =
721		StorageMap<_, Twox64Concat, T::AccountId, MemberRecord, OptionQuery>;
722
723	/// The number of rounds which have passed.
724	#[pallet::storage]
725	pub type RoundCount<T: Config<I>, I: 'static = ()> = StorageValue<_, RoundIndex, ValueQuery>;
726
727	/// The current bids, stored ordered by the value of the bid.
728	#[pallet::storage]
729	pub type Bids<T: Config<I>, I: 'static = ()> =
730		StorageValue<_, BoundedVec<Bid<T::AccountId, BalanceOf<T, I>>, T::MaxBids>, ValueQuery>;
731
732	#[pallet::storage]
733	pub type Candidates<T: Config<I>, I: 'static = ()> = StorageMap<
734		_,
735		Blake2_128Concat,
736		T::AccountId,
737		Candidacy<T::AccountId, BalanceOf<T, I>>,
738		OptionQuery,
739	>;
740
741	/// The current skeptic.
742	#[pallet::storage]
743	pub type Skeptic<T: Config<I>, I: 'static = ()> = StorageValue<_, T::AccountId, OptionQuery>;
744
745	/// Double map from Candidate -> Voter -> (Maybe) Vote.
746	#[pallet::storage]
747	pub type Votes<T: Config<I>, I: 'static = ()> = StorageDoubleMap<
748		_,
749		Twox64Concat,
750		T::AccountId,
751		Twox64Concat,
752		T::AccountId,
753		Vote,
754		OptionQuery,
755	>;
756
757	/// Clear-cursor for Vote, map from Candidate -> (Maybe) Cursor.
758	#[pallet::storage]
759	pub type VoteClearCursor<T: Config<I>, I: 'static = ()> =
760		StorageMap<_, Twox64Concat, T::AccountId, BoundedVec<u8, KeyLenOf<Votes<T, I>>>>;
761
762	/// At the end of the claim period, this contains the most recently approved members (along with
763	/// their bid and round ID) who is from the most recent round with the lowest bid. They will
764	/// become the new `Head`.
765	#[pallet::storage]
766	pub type NextHead<T: Config<I>, I: 'static = ()> =
767		StorageValue<_, IntakeRecordFor<T, I>, OptionQuery>;
768
769	/// The number of challenge rounds there have been. Used to identify stale DefenderVotes.
770	#[pallet::storage]
771	pub type ChallengeRoundCount<T: Config<I>, I: 'static = ()> =
772		StorageValue<_, RoundIndex, ValueQuery>;
773
774	/// The defending member currently being challenged, along with a running tally of votes.
775	#[pallet::storage]
776	pub type Defending<T: Config<I>, I: 'static = ()> =
777		StorageValue<_, (T::AccountId, T::AccountId, Tally)>;
778
779	/// Votes for the defender, keyed by challenge round.
780	#[pallet::storage]
781	pub type DefenderVotes<T: Config<I>, I: 'static = ()> =
782		StorageDoubleMap<_, Twox64Concat, RoundIndex, Twox64Concat, T::AccountId, Vote>;
783
784	/// Next intake rotation scheduled with [Config::BlockNumberProvider].
785	#[pallet::storage]
786	pub type NextIntakeAt<T: Config<I>, I: 'static = ()> = StorageValue<_, BlockNumberFor<T, I>>;
787
788	/// Next challenge rotation scheduled with [Config::BlockNumberProvider].
789	#[pallet::storage]
790	pub type NextChallengeAt<T: Config<I>, I: 'static = ()> = StorageValue<_, BlockNumberFor<T, I>>;
791
792	#[pallet::hooks]
793	impl<T: Config<I>, I: 'static> Hooks<SystemBlockNumberFor<T>> for Pallet<T, I> {
794		fn on_initialize(_n: SystemBlockNumberFor<T>) -> Weight {
795			let mut weight = Weight::zero();
796			let weights = T::BlockWeights::get();
797			let now = T::BlockNumberProvider::current_block_number();
798
799			let phrase = b"society_rotation";
800			// we'll need a random seed here.
801			// TODO: deal with randomness freshness
802			// https://github.com/paritytech/substrate/issues/8312
803			let (seed, _) = T::Randomness::random(phrase);
804			// seed needs to be guaranteed to be 32 bytes.
805			let seed = <[u8; 32]>::decode(&mut TrailingZeroInput::new(seed.as_ref()))
806				.expect("input is padded with zeroes; qed");
807			let mut rng = ChaChaRng::from_seed(seed);
808
809			// Run a candidate/membership rotation
810			let is_intake_moment = match Self::period() {
811				Period::Intake { .. } => true,
812				_ => false,
813			};
814			if is_intake_moment {
815				Self::rotate_intake(&mut rng);
816				weight.saturating_accrue(weights.max_block / 20);
817				Self::set_next_intake_at();
818			}
819
820			// Run a challenge rotation
821			if now >= Self::next_challenge_at() {
822				Self::rotate_challenge(&mut rng);
823				weight.saturating_accrue(weights.max_block / 20);
824				Self::set_next_challenge_at();
825			}
826
827			weight
828		}
829	}
830
831	#[pallet::genesis_config]
832	#[derive(frame_support::DefaultNoBound)]
833	pub struct GenesisConfig<T: Config<I>, I: 'static = ()> {
834		pub pot: BalanceOf<T, I>,
835	}
836
837	#[pallet::genesis_build]
838	impl<T: Config<I>, I: 'static> BuildGenesisConfig for GenesisConfig<T, I> {
839		fn build(&self) {
840			Pot::<T, I>::put(self.pot);
841		}
842	}
843
844	#[pallet::call]
845	impl<T: Config<I>, I: 'static> Pallet<T, I> {
846		/// A user outside of the society can make a bid for entry.
847		///
848		/// Payment: The group's Candidate Deposit will be reserved for making a bid. It is returned
849		/// when the bid becomes a member, or if the bid calls `unbid`.
850		///
851		/// The dispatch origin for this call must be _Signed_.
852		///
853		/// Parameters:
854		/// - `value`: A one time payment the bid would like to receive when joining the society.
855		#[pallet::call_index(0)]
856		#[pallet::weight(T::WeightInfo::bid())]
857		pub fn bid(origin: OriginFor<T>, value: BalanceOf<T, I>) -> DispatchResult {
858			let who = ensure_signed(origin)?;
859
860			let mut bids = Bids::<T, I>::get();
861			ensure!(!Self::has_bid(&bids, &who), Error::<T, I>::AlreadyBid);
862			ensure!(!Candidates::<T, I>::contains_key(&who), Error::<T, I>::AlreadyCandidate);
863			ensure!(!Members::<T, I>::contains_key(&who), Error::<T, I>::AlreadyMember);
864			ensure!(!SuspendedMembers::<T, I>::contains_key(&who), Error::<T, I>::Suspended);
865
866			let params = Parameters::<T, I>::get().ok_or(Error::<T, I>::NotGroup)?;
867			let deposit = params.candidate_deposit;
868			// NOTE: Reserve must happen before `insert_bid` since that could end up unreserving.
869			T::Currency::reserve(&who, deposit)?;
870			Self::insert_bid(&mut bids, &who, value, BidKind::Deposit(deposit));
871
872			Bids::<T, I>::put(bids);
873			Self::deposit_event(Event::<T, I>::Bid { candidate_id: who, offer: value });
874			Ok(())
875		}
876
877		/// A bidder can remove their bid for entry into society.
878		/// By doing so, they will have their candidate deposit returned or
879		/// they will unvouch their voucher.
880		///
881		/// Payment: The bid deposit is unreserved if the user made a bid.
882		///
883		/// The dispatch origin for this call must be _Signed_ and a bidder.
884		#[pallet::call_index(1)]
885		#[pallet::weight(T::WeightInfo::unbid())]
886		pub fn unbid(origin: OriginFor<T>) -> DispatchResult {
887			let who = ensure_signed(origin)?;
888
889			let mut bids = Bids::<T, I>::get();
890			let pos = bids.iter().position(|bid| bid.who == who).ok_or(Error::<T, I>::NotBidder)?;
891			Self::clean_bid(&bids.remove(pos));
892			Bids::<T, I>::put(bids);
893			Self::deposit_event(Event::<T, I>::Unbid { candidate: who });
894			Ok(())
895		}
896
897		/// As a member, vouch for someone to join society by placing a bid on their behalf.
898		///
899		/// There is no deposit required to vouch for a new bid, but a member can only vouch for
900		/// one bid at a time. If the bid becomes a suspended candidate and ultimately rejected by
901		/// the suspension judgement origin, the member will be banned from vouching again.
902		///
903		/// As a vouching member, you can claim a tip if the candidate is accepted. This tip will
904		/// be paid as a portion of the reward the member will receive for joining the society.
905		///
906		/// The dispatch origin for this call must be _Signed_ and a member.
907		///
908		/// Parameters:
909		/// - `who`: The user who you would like to vouch for.
910		/// - `value`: The total reward to be paid between you and the candidate if they become
911		/// a member in the society.
912		/// - `tip`: Your cut of the total `value` payout when the candidate is inducted into
913		/// the society. Tips larger than `value` will be saturated upon payout.
914		#[pallet::call_index(2)]
915		#[pallet::weight(T::WeightInfo::vouch())]
916		pub fn vouch(
917			origin: OriginFor<T>,
918			who: AccountIdLookupOf<T>,
919			value: BalanceOf<T, I>,
920			tip: BalanceOf<T, I>,
921		) -> DispatchResult {
922			let voucher = ensure_signed(origin)?;
923			let who = T::Lookup::lookup(who)?;
924
925			// Get bids and check user is not bidding.
926			let mut bids = Bids::<T, I>::get();
927			ensure!(!Self::has_bid(&bids, &who), Error::<T, I>::AlreadyBid);
928
929			// Check user is not already a candidate, member or suspended member.
930			ensure!(!Candidates::<T, I>::contains_key(&who), Error::<T, I>::AlreadyCandidate);
931			ensure!(!Members::<T, I>::contains_key(&who), Error::<T, I>::AlreadyMember);
932			ensure!(!SuspendedMembers::<T, I>::contains_key(&who), Error::<T, I>::Suspended);
933
934			// Check sender can vouch.
935			let mut record = Members::<T, I>::get(&voucher).ok_or(Error::<T, I>::NotMember)?;
936			ensure!(record.vouching.is_none(), Error::<T, I>::AlreadyVouching);
937
938			// Update voucher record.
939			record.vouching = Some(VouchingStatus::Vouching);
940			// Update bids
941			Self::insert_bid(&mut bids, &who, value, BidKind::Vouch(voucher.clone(), tip));
942
943			// Write new state.
944			Members::<T, I>::insert(&voucher, &record);
945			Bids::<T, I>::put(bids);
946			Self::deposit_event(Event::<T, I>::Vouch {
947				candidate_id: who,
948				offer: value,
949				vouching: voucher,
950			});
951			Ok(())
952		}
953
954		/// As a vouching member, unvouch a bid. This only works while vouched user is
955		/// only a bidder (and not a candidate).
956		///
957		/// The dispatch origin for this call must be _Signed_ and a vouching member.
958		///
959		/// Parameters:
960		/// - `pos`: Position in the `Bids` vector of the bid who should be unvouched.
961		#[pallet::call_index(3)]
962		#[pallet::weight(T::WeightInfo::unvouch())]
963		pub fn unvouch(origin: OriginFor<T>) -> DispatchResult {
964			let voucher = ensure_signed(origin)?;
965
966			let mut bids = Bids::<T, I>::get();
967			let pos = bids
968				.iter()
969				.position(|bid| bid.kind.is_vouch(&voucher))
970				.ok_or(Error::<T, I>::NotVouchingOnBidder)?;
971			let bid = bids.remove(pos);
972			Self::clean_bid(&bid);
973
974			Bids::<T, I>::put(bids);
975			Self::deposit_event(Event::<T, I>::Unvouch { candidate: bid.who });
976			Ok(())
977		}
978
979		/// As a member, vote on a candidate.
980		///
981		/// The dispatch origin for this call must be _Signed_ and a member.
982		///
983		/// Parameters:
984		/// - `candidate`: The candidate that the member would like to bid on.
985		/// - `approve`: A boolean which says if the candidate should be approved (`true`) or
986		///   rejected (`false`).
987		#[pallet::call_index(4)]
988		#[pallet::weight(T::WeightInfo::vote())]
989		pub fn vote(
990			origin: OriginFor<T>,
991			candidate: AccountIdLookupOf<T>,
992			approve: bool,
993		) -> DispatchResultWithPostInfo {
994			let voter = ensure_signed(origin)?;
995			let candidate = T::Lookup::lookup(candidate)?;
996
997			let mut candidacy =
998				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
999			let record = Members::<T, I>::get(&voter).ok_or(Error::<T, I>::NotMember)?;
1000
1001			let first_time = Votes::<T, I>::mutate(&candidate, &voter, |v| {
1002				let first_time = v.is_none();
1003				*v = Some(Self::do_vote(*v, approve, record.rank, &mut candidacy.tally));
1004				first_time
1005			});
1006
1007			Candidates::<T, I>::insert(&candidate, &candidacy);
1008			Self::deposit_event(Event::<T, I>::Vote { candidate, voter, vote: approve });
1009			Ok(if first_time { Pays::No } else { Pays::Yes }.into())
1010		}
1011
1012		/// As a member, vote on the defender.
1013		///
1014		/// The dispatch origin for this call must be _Signed_ and a member.
1015		///
1016		/// Parameters:
1017		/// - `approve`: A boolean which says if the candidate should be
1018		/// approved (`true`) or rejected (`false`).
1019		#[pallet::call_index(5)]
1020		#[pallet::weight(T::WeightInfo::defender_vote())]
1021		pub fn defender_vote(origin: OriginFor<T>, approve: bool) -> DispatchResultWithPostInfo {
1022			let voter = ensure_signed(origin)?;
1023
1024			let mut defending = Defending::<T, I>::get().ok_or(Error::<T, I>::NoDefender)?;
1025			let record = Members::<T, I>::get(&voter).ok_or(Error::<T, I>::NotMember)?;
1026
1027			let round = ChallengeRoundCount::<T, I>::get();
1028			let first_time = DefenderVotes::<T, I>::mutate(round, &voter, |v| {
1029				let first_time = v.is_none();
1030				*v = Some(Self::do_vote(*v, approve, record.rank, &mut defending.2));
1031				first_time
1032			});
1033
1034			Defending::<T, I>::put(defending);
1035			Self::deposit_event(Event::<T, I>::DefenderVote { voter, vote: approve });
1036			Ok(if first_time { Pays::No } else { Pays::Yes }.into())
1037		}
1038
1039		/// Transfer the first matured payout for the sender and remove it from the records.
1040		///
1041		/// NOTE: This extrinsic needs to be called multiple times to claim multiple matured
1042		/// payouts.
1043		///
1044		/// Payment: The member will receive a payment equal to their first matured
1045		/// payout to their free balance.
1046		///
1047		/// The dispatch origin for this call must be _Signed_ and a member with
1048		/// payouts remaining.
1049		#[pallet::call_index(6)]
1050		#[pallet::weight(T::WeightInfo::payout())]
1051		pub fn payout(origin: OriginFor<T>) -> DispatchResult {
1052			let who = ensure_signed(origin)?;
1053			ensure!(
1054				Members::<T, I>::get(&who).ok_or(Error::<T, I>::NotMember)?.rank == 0,
1055				Error::<T, I>::NoPayout
1056			);
1057			let mut record = Payouts::<T, I>::get(&who);
1058			let block_number = T::BlockNumberProvider::current_block_number();
1059			if let Some((when, amount)) = record.payouts.first() {
1060				if when <= &block_number {
1061					record.paid = record.paid.checked_add(amount).ok_or(Overflow)?;
1062					T::Currency::transfer(&Self::payouts(), &who, *amount, AllowDeath)?;
1063					record.payouts.remove(0);
1064					Payouts::<T, I>::insert(&who, record);
1065					return Ok(())
1066				}
1067			}
1068			Err(Error::<T, I>::NoPayout)?
1069		}
1070
1071		/// Repay the payment previously given to the member with the signed origin, remove any
1072		/// pending payments, and elevate them from rank 0 to rank 1.
1073		#[pallet::call_index(7)]
1074		#[pallet::weight(T::WeightInfo::waive_repay())]
1075		pub fn waive_repay(origin: OriginFor<T>, amount: BalanceOf<T, I>) -> DispatchResult {
1076			let who = ensure_signed(origin)?;
1077			let mut record = Members::<T, I>::get(&who).ok_or(Error::<T, I>::NotMember)?;
1078			let mut payout_record = Payouts::<T, I>::get(&who);
1079			ensure!(record.rank == 0, Error::<T, I>::AlreadyElevated);
1080			ensure!(amount >= payout_record.paid, Error::<T, I>::InsufficientFunds);
1081
1082			T::Currency::transfer(&who, &Self::account_id(), payout_record.paid, AllowDeath)?;
1083			payout_record.paid = Zero::zero();
1084			payout_record.payouts.clear();
1085			record.rank = 1;
1086			Members::<T, I>::insert(&who, record);
1087			Payouts::<T, I>::insert(&who, payout_record);
1088			Self::deposit_event(Event::<T, I>::Elevated { member: who, rank: 1 });
1089
1090			Ok(())
1091		}
1092
1093		/// Found the society.
1094		///
1095		/// This is done as a discrete action in order to allow for the
1096		/// pallet to be included into a running chain and can only be done once.
1097		///
1098		/// The dispatch origin for this call must be from the _FounderSetOrigin_.
1099		///
1100		/// Parameters:
1101		/// - `founder` - The first member and head of the newly founded society.
1102		/// - `max_members` - The initial max number of members for the society.
1103		/// - `max_intake` - The maximum number of candidates per intake period.
1104		/// - `max_strikes`: The maximum number of strikes a member may get before they become
1105		///   suspended and may only be reinstated by the founder.
1106		/// - `candidate_deposit`: The deposit required to make a bid for membership of the group.
1107		/// - `rules` - The rules of this society concerning membership.
1108		///
1109		/// Complexity: O(1)
1110		#[pallet::call_index(8)]
1111		#[pallet::weight(T::WeightInfo::found_society())]
1112		pub fn found_society(
1113			origin: OriginFor<T>,
1114			founder: AccountIdLookupOf<T>,
1115			max_members: u32,
1116			max_intake: u32,
1117			max_strikes: u32,
1118			candidate_deposit: BalanceOf<T, I>,
1119			rules: Vec<u8>,
1120		) -> DispatchResult {
1121			T::FounderSetOrigin::ensure_origin(origin)?;
1122			let founder = T::Lookup::lookup(founder)?;
1123			ensure!(!Head::<T, I>::exists(), Error::<T, I>::AlreadyFounded);
1124			ensure!(max_members > 1, Error::<T, I>::MaxMembers);
1125			// This should never fail in the context of this function...
1126			let params = GroupParams { max_members, max_intake, max_strikes, candidate_deposit };
1127			Parameters::<T, I>::put(params);
1128			Self::insert_member(&founder, 1)?;
1129			Head::<T, I>::put(&founder);
1130			Founder::<T, I>::put(&founder);
1131			Rules::<T, I>::put(T::Hashing::hash(&rules));
1132			Self::deposit_event(Event::<T, I>::Founded { founder });
1133			Ok(())
1134		}
1135
1136		/// Dissolve the society and remove all members.
1137		///
1138		/// The dispatch origin for this call must be Signed, and the signing account must be both
1139		/// the `Founder` and the `Head`. This implies that it may only be done when there is one
1140		/// member.
1141		#[pallet::call_index(9)]
1142		#[pallet::weight(T::WeightInfo::dissolve())]
1143		pub fn dissolve(origin: OriginFor<T>) -> DispatchResult {
1144			let founder = ensure_signed(origin)?;
1145			ensure!(Founder::<T, I>::get().as_ref() == Some(&founder), Error::<T, I>::NotFounder);
1146			ensure!(MemberCount::<T, I>::get() == 1, Error::<T, I>::NotHead);
1147
1148			let _ = Members::<T, I>::clear(u32::MAX, None);
1149			MemberCount::<T, I>::kill();
1150			let _ = MemberByIndex::<T, I>::clear(u32::MAX, None);
1151			let _ = SuspendedMembers::<T, I>::clear(u32::MAX, None);
1152			let _ = Payouts::<T, I>::clear(u32::MAX, None);
1153			let _ = Votes::<T, I>::clear(u32::MAX, None);
1154			let _ = VoteClearCursor::<T, I>::clear(u32::MAX, None);
1155			Head::<T, I>::kill();
1156			NextHead::<T, I>::kill();
1157			Founder::<T, I>::kill();
1158			Rules::<T, I>::kill();
1159			Parameters::<T, I>::kill();
1160			Pot::<T, I>::kill();
1161			RoundCount::<T, I>::kill();
1162			Bids::<T, I>::kill();
1163			Skeptic::<T, I>::kill();
1164			ChallengeRoundCount::<T, I>::kill();
1165			Defending::<T, I>::kill();
1166			let _ = DefenderVotes::<T, I>::clear(u32::MAX, None);
1167			let _ = Candidates::<T, I>::clear(u32::MAX, None);
1168			Self::deposit_event(Event::<T, I>::Unfounded { founder });
1169			Ok(())
1170		}
1171
1172		/// Allow suspension judgement origin to make judgement on a suspended member.
1173		///
1174		/// If a suspended member is forgiven, we simply add them back as a member, not affecting
1175		/// any of the existing storage items for that member.
1176		///
1177		/// If a suspended member is rejected, remove all associated storage items, including
1178		/// their payouts, and remove any vouched bids they currently have.
1179		///
1180		/// The dispatch origin for this call must be Signed from the Founder.
1181		///
1182		/// Parameters:
1183		/// - `who` - The suspended member to be judged.
1184		/// - `forgive` - A boolean representing whether the suspension judgement origin forgives
1185		///   (`true`) or rejects (`false`) a suspended member.
1186		#[pallet::call_index(10)]
1187		#[pallet::weight(T::WeightInfo::judge_suspended_member())]
1188		pub fn judge_suspended_member(
1189			origin: OriginFor<T>,
1190			who: AccountIdLookupOf<T>,
1191			forgive: bool,
1192		) -> DispatchResultWithPostInfo {
1193			ensure!(
1194				Some(ensure_signed(origin)?) == Founder::<T, I>::get(),
1195				Error::<T, I>::NotFounder
1196			);
1197			let who = T::Lookup::lookup(who)?;
1198			let record = SuspendedMembers::<T, I>::get(&who).ok_or(Error::<T, I>::NotSuspended)?;
1199			if forgive {
1200				// Try to add member back to society. Can fail with `MaxMembers` limit.
1201				Self::reinstate_member(&who, record.rank)?;
1202			} else {
1203				let payout_record = Payouts::<T, I>::take(&who);
1204				let total = payout_record
1205					.payouts
1206					.into_iter()
1207					.map(|x| x.1)
1208					.fold(Zero::zero(), |acc: BalanceOf<T, I>, x| acc.saturating_add(x));
1209				Self::unreserve_payout(total);
1210			}
1211			SuspendedMembers::<T, I>::remove(&who);
1212			Self::deposit_event(Event::<T, I>::SuspendedMemberJudgement { who, judged: forgive });
1213			Ok(Pays::No.into())
1214		}
1215
1216		/// Change the maximum number of members in society and the maximum number of new candidates
1217		/// in a single intake period.
1218		///
1219		/// The dispatch origin for this call must be Signed by the Founder.
1220		///
1221		/// Parameters:
1222		/// - `max_members` - The maximum number of members for the society. This must be no less
1223		///   than the current number of members.
1224		/// - `max_intake` - The maximum number of candidates per intake period.
1225		/// - `max_strikes`: The maximum number of strikes a member may get before they become
1226		///   suspended and may only be reinstated by the founder.
1227		/// - `candidate_deposit`: The deposit required to make a bid for membership of the group.
1228		#[pallet::call_index(11)]
1229		#[pallet::weight(T::WeightInfo::set_parameters())]
1230		pub fn set_parameters(
1231			origin: OriginFor<T>,
1232			max_members: u32,
1233			max_intake: u32,
1234			max_strikes: u32,
1235			candidate_deposit: BalanceOf<T, I>,
1236		) -> DispatchResult {
1237			ensure!(
1238				Some(ensure_signed(origin)?) == Founder::<T, I>::get(),
1239				Error::<T, I>::NotFounder
1240			);
1241			ensure!(max_members >= MemberCount::<T, I>::get(), Error::<T, I>::MaxMembers);
1242			let params = GroupParams { max_members, max_intake, max_strikes, candidate_deposit };
1243			Parameters::<T, I>::put(&params);
1244			Self::deposit_event(Event::<T, I>::NewParams { params });
1245			Ok(())
1246		}
1247
1248		/// Punish the skeptic with a strike if they did not vote on a candidate. Callable by the
1249		/// candidate.
1250		#[pallet::call_index(12)]
1251		#[pallet::weight(T::WeightInfo::punish_skeptic())]
1252		pub fn punish_skeptic(origin: OriginFor<T>) -> DispatchResultWithPostInfo {
1253			let candidate = ensure_signed(origin)?;
1254			let mut candidacy =
1255				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1256			ensure!(!candidacy.skeptic_struck, Error::<T, I>::AlreadyPunished);
1257			ensure!(!Self::in_progress(candidacy.round), Error::<T, I>::InProgress);
1258			let punished = Self::check_skeptic(&candidate, &mut candidacy);
1259			Candidates::<T, I>::insert(&candidate, candidacy);
1260			Ok(if punished { Pays::No } else { Pays::Yes }.into())
1261		}
1262
1263		/// Transform an approved candidate into a member. Callable only by the
1264		/// the candidate, and only after the period for voting has ended.
1265		#[pallet::call_index(13)]
1266		#[pallet::weight(T::WeightInfo::claim_membership())]
1267		pub fn claim_membership(origin: OriginFor<T>) -> DispatchResultWithPostInfo {
1268			let candidate = ensure_signed(origin)?;
1269			let candidacy =
1270				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1271			ensure!(candidacy.tally.clear_approval(), Error::<T, I>::NotApproved);
1272			ensure!(!Self::in_progress(candidacy.round), Error::<T, I>::InProgress);
1273			Self::induct_member(candidate, candidacy, 0)?;
1274			Ok(Pays::No.into())
1275		}
1276
1277		/// Transform an approved candidate into a member. Callable only by the Signed origin of the
1278		/// Founder, only after the period for voting has ended and only when the candidate is not
1279		/// clearly rejected.
1280		#[pallet::call_index(14)]
1281		#[pallet::weight(T::WeightInfo::bestow_membership())]
1282		pub fn bestow_membership(
1283			origin: OriginFor<T>,
1284			candidate: T::AccountId,
1285		) -> DispatchResultWithPostInfo {
1286			ensure!(
1287				Some(ensure_signed(origin)?) == Founder::<T, I>::get(),
1288				Error::<T, I>::NotFounder
1289			);
1290			let candidacy =
1291				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1292			ensure!(!candidacy.tally.clear_rejection(), Error::<T, I>::Rejected);
1293			ensure!(!Self::in_progress(candidacy.round), Error::<T, I>::InProgress);
1294			Self::induct_member(candidate, candidacy, 0)?;
1295			Ok(Pays::No.into())
1296		}
1297
1298		/// Remove the candidate's application from the society. Callable only by the Signed origin
1299		/// of the Founder, only after the period for voting has ended, and only when they do not
1300		/// have a clear approval.
1301		///
1302		/// Any bid deposit is lost and voucher is banned.
1303		#[pallet::call_index(15)]
1304		#[pallet::weight(T::WeightInfo::kick_candidate())]
1305		pub fn kick_candidate(
1306			origin: OriginFor<T>,
1307			candidate: T::AccountId,
1308		) -> DispatchResultWithPostInfo {
1309			ensure!(
1310				Some(ensure_signed(origin)?) == Founder::<T, I>::get(),
1311				Error::<T, I>::NotFounder
1312			);
1313			let mut candidacy =
1314				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1315			ensure!(!Self::in_progress(candidacy.round), Error::<T, I>::InProgress);
1316			ensure!(!candidacy.tally.clear_approval(), Error::<T, I>::Approved);
1317			Self::check_skeptic(&candidate, &mut candidacy);
1318			Self::reject_candidate(&candidate, &candidacy.kind);
1319			Candidates::<T, I>::remove(&candidate);
1320			Ok(Pays::No.into())
1321		}
1322
1323		/// Remove the candidate's application from the society. Callable only by the candidate.
1324		///
1325		/// Any bid deposit is lost and voucher is banned.
1326		#[pallet::call_index(16)]
1327		#[pallet::weight(T::WeightInfo::resign_candidacy())]
1328		pub fn resign_candidacy(origin: OriginFor<T>) -> DispatchResultWithPostInfo {
1329			let candidate = ensure_signed(origin)?;
1330			let mut candidacy =
1331				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1332			if !Self::in_progress(candidacy.round) {
1333				Self::check_skeptic(&candidate, &mut candidacy);
1334			}
1335			Self::reject_candidate(&candidate, &candidacy.kind);
1336			Candidates::<T, I>::remove(&candidate);
1337			Ok(Pays::No.into())
1338		}
1339
1340		/// Remove a `candidate`'s failed application from the society. Callable by any
1341		/// signed origin but only at the end of the subsequent round and only for
1342		/// a candidate with more rejections than approvals.
1343		///
1344		/// The bid deposit is lost and the voucher is banned.
1345		#[pallet::call_index(17)]
1346		#[pallet::weight(T::WeightInfo::drop_candidate())]
1347		pub fn drop_candidate(
1348			origin: OriginFor<T>,
1349			candidate: T::AccountId,
1350		) -> DispatchResultWithPostInfo {
1351			ensure_signed(origin)?;
1352			let candidacy =
1353				Candidates::<T, I>::get(&candidate).ok_or(Error::<T, I>::NotCandidate)?;
1354			ensure!(candidacy.tally.clear_rejection(), Error::<T, I>::NotRejected);
1355			ensure!(RoundCount::<T, I>::get() > candidacy.round + 1, Error::<T, I>::TooEarly);
1356			Self::reject_candidate(&candidate, &candidacy.kind);
1357			Candidates::<T, I>::remove(&candidate);
1358			Ok(Pays::No.into())
1359		}
1360
1361		/// Remove up to `max` stale votes for the given `candidate`.
1362		///
1363		/// May be called by any Signed origin, but only after the candidate's candidacy is ended.
1364		#[pallet::call_index(18)]
1365		#[pallet::weight(T::WeightInfo::cleanup_candidacy())]
1366		pub fn cleanup_candidacy(
1367			origin: OriginFor<T>,
1368			candidate: T::AccountId,
1369			max: u32,
1370		) -> DispatchResultWithPostInfo {
1371			ensure_signed(origin)?;
1372			ensure!(!Candidates::<T, I>::contains_key(&candidate), Error::<T, I>::InProgress);
1373			let maybe_cursor = VoteClearCursor::<T, I>::get(&candidate);
1374			let r =
1375				Votes::<T, I>::clear_prefix(&candidate, max, maybe_cursor.as_ref().map(|x| &x[..]));
1376			if let Some(cursor) = r.maybe_cursor {
1377				VoteClearCursor::<T, I>::insert(&candidate, BoundedVec::truncate_from(cursor));
1378			}
1379			Ok(if r.loops == 0 { Pays::Yes } else { Pays::No }.into())
1380		}
1381
1382		/// Remove up to `max` stale votes for the defender in the given `challenge_round`.
1383		///
1384		/// May be called by any Signed origin, but only after the challenge round is ended.
1385		#[pallet::call_index(19)]
1386		#[pallet::weight(T::WeightInfo::cleanup_challenge())]
1387		pub fn cleanup_challenge(
1388			origin: OriginFor<T>,
1389			challenge_round: RoundIndex,
1390			max: u32,
1391		) -> DispatchResultWithPostInfo {
1392			ensure_signed(origin)?;
1393			ensure!(
1394				challenge_round < ChallengeRoundCount::<T, I>::get(),
1395				Error::<T, I>::InProgress
1396			);
1397			let _ = DefenderVotes::<T, I>::clear_prefix(challenge_round, max, None);
1398			// clear_prefix() v2 is always returning backend = 0, ignoring it till v3.
1399			// let (_, backend, _, _) = r.deconstruct();
1400			// if backend == 0 { return Err(Error::<T, I>::NoVotes.into()); };
1401			Ok(Pays::No.into())
1402		}
1403
1404		/// Poke the deposit reserved when bidding.
1405		///
1406		/// The dispatch origin for this call must be _Signed_ and must be the bidder.
1407		///
1408		/// The transaction fee is waived if the deposit is changed after poking/reconsideration.
1409		///
1410		/// Emits `DepositPoked` if successful.
1411		#[pallet::call_index(20)]
1412		#[pallet::weight(T::WeightInfo::poke_deposit())]
1413		pub fn poke_deposit(origin: OriginFor<T>) -> DispatchResultWithPostInfo {
1414			let who = ensure_signed(origin)?;
1415
1416			// Get current bids and find the bidder's bid
1417			let mut bids = Bids::<T, I>::get();
1418			let bid = bids.iter_mut().find(|bid| bid.who == who).ok_or(Error::<T, I>::NotBidder)?;
1419
1420			// Only handle deposit bids
1421			let old_deposit = match &bid.kind {
1422				BidKind::Deposit(amount) => *amount,
1423				_ => return Err(Error::<T, I>::NoDeposit.into()),
1424			};
1425
1426			let params = Parameters::<T, I>::get().ok_or(Error::<T, I>::NotGroup)?;
1427			let new_deposit = params.candidate_deposit;
1428
1429			if old_deposit == new_deposit {
1430				return Ok(Pays::Yes.into());
1431			}
1432
1433			if new_deposit > old_deposit {
1434				// Need to reserve more
1435				let extra = new_deposit.saturating_sub(old_deposit);
1436				T::Currency::reserve(&who, extra)?;
1437			} else {
1438				// Need to unreserve some
1439				let excess = old_deposit.saturating_sub(new_deposit);
1440				let remaining_unreserved = T::Currency::unreserve(&who, excess);
1441				if !remaining_unreserved.is_zero() {
1442					defensive!(
1443						"Failed to unreserve for full amount for bid (Requested, Actual)",
1444						(excess, excess.saturating_sub(remaining_unreserved))
1445					);
1446				}
1447			}
1448
1449			bid.kind = BidKind::Deposit(new_deposit);
1450			Bids::<T, I>::put(bids);
1451
1452			Self::deposit_event(Event::<T, I>::DepositPoked {
1453				who: who.clone(),
1454				old_deposit,
1455				new_deposit,
1456			});
1457
1458			Ok(Pays::No.into())
1459		}
1460	}
1461}
1462
1463/// Simple ensure origin struct to filter for the founder account.
1464pub struct EnsureFounder<T>(core::marker::PhantomData<T>);
1465impl<T: Config> EnsureOrigin<<T as frame_system::Config>::RuntimeOrigin> for EnsureFounder<T> {
1466	type Success = T::AccountId;
1467	fn try_origin(o: T::RuntimeOrigin) -> Result<Self::Success, T::RuntimeOrigin> {
1468		match (o.as_signer(), Founder::<T>::get()) {
1469			(Some(who), Some(f)) if *who == f => Ok(f),
1470			_ => Err(o),
1471		}
1472	}
1473
1474	#[cfg(feature = "runtime-benchmarks")]
1475	fn try_successful_origin() -> Result<T::RuntimeOrigin, ()> {
1476		let founder = Founder::<T>::get().ok_or(())?;
1477		Ok(T::RuntimeOrigin::from(frame_system::RawOrigin::Signed(founder)))
1478	}
1479}
1480
1481impl_ensure_origin_with_arg_ignoring_arg! {
1482	impl<{ T: Config, A }>
1483		EnsureOriginWithArg<T::RuntimeOrigin, A> for EnsureFounder<T>
1484	{}
1485}
1486
1487#[derive(Debug, PartialEq, Eq)]
1488pub enum Period<BlockNumber> {
1489	Voting { elapsed: BlockNumber, more: BlockNumber },
1490	Claim { elapsed: BlockNumber, more: BlockNumber },
1491	Intake { elapsed: BlockNumber },
1492}
1493
1494impl<T: Config<I>, I: 'static> Pallet<T, I> {
1495	/// Get the period we are currently in.
1496	fn period() -> Period<BlockNumberFor<T, I>> {
1497		let claim_period = T::ClaimPeriod::get();
1498		let voting_period = T::VotingPeriod::get();
1499		let rotation_period = voting_period + claim_period;
1500		let now = T::BlockNumberProvider::current_block_number();
1501		let phase = now % rotation_period;
1502		if now >= Self::next_intake_at() {
1503			Period::Intake { elapsed: now - Self::next_intake_at() }
1504		} else if phase < voting_period {
1505			Period::Voting { elapsed: phase, more: voting_period - phase }
1506		} else {
1507			Period::Claim { elapsed: phase - voting_period, more: rotation_period - phase }
1508		}
1509	}
1510
1511	/// Next intake (candidate/membership) rotation scheduled with [Config::BlockNumberProvider].
1512	///
1513	/// Rounds the previous block number up to the next rotation period (voting + claim periods).
1514	pub fn next_intake_at() -> BlockNumberFor<T, I> {
1515		match NextIntakeAt::<T, I>::get() {
1516			Some(next) => next,
1517			None => {
1518				// executed once.
1519				let now = T::BlockNumberProvider::current_block_number();
1520				let prev_block = now.saturating_sub(BlockNumberFor::<T, I>::one());
1521				let rotation_period = T::VotingPeriod::get().saturating_add(T::ClaimPeriod::get());
1522				let elapsed = prev_block % rotation_period;
1523				let next_intake_at = prev_block + (rotation_period - elapsed);
1524				NextIntakeAt::<T, I>::put(next_intake_at);
1525				next_intake_at
1526			},
1527		}
1528	}
1529
1530	/// Set the next intake (candidate/membership) rotation.
1531	///
1532	/// This supposed to be called once the current intake is executed.
1533	fn set_next_intake_at() {
1534		let prev_next_intake_at = Self::next_intake_at();
1535		let next_intake_at = prev_next_intake_at
1536			.saturating_add(T::VotingPeriod::get().saturating_add(T::ClaimPeriod::get()));
1537		NextIntakeAt::<T, I>::put(next_intake_at);
1538	}
1539
1540	/// Returns the next challenge rotation scheduled with [Config::BlockNumberProvider].
1541	///
1542	/// Rounds the previous block number up to the next multiple of the challenge duration.
1543	pub fn next_challenge_at() -> BlockNumberFor<T, I> {
1544		match NextChallengeAt::<T, I>::get() {
1545			Some(next) => next,
1546			None => {
1547				// executed once.
1548				let now = T::BlockNumberProvider::current_block_number();
1549				let prev_block = now.saturating_sub(BlockNumberFor::<T, I>::one());
1550				let challenge_period = T::ChallengePeriod::get();
1551				let elapsed = prev_block % challenge_period;
1552				let next_challenge_at = prev_block + (challenge_period - elapsed);
1553				NextChallengeAt::<T, I>::put(next_challenge_at);
1554				next_challenge_at
1555			},
1556		}
1557	}
1558
1559	/// Set the next challenge rotation.
1560	///
1561	/// This supposed to be called once the current challenge is executed.
1562	fn set_next_challenge_at() {
1563		let prev_next_challenge_at = Self::next_challenge_at();
1564		let next_challenge_at = prev_next_challenge_at.saturating_add(T::ChallengePeriod::get());
1565		NextChallengeAt::<T, I>::put(next_challenge_at);
1566	}
1567
1568	/// Returns true if the given `target_round` is still in its initial voting phase.
1569	fn in_progress(target_round: RoundIndex) -> bool {
1570		let round = RoundCount::<T, I>::get();
1571		target_round == round && matches!(Self::period(), Period::Voting { .. })
1572	}
1573
1574	/// Returns the new vote.
1575	fn do_vote(maybe_old: Option<Vote>, approve: bool, rank: Rank, tally: &mut Tally) -> Vote {
1576		match maybe_old {
1577			Some(Vote { approve: true, weight }) => tally.approvals.saturating_reduce(weight),
1578			Some(Vote { approve: false, weight }) => tally.rejections.saturating_reduce(weight),
1579			_ => {},
1580		}
1581		let weight_root = rank + 1;
1582		let weight = weight_root * weight_root;
1583		match approve {
1584			true => tally.approvals.saturating_accrue(weight),
1585			false => tally.rejections.saturating_accrue(weight),
1586		}
1587		Vote { approve, weight }
1588	}
1589
1590	/// Returns `true` if a punishment was given.
1591	fn check_skeptic(
1592		candidate: &T::AccountId,
1593		candidacy: &mut Candidacy<T::AccountId, BalanceOf<T, I>>,
1594	) -> bool {
1595		if RoundCount::<T, I>::get() != candidacy.round || candidacy.skeptic_struck {
1596			return false
1597		}
1598		// We expect the skeptic to have voted.
1599		let skeptic = match Skeptic::<T, I>::get() {
1600			Some(s) => s,
1601			None => return false,
1602		};
1603		let maybe_vote = Votes::<T, I>::get(&candidate, &skeptic);
1604		let approved = candidacy.tally.clear_approval();
1605		let rejected = candidacy.tally.clear_rejection();
1606		match (maybe_vote, approved, rejected) {
1607			(None, _, _) |
1608			(Some(Vote { approve: true, .. }), false, true) |
1609			(Some(Vote { approve: false, .. }), true, false) => {
1610				// Can't do much if the punishment doesn't work out.
1611				if Self::strike_member(&skeptic).is_ok() {
1612					candidacy.skeptic_struck = true;
1613					true
1614				} else {
1615					false
1616				}
1617			},
1618			_ => false,
1619		}
1620	}
1621
1622	/// End the current challenge period and start a new one.
1623	fn rotate_challenge(rng: &mut impl RngCore) {
1624		let mut next_defender = None;
1625		let mut round = ChallengeRoundCount::<T, I>::get();
1626
1627		// End current defender rotation
1628		if let Some((defender, skeptic, tally)) = Defending::<T, I>::get() {
1629			// We require strictly more approvals, since the member should be voting for themselves.
1630			if !tally.more_approvals() {
1631				// Member has failed the challenge: Suspend them. This will fail if they are Head
1632				// or Founder, in which case we ignore.
1633				let _ = Self::suspend_member(&defender);
1634			}
1635
1636			// Check defender skeptic voted and that their vote was with the majority.
1637			let skeptic_vote = DefenderVotes::<T, I>::get(round, &skeptic);
1638			match (skeptic_vote, tally.more_approvals(), tally.more_rejections()) {
1639				(None, _, _) |
1640				(Some(Vote { approve: true, .. }), false, true) |
1641				(Some(Vote { approve: false, .. }), true, false) => {
1642					// Punish skeptic and challenge them next.
1643					let _ = Self::strike_member(&skeptic);
1644					let founder = Founder::<T, I>::get();
1645					let head = Head::<T, I>::get();
1646					if Some(&skeptic) != founder.as_ref() && Some(&skeptic) != head.as_ref() {
1647						next_defender = Some(skeptic);
1648					}
1649				},
1650				_ => {},
1651			}
1652			round.saturating_inc();
1653			ChallengeRoundCount::<T, I>::put(round);
1654		}
1655
1656		// Avoid challenging if there's only two members since we never challenge the Head or
1657		// the Founder.
1658		if MemberCount::<T, I>::get() > 2 {
1659			let defender = next_defender
1660				.or_else(|| Self::pick_defendant(rng))
1661				.expect("exited if members empty; qed");
1662			let skeptic =
1663				Self::pick_member_except(rng, &defender).expect("exited if members empty; qed");
1664			Self::deposit_event(Event::<T, I>::Challenged { member: defender.clone() });
1665			Defending::<T, I>::put((defender, skeptic, Tally::default()));
1666		} else {
1667			Defending::<T, I>::kill();
1668		}
1669	}
1670
1671	/// End the current intake period and begin a new one.
1672	///
1673	/// ---------------------------------------------
1674	///  #10  || #11           _              || #12
1675	///       || Voting        | Claiming     ||
1676	/// ---------------------------------------------
1677	fn rotate_intake(rng: &mut impl RngCore) {
1678		// We assume there's at least one member or this logic won't work.
1679		let member_count = MemberCount::<T, I>::get();
1680		if member_count < 1 {
1681			return
1682		}
1683		let maybe_head = NextHead::<T, I>::take();
1684		if let Some(head) = maybe_head {
1685			Head::<T, I>::put(&head.who);
1686		}
1687
1688		// Bump the pot by at most `PeriodSpend`, but less if there's not very much left in our
1689		// account.
1690		let mut pot = Pot::<T, I>::get();
1691		let unaccounted = T::Currency::free_balance(&Self::account_id()).saturating_sub(pot);
1692		pot.saturating_accrue(T::PeriodSpend::get().min(unaccounted / 2u8.into()));
1693		Pot::<T, I>::put(&pot);
1694
1695		// Bump round and create the new intake.
1696		let mut round_count = RoundCount::<T, I>::get();
1697		round_count.saturating_inc();
1698		let candidate_count = Self::select_new_candidates(round_count, member_count, pot);
1699		if candidate_count > 0 {
1700			// Select a member at random and make them the skeptic for this round.
1701			let skeptic = Self::pick_member(rng).expect("exited if members empty; qed");
1702			Skeptic::<T, I>::put(skeptic);
1703		}
1704		RoundCount::<T, I>::put(round_count);
1705	}
1706
1707	/// Remove a selection of bidding accounts such that the total bids is no greater than `Pot` and
1708	/// the number of bids would not surpass `MaxMembers` if all were accepted. At most one bid may
1709	/// be zero.
1710	///
1711	/// Candidates are inserted from each bidder.
1712	///
1713	/// The number of candidates inserted are returned.
1714	pub fn select_new_candidates(
1715		round: RoundIndex,
1716		member_count: u32,
1717		pot: BalanceOf<T, I>,
1718	) -> u32 {
1719		// Get the number of left-most bidders whose bids add up to less than `pot`.
1720		let mut bids = Bids::<T, I>::get();
1721		let params = match Parameters::<T, I>::get() {
1722			Some(params) => params,
1723			None => return 0,
1724		};
1725		let max_selections: u32 = params
1726			.max_intake
1727			.min(params.max_members.saturating_sub(member_count))
1728			.min(bids.len() as u32);
1729
1730		let mut selections = 0;
1731		// A running total of the cost to onboard these bids
1732		let mut total_cost: BalanceOf<T, I> = Zero::zero();
1733
1734		bids.retain(|bid| {
1735			// We only accept a zero bid as the first selection.
1736			total_cost.saturating_accrue(bid.value);
1737			let accept = selections < max_selections &&
1738				(!bid.value.is_zero() || selections == 0) &&
1739				total_cost <= pot;
1740			if accept {
1741				let candidacy = Candidacy {
1742					round,
1743					kind: bid.kind.clone(),
1744					bid: bid.value,
1745					tally: Default::default(),
1746					skeptic_struck: false,
1747				};
1748				Candidates::<T, I>::insert(&bid.who, candidacy);
1749				selections.saturating_inc();
1750			}
1751			!accept
1752		});
1753
1754		// No need to reset Bids if we're not taking anything.
1755		Bids::<T, I>::put(&bids);
1756		selections
1757	}
1758
1759	/// Puts a bid into storage ordered by smallest to largest value.
1760	/// Allows a maximum of 1000 bids in queue, removing largest value people first.
1761	fn insert_bid(
1762		bids: &mut BoundedVec<Bid<T::AccountId, BalanceOf<T, I>>, T::MaxBids>,
1763		who: &T::AccountId,
1764		value: BalanceOf<T, I>,
1765		bid_kind: BidKind<T::AccountId, BalanceOf<T, I>>,
1766	) {
1767		let pos = bids.iter().position(|bid| bid.value > value).unwrap_or(bids.len());
1768		let r = bids.force_insert_keep_left(pos, Bid { value, who: who.clone(), kind: bid_kind });
1769		let maybe_discarded = match r {
1770			Ok(x) => x,
1771			Err(x) => Some(x),
1772		};
1773		if let Some(discarded) = maybe_discarded {
1774			Self::clean_bid(&discarded);
1775			Self::deposit_event(Event::<T, I>::AutoUnbid { candidate: discarded.who });
1776		}
1777	}
1778
1779	/// Either unreserve the deposit or free up the vouching member.
1780	///
1781	/// In neither case can we do much if the action isn't completable, but there's
1782	/// no reason that either should fail.
1783	///
1784	/// WARNING: This alters the voucher item of `Members`. You must ensure that you do not
1785	/// accidentally overwrite it with an older value after calling this.
1786	fn clean_bid(bid: &Bid<T::AccountId, BalanceOf<T, I>>) {
1787		match &bid.kind {
1788			BidKind::Deposit(deposit) => {
1789				let err_amount = T::Currency::unreserve(&bid.who, *deposit);
1790				debug_assert!(err_amount.is_zero());
1791			},
1792			BidKind::Vouch(voucher, _) => {
1793				Members::<T, I>::mutate_extant(voucher, |record| record.vouching = None);
1794			},
1795		}
1796	}
1797
1798	/// Either repatriate the deposit into the Society account or ban the vouching member.
1799	///
1800	/// In neither case can we do much if the action isn't completable, but there's
1801	/// no reason that either should fail.
1802	///
1803	/// WARNING: This alters the voucher item of `Members`. You must ensure that you do not
1804	/// accidentally overwrite it with an older value after calling this.
1805	fn reject_candidate(who: &T::AccountId, kind: &BidKind<T::AccountId, BalanceOf<T, I>>) {
1806		match kind {
1807			BidKind::Deposit(deposit) => {
1808				let pot = Self::account_id();
1809				let free = BalanceStatus::Free;
1810				let r = T::Currency::repatriate_reserved(&who, &pot, *deposit, free);
1811				debug_assert!(r.is_ok());
1812			},
1813			BidKind::Vouch(voucher, _) => {
1814				Members::<T, I>::mutate_extant(voucher, |record| {
1815					record.vouching = Some(VouchingStatus::Banned)
1816				});
1817			},
1818		}
1819	}
1820
1821	/// Check a user has a bid.
1822	fn has_bid(bids: &Vec<Bid<T::AccountId, BalanceOf<T, I>>>, who: &T::AccountId) -> bool {
1823		// Bids are ordered by `value`, so we cannot binary search for a user.
1824		bids.iter().any(|bid| bid.who == *who)
1825	}
1826
1827	/// Add a member to the members list. If the user is already a member, do nothing. Can fail when
1828	/// `MaxMember` limit is reached, but in that case it has no side-effects.
1829	///
1830	/// Set the `payouts` for the member. NOTE: This *WILL NOT RESERVE THE FUNDS TO MAKE THE
1831	/// PAYOUT*. Only set this to be non-empty if you already have the funds reserved in the Payouts
1832	/// account.
1833	///
1834	/// NOTE: Generally you should not use this, and instead use `add_new_member` or
1835	/// `reinstate_member`, whose names clearly match the desired intention.
1836	fn insert_member(who: &T::AccountId, rank: Rank) -> DispatchResult {
1837		let params = Parameters::<T, I>::get().ok_or(Error::<T, I>::NotGroup)?;
1838		ensure!(MemberCount::<T, I>::get() < params.max_members, Error::<T, I>::MaxMembers);
1839		let index = MemberCount::<T, I>::mutate(|i| {
1840			i.saturating_accrue(1);
1841			*i - 1
1842		});
1843		let record = MemberRecord { rank, strikes: 0, vouching: None, index };
1844		Members::<T, I>::insert(who, record);
1845		MemberByIndex::<T, I>::insert(index, who);
1846		Ok(())
1847	}
1848
1849	/// Add a member back to the members list, setting their `rank` and `payouts`.
1850	///
1851	/// Can fail when `MaxMember` limit is reached, but in that case it has no side-effects.
1852	///
1853	/// The `payouts` value must be exactly as it was prior to suspension since no further funds
1854	/// will be reserved.
1855	fn reinstate_member(who: &T::AccountId, rank: Rank) -> DispatchResult {
1856		Self::insert_member(who, rank)
1857	}
1858
1859	/// Add a member to the members list. If the user is already a member, do nothing. Can fail when
1860	/// `MaxMember` limit is reached, but in that case it has no side-effects.
1861	fn add_new_member(who: &T::AccountId, rank: Rank) -> DispatchResult {
1862		Self::insert_member(who, rank)
1863	}
1864
1865	/// Induct a new member into the set.
1866	fn induct_member(
1867		candidate: T::AccountId,
1868		mut candidacy: Candidacy<T::AccountId, BalanceOf<T, I>>,
1869		rank: Rank,
1870	) -> DispatchResult {
1871		Self::add_new_member(&candidate, rank)?;
1872		Self::check_skeptic(&candidate, &mut candidacy);
1873
1874		let next_head = NextHead::<T, I>::get()
1875			.filter(|old| {
1876				old.round > candidacy.round ||
1877					old.round == candidacy.round && old.bid < candidacy.bid
1878			})
1879			.unwrap_or_else(|| IntakeRecord {
1880				who: candidate.clone(),
1881				bid: candidacy.bid,
1882				round: candidacy.round,
1883			});
1884		NextHead::<T, I>::put(next_head);
1885
1886		let now = T::BlockNumberProvider::current_block_number();
1887		let maturity = now + Self::lock_duration(MemberCount::<T, I>::get());
1888		Self::reward_bidder(&candidate, candidacy.bid, candidacy.kind, maturity);
1889
1890		Candidates::<T, I>::remove(&candidate);
1891		Ok(())
1892	}
1893
1894	fn strike_member(who: &T::AccountId) -> DispatchResult {
1895		let mut record = Members::<T, I>::get(who).ok_or(Error::<T, I>::NotMember)?;
1896		record.strikes.saturating_inc();
1897		Members::<T, I>::insert(who, &record);
1898		// ^^^ Keep the member record mutation self-contained as we might be suspending them later
1899		// in this function.
1900
1901		if record.strikes >= T::GraceStrikes::get() {
1902			// Too many strikes: slash the payout in half.
1903			let total_payout = Payouts::<T, I>::get(who)
1904				.payouts
1905				.iter()
1906				.fold(BalanceOf::<T, I>::zero(), |acc, x| acc.saturating_add(x.1));
1907			Self::slash_payout(who, total_payout / 2u32.into());
1908		}
1909
1910		let params = Parameters::<T, I>::get().ok_or(Error::<T, I>::NotGroup)?;
1911		if record.strikes >= params.max_strikes {
1912			// Way too many strikes: suspend.
1913			let _ = Self::suspend_member(who);
1914		}
1915		Ok(())
1916	}
1917
1918	/// Remove a member from the members list and return the candidacy.
1919	///
1920	/// If the member was vouching, then this will be reset. Any bidders that the member was
1921	/// vouching for will be cancelled unless they are already selected as candidates (in which case
1922	/// they will be able to stand).
1923	///
1924	/// If the member has existing payouts, they will be retained in the resultant `MemberRecord`
1925	/// and the funds will remain reserved.
1926	///
1927	/// The Head and the Founder may never be removed.
1928	pub fn remove_member(m: &T::AccountId) -> Result<MemberRecord, DispatchError> {
1929		ensure!(Head::<T, I>::get().as_ref() != Some(m), Error::<T, I>::Head);
1930		ensure!(Founder::<T, I>::get().as_ref() != Some(m), Error::<T, I>::Founder);
1931		if let Some(mut record) = Members::<T, I>::get(m) {
1932			let index = record.index;
1933			let last_index = MemberCount::<T, I>::mutate(|i| {
1934				i.saturating_reduce(1);
1935				*i
1936			});
1937			if index != last_index {
1938				// Move the member with the last index down to the index of the member to be
1939				// removed.
1940				if let Some(other) = MemberByIndex::<T, I>::get(last_index) {
1941					MemberByIndex::<T, I>::insert(index, &other);
1942					Members::<T, I>::mutate(other, |m_r| {
1943						if let Some(r) = m_r {
1944							r.index = index
1945						}
1946					});
1947				} else {
1948					debug_assert!(false, "ERROR: No member at the last index position?");
1949				}
1950			}
1951
1952			MemberByIndex::<T, I>::remove(last_index);
1953			Members::<T, I>::remove(m);
1954			// Remove their vouching status, potentially unbanning them in the future.
1955			if record.vouching.take() == Some(VouchingStatus::Vouching) {
1956				// Try to remove their bid if they are vouching.
1957				// If their vouch is already a candidate, do nothing.
1958				Bids::<T, I>::mutate(|bids|
1959					// Try to find the matching bid
1960					if let Some(pos) = bids.iter().position(|b| b.kind.is_vouch(&m)) {
1961						// Remove the bid, and emit an event
1962						let vouched = bids.remove(pos).who;
1963						Self::deposit_event(Event::<T, I>::Unvouch { candidate: vouched });
1964					}
1965				);
1966			}
1967			Ok(record)
1968		} else {
1969			Err(Error::<T, I>::NotMember.into())
1970		}
1971	}
1972
1973	/// Remove a member from the members set and add them to the suspended members.
1974	///
1975	/// If the member was vouching, then this will be reset. Any bidders that the member was
1976	/// vouching for will be cancelled unless they are already selected as candidates (in which case
1977	/// they will be able to stand).
1978	fn suspend_member(who: &T::AccountId) -> DispatchResult {
1979		let record = Self::remove_member(&who)?;
1980		SuspendedMembers::<T, I>::insert(who, record);
1981		Self::deposit_event(Event::<T, I>::MemberSuspended { member: who.clone() });
1982		Ok(())
1983	}
1984
1985	/// Select a member at random, given the RNG `rng`.
1986	///
1987	/// If no members exist (or the state is inconsistent), then `None` may be returned.
1988	fn pick_member(rng: &mut impl RngCore) -> Option<T::AccountId> {
1989		let member_count = MemberCount::<T, I>::get();
1990		if member_count == 0 {
1991			return None
1992		}
1993		let random_index = rng.next_u32() % member_count;
1994		MemberByIndex::<T, I>::get(random_index)
1995	}
1996
1997	/// Select a member at random except `exception`, given the RNG `rng`.
1998	///
1999	/// If `exception` is the only member (or the state is inconsistent), then `None` may be
2000	/// returned.
2001	fn pick_member_except(
2002		rng: &mut impl RngCore,
2003		exception: &T::AccountId,
2004	) -> Option<T::AccountId> {
2005		let member_count = MemberCount::<T, I>::get();
2006		if member_count <= 1 {
2007			return None
2008		}
2009		let random_index = rng.next_u32() % (member_count - 1);
2010		let pick = MemberByIndex::<T, I>::get(random_index);
2011		if pick.as_ref() == Some(exception) {
2012			MemberByIndex::<T, I>::get(member_count - 1)
2013		} else {
2014			pick
2015		}
2016	}
2017
2018	/// Select a member who is able to defend at random, given the RNG `rng`.
2019	///
2020	/// If only the Founder and Head members exist (or the state is inconsistent), then `None`
2021	/// may be returned.
2022	fn pick_defendant(rng: &mut impl RngCore) -> Option<T::AccountId> {
2023		let member_count = MemberCount::<T, I>::get();
2024		if member_count <= 2 {
2025			return None
2026		}
2027		// Founder is always at index 0, so we should never pick that one.
2028		// Head will typically but not always be the highest index. We assume it is for now and
2029		// fix it up later if not.
2030		let head = Head::<T, I>::get();
2031		let pickable_count = member_count - if head.is_some() { 2 } else { 1 };
2032		let random_index = rng.next_u32() % pickable_count + 1;
2033		let pick = MemberByIndex::<T, I>::get(random_index);
2034		if pick == head && head.is_some() {
2035			// Turns out that head was not the last index since we managed to pick it. Exchange our
2036			// pick for the last index.
2037			MemberByIndex::<T, I>::get(member_count - 1)
2038		} else {
2039			pick
2040		}
2041	}
2042
2043	/// Pay an accepted candidate their bid value.
2044	fn reward_bidder(
2045		candidate: &T::AccountId,
2046		value: BalanceOf<T, I>,
2047		kind: BidKind<T::AccountId, BalanceOf<T, I>>,
2048		maturity: BlockNumberFor<T, I>,
2049	) {
2050		let value = match kind {
2051			BidKind::Deposit(deposit) => {
2052				// In the case that a normal deposit bid is accepted we unreserve
2053				// the deposit.
2054				let err_amount = T::Currency::unreserve(candidate, deposit);
2055				debug_assert!(err_amount.is_zero());
2056				value
2057			},
2058			BidKind::Vouch(voucher, tip) => {
2059				// Check that the voucher is still vouching, else some other logic may have removed
2060				// their status.
2061				if let Some(mut record) = Members::<T, I>::get(&voucher) {
2062					if let Some(VouchingStatus::Vouching) = record.vouching {
2063						// In the case that a vouched-for bid is accepted we unset the
2064						// vouching status and transfer the tip over to the voucher.
2065						record.vouching = None;
2066						Self::bump_payout(&voucher, maturity, tip.min(value));
2067						Members::<T, I>::insert(&voucher, record);
2068						value.saturating_sub(tip)
2069					} else {
2070						value
2071					}
2072				} else {
2073					value
2074				}
2075			},
2076		};
2077
2078		Self::bump_payout(candidate, maturity, value);
2079	}
2080
2081	/// Bump the payout amount of `who`, to be unlocked at the given block number.
2082	///
2083	/// It is the caller's duty to ensure that `who` is already a member. This does nothing if `who`
2084	/// is not a member or if `value` is zero.
2085	fn bump_payout(who: &T::AccountId, when: BlockNumberFor<T, I>, value: BalanceOf<T, I>) {
2086		if value.is_zero() {
2087			return
2088		}
2089		if let Some(MemberRecord { rank: 0, .. }) = Members::<T, I>::get(who) {
2090			Payouts::<T, I>::mutate(who, |record| {
2091				// Members of rank 1 never get payouts.
2092				match record.payouts.binary_search_by_key(&when, |x| x.0) {
2093					Ok(index) => record.payouts[index].1.saturating_accrue(value),
2094					Err(index) => {
2095						// If they have too many pending payouts, then we take discard the payment.
2096						let _ = record.payouts.try_insert(index, (when, value));
2097					},
2098				}
2099			});
2100			Self::reserve_payout(value);
2101		}
2102	}
2103
2104	/// Attempt to slash the payout of some member. Return the total amount that was deducted.
2105	fn slash_payout(who: &T::AccountId, value: BalanceOf<T, I>) -> BalanceOf<T, I> {
2106		let mut record = Payouts::<T, I>::get(who);
2107		let mut rest = value;
2108		while !record.payouts.is_empty() {
2109			if let Some(new_rest) = rest.checked_sub(&record.payouts[0].1) {
2110				// not yet totally slashed after this one; drop it completely.
2111				rest = new_rest;
2112				record.payouts.remove(0);
2113			} else {
2114				// whole slash is accounted for.
2115				record.payouts[0].1.saturating_reduce(rest);
2116				rest = Zero::zero();
2117				break
2118			}
2119		}
2120		Payouts::<T, I>::insert(who, record);
2121		value - rest
2122	}
2123
2124	/// Transfer some `amount` from the main account into the payouts account and reduce the Pot
2125	/// by this amount.
2126	fn reserve_payout(amount: BalanceOf<T, I>) {
2127		// Transfer payout from the Pot into the payouts account.
2128		Pot::<T, I>::mutate(|pot| pot.saturating_reduce(amount));
2129
2130		// this should never fail since we ensure we can afford the payouts in a previous
2131		// block, but there's not much we can do to recover if it fails anyway.
2132		let res = T::Currency::transfer(&Self::account_id(), &Self::payouts(), amount, AllowDeath);
2133		debug_assert!(res.is_ok());
2134	}
2135
2136	/// Transfer some `amount` from the main account into the payouts account and increase the Pot
2137	/// by this amount.
2138	fn unreserve_payout(amount: BalanceOf<T, I>) {
2139		// Transfer payout from the Pot into the payouts account.
2140		Pot::<T, I>::mutate(|pot| pot.saturating_accrue(amount));
2141
2142		// this should never fail since we ensure we can afford the payouts in a previous
2143		// block, but there's not much we can do to recover if it fails anyway.
2144		let res = T::Currency::transfer(&Self::payouts(), &Self::account_id(), amount, AllowDeath);
2145		debug_assert!(res.is_ok());
2146	}
2147
2148	/// The account ID of the treasury pot.
2149	///
2150	/// This actually does computation. If you need to keep using it, then make sure you cache the
2151	/// value and only call this once.
2152	pub fn account_id() -> T::AccountId {
2153		T::PalletId::get().into_account_truncating()
2154	}
2155
2156	/// The account ID of the payouts pot. This is where payouts are made from.
2157	///
2158	/// This actually does computation. If you need to keep using it, then make sure you cache the
2159	/// value and only call this once.
2160	pub fn payouts() -> T::AccountId {
2161		T::PalletId::get().into_sub_account_truncating(b"payouts")
2162	}
2163
2164	/// Return the duration of the lock, in blocks, with the given number of members.
2165	///
2166	/// This is a rather opaque calculation based on the formula here:
2167	/// https://www.desmos.com/calculator/9itkal1tce
2168	fn lock_duration(x: u32) -> BlockNumberFor<T, I> {
2169		let lock_pc = 100 - 50_000 / (x + 500);
2170		Percent::from_percent(lock_pc as u8) * T::MaxLockDuration::get()
2171	}
2172}
2173
2174impl<T: Config<I>, I: 'static> OnUnbalanced<NegativeImbalanceOf<T, I>> for Pallet<T, I> {
2175	fn on_nonzero_unbalanced(amount: NegativeImbalanceOf<T, I>) {
2176		let numeric_amount = amount.peek();
2177
2178		// Must resolve into existing but better to be safe.
2179		let _ = T::Currency::resolve_creating(&Self::account_id(), amount);
2180
2181		Self::deposit_event(Event::<T, I>::Deposit { value: numeric_amount });
2182	}
2183}