lib.rs 39 KiB
Newer Older
Gav Wood's avatar
Gav Wood committed
// Copyright 2017 Parity Technologies (UK) Ltd.
// This file is part of Substrate Demo.

// Substrate Demo is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.

// Substrate Demo is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.

// You should have received a copy of the GNU General Public License
// along with Substrate Demo.  If not, see <http://www.gnu.org/licenses/>.

//! Staking manager: Handles balances and periodically determines the best set of validators.

#![cfg_attr(not(feature = "std"), no_std)]

#[cfg(feature = "std")]
extern crate serde;

Gav Wood's avatar
Gav Wood committed
#[cfg(feature = "std")]
#[macro_use]
extern crate serde_derive;

Gav Wood's avatar
Gav Wood committed
#[macro_use]
extern crate substrate_runtime_support as runtime_support;

#[cfg_attr(feature = "std", macro_use)]
extern crate substrate_runtime_std as rstd;

Gav Wood's avatar
Gav Wood committed
#[macro_use]
extern crate substrate_codec_derive;

Gav Wood's avatar
Gav Wood committed
extern crate substrate_codec as codec;
extern crate substrate_primitives;
extern crate substrate_runtime_io as runtime_io;
extern crate substrate_runtime_primitives as primitives;
extern crate substrate_runtime_consensus as consensus;
extern crate substrate_runtime_sandbox as sandbox;
Gav Wood's avatar
Gav Wood committed
extern crate substrate_runtime_session as session;
extern crate substrate_runtime_system as system;
extern crate substrate_runtime_timestamp as timestamp;
Gav Wood's avatar
Gav Wood committed

#[cfg(test)] use std::fmt::Debug;
use rstd::prelude::*;
use rstd::{cmp, result};
use codec::{Encode, Decode, Codec, Input, Output};
Gav Wood's avatar
Gav Wood committed
use runtime_support::{StorageValue, StorageMap, Parameter};
use runtime_support::dispatch::Result;
use session::OnSessionChange;
Gav Wood's avatar
Gav Wood committed
use primitives::traits::{Zero, One, Bounded, RefInto, SimpleArithmetic, Executable, MakePayment,
Gav Wood's avatar
Gav Wood committed
	As, AuxLookup, Member, CheckedAdd, CheckedSub, MaybeEmpty};
Gav Wood's avatar
Gav Wood committed
use address::Address as RawAddress;

mod mock;
Gav Wood's avatar
Gav Wood committed
mod tests;
mod genesis_config;

#[cfg(feature = "std")]
pub use genesis_config::GenesisConfig;

Gav Wood's avatar
Gav Wood committed
const DEFAULT_MINIMUM_VALIDATOR_COUNT: usize = 4;

Gav Wood's avatar
Gav Wood committed
/// Number of account IDs stored per enum set.
const ENUM_SET_SIZE: usize = 64;

/// The byte to identify intention to reclaim an existing account index.
const RECLAIM_INDEX_MAGIC: usize = 0x69;

pub type Address<T> = RawAddress<<T as system::Trait>::AccountId, <T as Trait>::AccountIndex>;
Gav Wood's avatar
Gav Wood committed

pub type Event<T> = RawEvent<
	<T as Trait>::Balance,
	<T as system::Trait>::AccountId,
	<T as Trait>::AccountIndex
>;

Gav Wood's avatar
Gav Wood committed
#[cfg(test)]
#[derive(Debug, PartialEq, Clone)]
pub enum LockStatus<BlockNumber: Debug + PartialEq + Clone> {
	Liquid,
	LockedUntil(BlockNumber),
	Staked,
}

#[cfg(not(test))]
#[derive(PartialEq, Clone)]
pub enum LockStatus<BlockNumber: PartialEq + Clone> {
	Liquid,
	LockedUntil(BlockNumber),
	Staked,
}

/// The account was the given id was killed.
pub trait OnFreeBalanceZero<AccountId> {
	/// The account was the given id was killed.
	fn on_free_balance_zero(who: &AccountId);
impl<AccountId> OnFreeBalanceZero<AccountId> for () {
	fn on_free_balance_zero(_who: &AccountId) {}
Gav Wood's avatar
Gav Wood committed
/// Preference of what happens on a slash event.
#[cfg_attr(feature = "std", derive(Debug, Serialize, Deserialize))]
#[derive(Encode, Decode, Eq, PartialEq, Clone, Copy)]
pub struct SlashPreference {
	/// Validator should ensure this many more slashes than is necessary before being unstaked.
	pub unstake_threshold: u32,
}

impl Default for SlashPreference {
	fn default() -> Self {
		SlashPreference {
			unstake_threshold: 3,
		}
	}
}

Gav Wood's avatar
Gav Wood committed
pub trait Trait: system::Trait + session::Trait {
Gav Wood's avatar
Gav Wood committed
	/// The allowed extrinsic position for `missed_proposal` inherent.
	const NOTE_MISSED_PROPOSAL_POSITION: u32;
Gav Wood's avatar
Gav Wood committed
	/// The balance of an account.
	type Balance: Parameter + SimpleArithmetic + Codec + Default + Copy + As<Self::AccountIndex> + As<usize> + As<u64>;
Gav Wood's avatar
Gav Wood committed
	/// Type used for storing an account's index; implies the maximum number of accounts the system
	/// can hold.
	type AccountIndex: Parameter + Member + Codec + SimpleArithmetic + As<u8> + As<u16> + As<u32> + As<u64> + As<usize> + Copy;
	/// A function which is invoked when the free-balance has fallen below the existential deposit and
	/// has been reduced to zero.
	///
	/// Gives a chance to clean up resources associated with the given account.
	type OnFreeBalanceZero: OnFreeBalanceZero<Self::AccountId>;

	/// The overarching event type. 
	type Event: From<Event<Self>> + Into<<Self as system::Trait>::Event>;
Gav Wood's avatar
Gav Wood committed
}

decl_module! {
	pub struct Module<T: Trait>;
Gav Wood's avatar
Gav Wood committed

	#[cfg_attr(feature = "std", derive(Serialize, Deserialize))]
Gav Wood's avatar
Gav Wood committed
	pub enum Call where aux: T::PublicAux {
Gav Wood's avatar
Gav Wood committed
		fn transfer(aux, dest: RawAddress<T::AccountId, T::AccountIndex>, value: T::Balance) -> Result = 0;
		fn stake(aux) -> Result = 1;
Gav Wood's avatar
Gav Wood committed
		fn unstake(aux, intentions_index: u32) -> Result = 2;
Gav Wood's avatar
Gav Wood committed
		fn nominate(aux, target: RawAddress<T::AccountId, T::AccountIndex>) -> Result = 3;
		fn unnominate(aux, target_index: u32) -> Result = 4;
Gav Wood's avatar
Gav Wood committed
		fn register_slash_preference(aux, intentions_index: u32, p: SlashPreference) -> Result = 5;
		fn note_missed_proposal(aux, offline_val_indices: Vec<u32>) -> Result = 6;
Gav Wood's avatar
Gav Wood committed
	}
Gav Wood's avatar
Gav Wood committed

	#[cfg_attr(feature = "std", derive(Serialize, Deserialize))]
Gav Wood's avatar
Gav Wood committed
	pub enum PrivCall {
		fn set_sessions_per_era(new: T::BlockNumber) -> Result = 0;
		fn set_bonding_duration(new: T::BlockNumber) -> Result = 1;
		fn set_validator_count(new: u32) -> Result = 2;
Gav Wood's avatar
Gav Wood committed
		fn force_new_era(apply_rewards: bool) -> Result = 3;
		fn set_offline_slash_grace(new: u32) -> Result = 4;
		fn set_balance(who: RawAddress<T::AccountId, T::AccountIndex>, free: T::Balance, reserved: T::Balance) -> Result = 5;
/// An event in this module.
#[cfg_attr(feature = "std", derive(Serialize, Deserialize, Debug))]
#[derive(Encode, Decode, PartialEq, Eq, Clone)]
pub enum RawEvent<Balance, AccountId, AccountIndex> {
	/// All validators have been rewarded by the given balance.
	Reward(Balance),
	/// One validator (and their nominators) has been given a offline-warning (they're still within
	/// their grace). The accrued number of slashes is recorded, too.
	OfflineWarning(AccountId, u32),
	/// One validator (and their nominators) has been slashed by the given amount.
	OfflineSlash(AccountId, Balance),
	/// A new account was created.
	NewAccount(AccountId, AccountIndex, NewAccountOutcome),
	/// An account was reaped.
	ReapedAccount(AccountId),
}
impl<B, A, I> From<RawEvent<B, A, I>> for () {
	fn from(_: RawEvent<B, A, I>) -> () { () }
}

Gav Wood's avatar
Gav Wood committed
decl_storage! {
	trait Store for Module<T: Trait> as Staking {

		// The length of the bonding duration in eras.
		pub BondingDuration get(bonding_duration): required T::BlockNumber;
		// The ideal number of staking participants.
		pub ValidatorCount get(validator_count): required u32;
		// Minimum number of staking participants before emergency conditions are imposed.
		pub MinimumValidatorCount: u32;
		// The length of a staking era in sessions.
		pub SessionsPerEra get(sessions_per_era): required T::BlockNumber;
		// The total amount of stake on the system.
		// TODO: this doesn't actually track total stake yet - it should do.
		pub TotalStake get(total_stake): required T::Balance;
		// The fee to be paid for making a transaction; the base.
		pub TransactionBaseFee get(transaction_base_fee): required T::Balance;
		// The fee to be paid for making a transaction; the per-byte portion.
		pub TransactionByteFee get(transaction_byte_fee): required T::Balance;
		// The minimum amount allowed to keep an account open.
		pub ExistentialDeposit get(existential_deposit): required T::Balance;
		// The amount credited to a destination's account whose index was reclaimed.
		pub ReclaimRebate get(reclaim_rebate): required T::Balance;
		// The fee required to make a transfer.
		pub TransferFee get(transfer_fee): required T::Balance;
		// The fee required to create an account. At least as big as ReclaimRebate.
		pub CreationFee get(creation_fee): required T::Balance;
		// Maximum reward, per validator, that is provided per acceptable session.
		pub SessionReward get(session_reward): required T::Balance;
		// Slash, per validator that is taken per abnormal era end.
		pub EarlyEraSlash get(early_era_slash): required T::Balance;
		// Number of instances of offline reports before slashing begins for validators.
		pub OfflineSlashGrace get(offline_slash_grace): default u32;

		// The current era index.
		pub CurrentEra get(current_era): required T::BlockNumber;
		// Preference over how many times the validator should get slashed for being offline before they are automatically unstaked.
		pub SlashPreferenceOf get(slash_preference_of): default map [ T::AccountId => SlashPreference ];
		// All the accounts with a desire to stake.
		pub Intentions get(intentions): default Vec<T::AccountId>;
		// All nominator -> nominee relationships.
		pub Nominating get(nominating): map [ T::AccountId => T::AccountId ];
		// Nominators for a particular account.
		pub NominatorsFor get(nominators_for): default map [ T::AccountId => Vec<T::AccountId> ];
		// Nominators for a particular account that is in action right now.
		pub CurrentNominatorsFor get(current_nominators_for): default map [ T::AccountId => Vec<T::AccountId> ];
		// The next value of sessions per era.
		pub NextSessionsPerEra get(next_sessions_per_era): T::BlockNumber;
		// The session index at which the era length last changed.
		pub LastEraLengthChange get(last_era_length_change): default T::BlockNumber;
		// The current era stake threshold
		pub StakeThreshold get(stake_threshold): required T::Balance;

		// The number of times a given validator has been reported offline. This gets decremented by one each era that passes.
		pub SlashCount get(slash_count): default map [ T::AccountId => u32 ];

		// The next free enumeration set.
		pub NextEnumSet get(next_enum_set): required T::AccountIndex;
		// The enumeration sets.
		pub EnumSet get(enum_set): default map [ T::AccountIndex => Vec<T::AccountId> ];

		// We are forcing a new era.
		pub ForcingNewEra get(forcing_new_era): ();

		// The "free" balance of a given account.
		//
		// This is the only balance that matters in terms of most operations on tokens. It is
		// alone used to determine the balance when in the contract execution environment. When this
		// balance falls below the value of `ExistentialDeposit`, then the "current account" is
		// deleted: specifically, `Bondage` and `FreeBalance`. Furthermore, `OnFreeBalanceZero` callback
		// is invoked, giving a chance to external modules to cleanup data associated with
		// the deleted account.
		//
		// `system::AccountNonce` is also deleted if `ReservedBalance` is also zero (it also gets
		// collapsed to zero if it ever becomes less than `ExistentialDeposit`.
		pub FreeBalance get(free_balance): default map [ T::AccountId => T::Balance ];

		// The amount of the balance of a given account that is exterally reserved; this can still get
		// slashed, but gets slashed last of all.
		//
		// This balance is a "reserve" balance that other subsystems use in order to set aside tokens
		// that are still "owned" by the account holder, but which are unspendable. This is different
		// and wholly unrelated to the `Bondage` system used for staking.
		//
		// When this balance falls below the value of `ExistentialDeposit`, then this "reserve account"
		// is deleted: specifically, `ReservedBalance`.
		//
		// `system::AccountNonce` is also deleted if `FreeBalance` is also zero (it also gets
		// collapsed to zero if it ever becomes less than `ExistentialDeposit`.
		pub ReservedBalance get(reserved_balance): default map [ T::AccountId => T::Balance ];

		// The block at which the `who`'s funds become entirely liquid.
		pub Bondage get(bondage): default map [ T::AccountId => T::BlockNumber ];
	}
/// Whatever happened about the hint given when creating the new account.
#[cfg_attr(feature = "std", derive(Serialize, Deserialize, Debug))]
#[derive(Encode, Decode, PartialEq, Eq, Clone, Copy)]
pub enum NewAccountOutcome {
Gav Wood's avatar
Gav Wood committed
	NoHint,
	GoodHint,
	BadHint,
/// Outcome of a balance update.
pub enum UpdateBalanceOutcome {
	/// Account balance was simply updated.
	Updated,
	/// The update has led to killing of the account.
	AccountKilled,
}

Gav Wood's avatar
Gav Wood committed
impl<T: Trait> Module<T> {

	// PUBLIC IMMUTABLES

Gav Wood's avatar
Gav Wood committed
	pub fn minimum_validator_count() -> usize {
		<MinimumValidatorCount<T>>::get().map(|v| v as usize).unwrap_or(DEFAULT_MINIMUM_VALIDATOR_COUNT)
	}

Gav Wood's avatar
Gav Wood committed
	/// The length of a staking era in blocks.
	pub fn era_length() -> T::BlockNumber {
		Self::sessions_per_era() * <session::Module<T>>::length()
	}

	/// The combined balance of `who`.
Gav Wood's avatar
Gav Wood committed
	pub fn voting_balance(who: &T::AccountId) -> T::Balance {
Gav Wood's avatar
Gav Wood committed
		Self::free_balance(who) + Self::reserved_balance(who)
	}

Gav Wood's avatar
Gav Wood committed
	/// Balance of a (potential) validator that includes all nominators.
	pub fn nomination_balance(who: &T::AccountId) -> T::Balance {
		Self::nominators_for(who).iter()
			.map(Self::voting_balance)
			.fold(Zero::zero(), |acc, x| acc + x)
	}

	/// The total balance that can be slashed from an account.
	pub fn slashable_balance(who: &T::AccountId) -> T::Balance {
		Self::nominators_for(who).iter()
			.map(Self::voting_balance)
			.fold(Self::voting_balance(who), |acc, x| acc + x)
	}

	/// Some result as `slash(who, value)` (but without the side-effects) assuming there are no
Gav Wood's avatar
Gav Wood committed
	/// balance changes in the meantime and only the reserved balance is not taken into account.
Gav Wood's avatar
Gav Wood committed
	pub fn can_slash(who: &T::AccountId, value: T::Balance) -> bool {
Gav Wood's avatar
Gav Wood committed
		Self::free_balance(who) >= value
Gav Wood's avatar
Gav Wood committed
	/// Same result as `reserve(who, value)` (but without the side-effects) assuming there
	/// are no balance changes in the meantime.
Gav Wood's avatar
Gav Wood committed
	pub fn can_reserve(who: &T::AccountId, value: T::Balance) -> bool {
		if let LockStatus::Liquid = Self::unlock_block(who) {
			Self::free_balance(who) >= value
		} else {
			false
		}
	}

Gav Wood's avatar
Gav Wood committed
	/// Lookup an T::AccountIndex to get an Id, if there's one there.
	pub fn lookup_index(index: T::AccountIndex) -> Option<T::AccountId> {
		let enum_set_size = Self::enum_set_size();
		let set = Self::enum_set(index / enum_set_size);
		let i: usize = (index % enum_set_size).as_();
		set.get(i).map(|x| x.clone())
	}

	/// `true` if the account `index` is ready for reclaim.
	pub fn can_reclaim(try_index: T::AccountIndex) -> bool {
		let enum_set_size = Self::enum_set_size();
		let try_set = Self::enum_set(try_index / enum_set_size);
		let i = (try_index % enum_set_size).as_();
		i < try_set.len() && Self::voting_balance(&try_set[i]).is_zero()
	}

Gav Wood's avatar
Gav Wood committed
	/// The block at which the `who`'s funds become entirely liquid.
	pub fn unlock_block(who: &T::AccountId) -> LockStatus<T::BlockNumber> {
		match Self::bondage(who) {
			i if i == T::BlockNumber::max_value() => LockStatus::Staked,
			i if i <= <system::Module<T>>::block_number() => LockStatus::Liquid,
			i => LockStatus::LockedUntil(i),
		}
	}

	/// Lookup an address to get an Id, if there's one there.
	pub fn lookup_address(a: address::Address<T::AccountId, T::AccountIndex>) -> Option<T::AccountId> {
		match a {
			address::Address::Id(i) => Some(i),
			address::Address::Index(i) => Self::lookup_index(i),
		}
	}

Gav Wood's avatar
Gav Wood committed
	// PUBLIC DISPATCH

	/// Transfer some unlocked staking balance to another staker.
	pub fn transfer(aux: &T::PublicAux, dest: Address<T>, value: T::Balance) -> Result {
Gav Wood's avatar
Gav Wood committed
		let dest = Self::lookup(dest)?;

		let transactor = aux.ref_into();
		let from_balance = Self::free_balance(transactor);
		let would_create = from_balance.is_zero();
		let fee = if would_create { Self::creation_fee() } else { Self::transfer_fee() };
		let liability = value + fee;

		let new_from_balance = match from_balance.checked_sub(&liability) {
			Some(b) => b,
			None => return Err("balance too low to send value"),
		};
		if would_create && value < Self::existential_deposit() {
			return Err("value too low to create account");
Gav Wood's avatar
Gav Wood committed
		}
		if <Bondage<T>>::get(transactor) > <system::Module<T>>::block_number() {
			return Err("bondage too high to send value");
		}

		let to_balance = Self::free_balance(&dest);
		let new_to_balance = match to_balance.checked_add(&value) {
			Some(b) => b,
			None => return Err("destination balance too high to receive value"),
		};

		if transactor != &dest {
			Self::set_free_balance(transactor, new_from_balance);			
			Self::decrease_total_stake_by(fee);
			Self::set_free_balance_creating(&dest, new_to_balance);
		}

Gav Wood's avatar
Gav Wood committed
	}

	/// Declare the desire to stake for the transactor.
	///
	/// Effects will be felt at the beginning of the next era.
	fn stake(aux: &T::PublicAux) -> Result {
Gav Wood's avatar
Gav Wood committed
		let aux = aux.ref_into();
		ensure!(Self::nominating(aux).is_none(), "Cannot stake if already nominating.");
Gav Wood's avatar
Gav Wood committed
		let mut intentions = <Intentions<T>>::get();
		// can't be in the list twice.
Gav Wood's avatar
Gav Wood committed
		ensure!(intentions.iter().find(|&t| t == aux).is_none(), "Cannot stake if already staked.");
		intentions.push(aux.clone());
Gav Wood's avatar
Gav Wood committed
		<Intentions<T>>::put(intentions);
Gav Wood's avatar
Gav Wood committed
		<Bondage<T>>::insert(aux, T::BlockNumber::max_value());
Gav Wood's avatar
Gav Wood committed
	}

	/// Retract the desire to stake for the transactor.
	///
	/// Effects will be felt at the beginning of the next era.
Gav Wood's avatar
Gav Wood committed
	fn unstake(aux: &T::PublicAux, intentions_index: u32) -> Result {
		// unstake fails in degenerate case of having too few existing staked parties
		if Self::intentions().len() <= Self::minimum_validator_count() {
			return Err("cannot unstake when there are too few staked participants")
Gav Wood's avatar
Gav Wood committed
		}
Gav Wood's avatar
Gav Wood committed
		Self::apply_unstake(aux.ref_into(), intentions_index as usize)
Gav Wood's avatar
Gav Wood committed
	fn nominate(aux: &T::PublicAux, target: RawAddress<T::AccountId, T::AccountIndex>) -> Result {
		let target = Self::lookup(target)?;
		let aux = aux.ref_into();

		ensure!(Self::nominating(aux).is_none(), "Cannot nominate if already nominating.");
		ensure!(Self::intentions().iter().find(|&t| t == aux.ref_into()).is_none(), "Cannot nominate if already staked.");

		// update nominators_for
		let mut t = Self::nominators_for(&target);
		t.push(aux.clone());
		<NominatorsFor<T>>::insert(&target, t);

		// update nominating
		<Nominating<T>>::insert(aux, &target);

		// Update bondage
		<Bondage<T>>::insert(aux.ref_into(), T::BlockNumber::max_value());

		Ok(())
	}

	/// Will panic if called when source isn't currently nominating target.
	/// Updates Nominating, NominatorsFor and NominationBalance.
	fn unnominate(aux: &T::PublicAux, target_index: u32) -> Result {
		let source = aux.ref_into();
		let target_index = target_index as usize;

		let target = <Nominating<T>>::get(source).ok_or("Account must be nominating")?;

		let mut t = Self::nominators_for(&target);
		if t.get(target_index) != Some(source) {
			return Err("Invalid target index")
		}

		// Ok - all valid.

		// update nominators_for
		t.swap_remove(target_index);
		<NominatorsFor<T>>::insert(&target, t);

		// update nominating
		<Nominating<T>>::remove(source);

		// update bondage
		<Bondage<T>>::insert(aux.ref_into(), Self::current_era() + Self::bonding_duration());
		Ok(())
	}

Gav Wood's avatar
Gav Wood committed
	/// Set the given account's preference for slashing behaviour should they be a validator. 
	/// 
	/// An error (no-op) if `Self::intentions()[intentions_index] != aux`.
	fn register_slash_preference(
		aux: &T::PublicAux,
		intentions_index: u32,
		p: SlashPreference
	) -> Result {
		let aux = aux.ref_into();

		if Self::intentions().get(intentions_index as usize) != Some(aux) {
			return Err("Invalid index")
		}
		
		<SlashPreferenceOf<T>>::insert(aux, p);

		Ok(())
	}

	/// Note the previous block's validator missed their opportunity to propose a block. This only comes in
	/// if 2/3+1 of the validators agree that no proposal was submitted. It's only relevant
	/// for the previous block.
	fn note_missed_proposal(aux: &T::PublicAux, offline_val_indices: Vec<u32>) -> Result {
		assert!(aux.is_empty());
		assert!(
			<system::Module<T>>::extrinsic_index() == Some(T::NOTE_MISSED_PROPOSAL_POSITION),
Gav Wood's avatar
Gav Wood committed
			"note_missed_proposal extrinsic must be at position {} in the block",
			T::NOTE_MISSED_PROPOSAL_POSITION
		);

		for validator_index in offline_val_indices.into_iter() {
			let v = <session::Module<T>>::validators()[validator_index as usize].clone();
			let slash_count = Self::slash_count(&v);
			<SlashCount<T>>::insert(v.clone(), slash_count + 1);
			let grace = Self::offline_slash_grace();

			let event = if slash_count >= grace {
Gav Wood's avatar
Gav Wood committed
				let instances = slash_count - grace;
				let slash = Self::early_era_slash() << instances;
				let next_slash = slash << 1u32;
				let _ = Self::slash_validator(&v, slash);
				if instances >= Self::slash_preference_of(&v).unstake_threshold
					|| Self::slashable_balance(&v) < next_slash
				{
					if let Some(pos) = Self::intentions().into_iter().position(|x| &x == &v) {
						Self::apply_unstake(&v, pos)
							.expect("pos derived correctly from Self::intentions(); \
								apply_unstake can only fail if pos wrong; \
								Self::intentions() doesn't change; qed");
					}
					let _ = Self::force_new_era(false);
				}
				RawEvent::OfflineSlash(v, slash)
			} else {
				RawEvent::OfflineWarning(v, slash_count)
			};
			Self::deposit_event(event);
Gav Wood's avatar
Gav Wood committed
	// PRIV DISPATCH

	/// Deposit one of this module's events.
	fn deposit_event(event: Event<T>) {
		<system::Module<T>>::deposit_event(<T as Trait>::Event::from(event).into());
	}

Gav Wood's avatar
Gav Wood committed
	/// Set the number of sessions in an era.
	fn set_sessions_per_era(new: T::BlockNumber) -> Result {
Gav Wood's avatar
Gav Wood committed
		<NextSessionsPerEra<T>>::put(&new);
Gav Wood's avatar
Gav Wood committed
	}

	/// The length of the bonding duration in eras.
	fn set_bonding_duration(new: T::BlockNumber) -> Result {
Gav Wood's avatar
Gav Wood committed
		<BondingDuration<T>>::put(&new);
Gav Wood's avatar
Gav Wood committed
	}

	/// The length of a staking era in sessions.
	fn set_validator_count(new: u32) -> Result {
Gav Wood's avatar
Gav Wood committed
		<ValidatorCount<T>>::put(&new);
Gav Wood's avatar
Gav Wood committed
	/// Force there to be a new era. This also forces a new session immediately after.
	/// `apply_rewards` should be true for validators to get the session reward.
	fn force_new_era(apply_rewards: bool) -> Result {
		<ForcingNewEra<T>>::put(());
Gav Wood's avatar
Gav Wood committed
		<session::Module<T>>::force_new_session(apply_rewards)
	}

	/// Set the offline slash grace period.
	fn set_offline_slash_grace(new: u32) -> Result {
		<OfflineSlashGrace<T>>::put(&new);
		Ok(())
	}

	/// Set the balances of a given account.
	fn set_balance(who: Address<T>, free: T::Balance, reserved: T::Balance) -> Result {
		let who = Self::lookup(who)?;
		Self::set_free_balance(&who, free);
		Self::set_reserved_balance(&who, reserved);
		Ok(())
Gav Wood's avatar
Gav Wood committed
	}

	// PUBLIC MUTABLES (DANGEROUS)

	/// Set the free balance of an account to some new value.
	///
	/// Will enforce ExistentialDeposit law, anulling the account as needed.
	/// In that case it will return `AccountKilled`.
	pub fn set_reserved_balance(who: &T::AccountId, balance: T::Balance) -> UpdateBalanceOutcome {
Gav Wood's avatar
Gav Wood committed
		if balance < Self::existential_deposit() {
			<ReservedBalance<T>>::insert(who, balance);
Gav Wood's avatar
Gav Wood committed
			Self::on_reserved_too_low(who);
			UpdateBalanceOutcome::AccountKilled
Gav Wood's avatar
Gav Wood committed
		} else {
			<ReservedBalance<T>>::insert(who, balance);
			UpdateBalanceOutcome::Updated
Gav Wood's avatar
Gav Wood committed
	/// Set the free balance of an account to some new value. Will enforce ExistentialDeposit
	/// law anulling the account as needed.
	///
	/// Doesn't do any preparatory work for creating a new account, so should only be used when it
	/// is known that the account already exists.
	///
	/// Returns if the account was successfully updated or update has led to killing of the account.
	pub fn set_free_balance(who: &T::AccountId, balance: T::Balance) -> UpdateBalanceOutcome {
Gav Wood's avatar
Gav Wood committed
		// Commented out for no - but consider it instructive.
		// assert!(!Self::voting_balance(who).is_zero());
Gav Wood's avatar
Gav Wood committed
		if balance < Self::existential_deposit() {
			<FreeBalance<T>>::insert(who, balance);
Gav Wood's avatar
Gav Wood committed
			Self::on_free_too_low(who);
			UpdateBalanceOutcome::AccountKilled
Gav Wood's avatar
Gav Wood committed
		} else {
			<FreeBalance<T>>::insert(who, balance);
			UpdateBalanceOutcome::Updated
		}
	}

	/// Set the free balance on an account to some new value.
	///
	/// Same as [`set_free_balance`], but will create a new account.
	///
	/// Returns if the account was successfully updated or update has led to killing of the account.
	///
	/// [`set_free_balance`]: #method.set_free_balance
	pub fn set_free_balance_creating(who: &T::AccountId, balance: T::Balance) -> UpdateBalanceOutcome {
		let ed = <Module<T>>::existential_deposit();
		// If the balance is too low, then the account is reaped.
		// NOTE: There are two balances for every account: `reserved_balance` and
		// `free_balance`. This contract subsystem only cares about the latter: whenever
		// the term "balance" is used *here* it should be assumed to mean "free balance"
		// in the rest of the module.
		// Free balance can never be less than ED. If that happens, it gets reduced to zero
		// and the account information relevant to this subsystem is deleted (i.e. the
		// account is reaped).
		// NOTE: This is orthogonal to the `Bondage` value that an account has, a high
		// value of which makes even the `free_balance` unspendable.
		// TODO: enforce this for the other balance-altering functions.
		if balance < ed {
			Self::set_free_balance(who, balance);
			UpdateBalanceOutcome::AccountKilled
		} else {
			if !<FreeBalance<T>>::exists(who) {
				let outcome = Self::new_account(&who, balance);
				let credit = match outcome {
					NewAccountOutcome::GoodHint => balance + <Module<T>>::reclaim_rebate(),
					_ => balance,
				};
				Self::set_free_balance(who, credit);
				Self::increase_total_stake_by(credit - balance);
			} else {	
				Self::set_free_balance(who, balance);
			UpdateBalanceOutcome::Updated
Gav Wood's avatar
Gav Wood committed
	/// Deducts up to `value` from the combined balance of `who`, preferring to deduct from the
	/// free balance. This function cannot fail.
	///
	/// As much funds up to `value` will be deducted as possible. If this is less than `value`,
	/// then `Some(remaining)` will be returned. Full completion is given by `None`.
Gav Wood's avatar
Gav Wood committed
	pub fn slash(who: &T::AccountId, value: T::Balance) -> Option<T::Balance> {
Gav Wood's avatar
Gav Wood committed
		let free_balance = Self::free_balance(who);
		let free_slash = cmp::min(free_balance, value);
Gav Wood's avatar
Gav Wood committed
		Self::set_free_balance(who, free_balance - free_slash);
		Self::decrease_total_stake_by(free_slash);
Gav Wood's avatar
Gav Wood committed
		if free_slash < value {
			Self::slash_reserved(who, value - free_slash)
		} else {
Gav Wood's avatar
Gav Wood committed
			None
	/// Adds up to `value` to the free balance of `who`.
	///
	/// If `who` doesn't exist, nothing is done and an Err returned.
	pub fn reward(who: &T::AccountId, value: T::Balance) -> Result {
		if Self::voting_balance(who).is_zero() {
			return Err("beneficiary account must pre-exist");
		}
		Self::set_free_balance(who, Self::free_balance(who) + value);
		Self::increase_total_stake_by(value);
Gav Wood's avatar
Gav Wood committed
	/// Moves `value` from balance to reserved balance.
Gav Wood's avatar
Gav Wood committed
	///
	/// If the free balance is lower than `value`, then no funds will be moved and an `Err` will
	/// be returned to notify of this. This is different behaviour to `unreserve`.
	pub fn reserve(who: &T::AccountId, value: T::Balance) -> Result {
Gav Wood's avatar
Gav Wood committed
		let b = Self::free_balance(who);
		if b < value {
			return Err("not enough free funds")
Gav Wood's avatar
Gav Wood committed
		if Self::unlock_block(who) != LockStatus::Liquid {
			return Err("free funds are still bonded")
		}
		Self::set_reserved_balance(who, Self::reserved_balance(who) + value);
		Self::set_free_balance(who, b - value);
Gav Wood's avatar
Gav Wood committed
	/// Moves up to `value` from reserved balance to balance. This function cannot fail.
	///
	/// As much funds up to `value` will be deducted as possible. If this is less than `value`,
	/// then `Some(remaining)` will be returned. Full completion is given by `None`.
Gav Wood's avatar
Gav Wood committed
	/// NOTE: This is different to `reserve`.
	pub fn unreserve(who: &T::AccountId, value: T::Balance) -> Option<T::Balance> {
Gav Wood's avatar
Gav Wood committed
		let b = Self::reserved_balance(who);
Gav Wood's avatar
Gav Wood committed
		let actual = cmp::min(b, value);
		Self::set_free_balance(who, Self::free_balance(who) + actual);
		Self::set_reserved_balance(who, b - actual);
		if actual == value {
			None
		} else {
			Some(value - actual)
		}
Gav Wood's avatar
Gav Wood committed
	/// Deducts up to `value` from reserved balance of `who`. This function cannot fail.
	///
	/// As much funds up to `value` will be deducted as possible. If this is less than `value`,
	/// then `Some(remaining)` will be returned. Full completion is given by `None`.
Gav Wood's avatar
Gav Wood committed
	pub fn slash_reserved(who: &T::AccountId, value: T::Balance) -> Option<T::Balance> {
Gav Wood's avatar
Gav Wood committed
		let b = Self::reserved_balance(who);
		let slash = cmp::min(b, value);
Gav Wood's avatar
Gav Wood committed
		Self::set_reserved_balance(who, b - slash);
		Self::decrease_total_stake_by(slash);
		if value == slash {
Gav Wood's avatar
Gav Wood committed
			None
Gav Wood's avatar
Gav Wood committed
			Some(value - slash)
Gav Wood's avatar
Gav Wood committed
	/// Moves up to `value` from reserved balance of account `slashed` to free balance of account
	/// `beneficiary`. `beneficiary` must exist for this to succeed. If it does not, `Err` will be
	/// returned.
	///
	/// As much funds up to `value` will be moved as possible. If this is less than `value`, then
	/// `Ok(Some(remaining))` will be returned. Full completion is given by `Ok(None)`.
Gav Wood's avatar
Gav Wood committed
	pub fn transfer_reserved(
		slashed: &T::AccountId,
		beneficiary: &T::AccountId,
		value: T::Balance
	) -> result::Result<Option<T::Balance>, &'static str> {
		if Self::voting_balance(beneficiary).is_zero() {
			return Err("beneficiary account must pre-exist");
		}
Gav Wood's avatar
Gav Wood committed
		let b = Self::reserved_balance(slashed);
		let slash = cmp::min(b, value);
Gav Wood's avatar
Gav Wood committed
		Self::set_free_balance(beneficiary, Self::free_balance(beneficiary) + slash);
		Self::set_reserved_balance(slashed, b - slash);
		if value == slash {
Gav Wood's avatar
Gav Wood committed
			Ok(None)
Gav Wood's avatar
Gav Wood committed
			Ok(Some(value - slash))
Gav Wood's avatar
Gav Wood committed
	/// Slash a given validator by a specific amount. Removes the slash from their balance by preference,
	/// and reduces the nominators' balance if needed.
	fn slash_validator(v: &T::AccountId, slash: T::Balance) {
		// skip the slash in degenerate case of having only 4 staking participants despite having a larger
		// desired number of validators (validator_count).
		if Self::intentions().len() <= Self::minimum_validator_count() {
			return
		}

		if let Some(rem) = Self::slash(v, slash) {
			let noms = Self::current_nominators_for(v);
			let total = noms.iter().map(Self::voting_balance).fold(T::Balance::zero(), |acc, x| acc + x);
			if !total.is_zero() {
				let safe_mul_rational = |b| b * rem / total;// TODO: avoid overflow
				for n in noms.iter() {
					let _ = Self::slash(n, safe_mul_rational(Self::voting_balance(n)));	// best effort - not much that can be done on fail.
Gav Wood's avatar
Gav Wood committed
	/// Reward a given validator by a specific amount. Add the reward to their, and their nominators'
	/// balance, pro-rata.
	fn reward_validator(who: &T::AccountId, reward: T::Balance) {
		let noms = Self::current_nominators_for(who);
		let total = noms.iter().map(Self::voting_balance).fold(Self::voting_balance(who), |acc, x| acc + x);
		if !total.is_zero() {
			let safe_mul_rational = |b| b * reward / total;// TODO: avoid overflow
			for n in noms.iter() {
				let _ = Self::reward(n, safe_mul_rational(Self::voting_balance(n)));
			}
			let _ = Self::reward(who, safe_mul_rational(Self::voting_balance(who)));
		}
	}

	/// Actually carry out the unstake operation.
	/// Assumes `intentions()[intentions_index] == who`.
	fn apply_unstake(who: &T::AccountId, intentions_index: usize) -> Result {
		let mut intentions = Self::intentions();
		if intentions.get(intentions_index) != Some(who) {
			return Err("Invalid index");
		}
		intentions.swap_remove(intentions_index);
		<Intentions<T>>::put(intentions);
		<SlashPreferenceOf<T>>::remove(who);
		<SlashCount<T>>::remove(who);
		<Bondage<T>>::insert(who, Self::current_era() + Self::bonding_duration());
		Ok(())
	}

	/// Get the reward for the session, assuming it ends with this block.
	fn this_session_reward(actual_elapsed: T::Moment) -> T::Balance {
		let ideal_elapsed = <session::Module<T>>::ideal_session_duration();
		let per65536: u64 = (T::Moment::sa(65536u64) * ideal_elapsed.clone() / actual_elapsed.max(ideal_elapsed)).as_();
		Self::session_reward() * T::Balance::sa(per65536) / T::Balance::sa(65536u64)
	}

	/// Session has just changed. We need to determine whether we pay a reward, slash and/or
	/// move to a new era.
	fn new_session(actual_elapsed: T::Moment, should_reward: bool) {
		if should_reward {
			// apply good session reward
Gav Wood's avatar
Gav Wood committed
			let reward = Self::this_session_reward(actual_elapsed);
			for v in <session::Module<T>>::validators().iter() {
Gav Wood's avatar
Gav Wood committed
				Self::reward_validator(v, reward);
			Self::deposit_event(RawEvent::Reward(reward));
Gav Wood's avatar
Gav Wood committed

		let session_index = <session::Module<T>>::current_index();
		if <ForcingNewEra<T>>::take().is_some()
			|| ((session_index - Self::last_era_length_change()) % Self::sessions_per_era()).is_zero()
		{
Gav Wood's avatar
Gav Wood committed
			Self::new_era();
		}
	}

	/// The era has changed - enact new staking set.
	///
	/// NOTE: This always happens immediately before a session change to ensure that new validators
	/// get a chance to set their session keys.
	fn new_era() {
		// Increment current era.
		<CurrentEra<T>>::put(&(<CurrentEra<T>>::get() + One::one()));

		// Enact era length change.
		if let Some(next_spe) = Self::next_sessions_per_era() {
			if next_spe != Self::sessions_per_era() {
				<SessionsPerEra<T>>::put(&next_spe);
				<LastEraLengthChange<T>>::put(&<session::Module<T>>::current_index());
Gav Wood's avatar
Gav Wood committed
			}
		}

		// evaluate desired staking amounts and nominations and optimise to find the best
		// combination of validators, then use session::internal::set_validators().
		// for now, this just orders would-be stakers by their balances and chooses the top-most
		// <ValidatorCount<T>>::get() of them.
Gav Wood's avatar
Gav Wood committed
		// TODO: this is not sound. this should be moved to an off-chain solution mechanism.
Gav Wood's avatar
Gav Wood committed
		let mut intentions = Self::intentions()
Gav Wood's avatar
Gav Wood committed
			.into_iter()
Gav Wood's avatar
Gav Wood committed
			.map(|v| (Self::slashable_balance(&v), v))
Gav Wood's avatar
Gav Wood committed
			.collect::<Vec<_>>();
Gav Wood's avatar
Gav Wood committed

		// Avoid reevaluate validator set if it would leave us with fewer than the minimum
		// needed validators
		if intentions.len() < Self::minimum_validator_count() {
			return
		}

Gav Wood's avatar
Gav Wood committed
		intentions.sort_unstable_by(|&(ref b1, _), &(ref b2, _)| b2.cmp(&b1));
Gav Wood's avatar
Gav Wood committed
		<StakeThreshold<T>>::put(
			if !intentions.is_empty() {
Gav Wood's avatar
Gav Wood committed
				let i = (<ValidatorCount<T>>::get() as usize).min(intentions.len() - 1);
				intentions[i].0.clone()
			} else { Zero::zero() }
		);
		let vals = &intentions.into_iter()
Gav Wood's avatar
Gav Wood committed
			.map(|(_, v)| v)
			.take(<ValidatorCount<T>>::get() as usize)
			.collect::<Vec<_>>();
Gav Wood's avatar
Gav Wood committed
		for v in <session::Module<T>>::validators().iter() {
			<CurrentNominatorsFor<T>>::remove(v);
Gav Wood's avatar
Gav Wood committed
			let slash_count = <SlashCount<T>>::take(v);
			if slash_count > 1 {
				<SlashCount<T>>::insert(v, slash_count - 1);
			}
Gav Wood's avatar
Gav Wood committed
		}
		for v in vals.iter() {
			<CurrentNominatorsFor<T>>::insert(v, Self::nominators_for(v));
		}
		<session::Module<T>>::set_validators(vals);
Gav Wood's avatar
Gav Wood committed
	}
Gav Wood's avatar
Gav Wood committed

	fn enum_set_size() -> T::AccountIndex {
		T::AccountIndex::sa(ENUM_SET_SIZE)
	}

	/// Register a new account (with existential balance).
	fn new_account(who: &T::AccountId, balance: T::Balance) -> NewAccountOutcome {
		let enum_set_size = Self::enum_set_size();
		let next_set_index = Self::next_enum_set();
		let reclaim_index_magic = T::AccountIndex::sa(RECLAIM_INDEX_MAGIC);
		let reclaim_index_modulus = T::AccountIndex::sa(256usize);
		let quantization = T::AccountIndex::sa(256usize);

		// A little easter-egg for reclaiming dead indexes..
		let ret = {
			// we quantise the number of accounts so it stays constant over a reasonable
			// period of time.
			let quantized_account_count: T::AccountIndex = (next_set_index * enum_set_size / quantization + One::one()) * quantization;
			// then modify the starting balance to be modulo this to allow it to potentially
			// identify an account index for reuse.
			let maybe_try_index = balance % <T::Balance as As<T::AccountIndex>>::sa(quantized_account_count * reclaim_index_modulus);
			let maybe_try_index = As::<T::AccountIndex>::as_(maybe_try_index);

			// this identifier must end with magic byte 0x69 to trigger this check (a minor
			// optimisation to ensure we don't check most unintended account creations).
			if maybe_try_index % reclaim_index_modulus == reclaim_index_magic {
				// reuse is probably intended. first, remove magic byte.
				let try_index = maybe_try_index / reclaim_index_modulus;

				// then check to see if this balance identifies a dead account index.
				let set_index = try_index / enum_set_size;
				let mut try_set = Self::enum_set(set_index);
				let item_index = (try_index % enum_set_size).as_();
				if item_index < try_set.len() {
					if Self::voting_balance(&try_set[item_index]).is_zero() {
						// yup - this index refers to a dead account. can be reused.
						try_set[item_index] = who.clone();
						<EnumSet<T>>::insert(set_index, try_set);

						Self::deposit_event(RawEvent::NewAccount(who.clone(), try_index, NewAccountOutcome::GoodHint));

						return NewAccountOutcome::GoodHint
Gav Wood's avatar
Gav Wood committed
					}
				}
				NewAccountOutcome::BadHint
			} else {
				NewAccountOutcome::NoHint
			}
		};

		// insert normally as a back up
		let mut set_index = next_set_index;
		// defensive only: this loop should never iterate since we keep NextEnumSet up to date later.
		let mut set = loop {
			let set = Self::enum_set(set_index);
			if set.len() < ENUM_SET_SIZE {
				break set;
			}
			set_index += One::one();
		};

		let index = T::AccountIndex::sa(set_index.as_() * ENUM_SET_SIZE + set.len());

Gav Wood's avatar
Gav Wood committed
		// update set.
		set.push(who.clone());

		// keep NextEnumSet up to date
		if set.len() == ENUM_SET_SIZE {
			<NextEnumSet<T>>::put(set_index + One::one());
		}

		// write set.
		<EnumSet<T>>::insert(set_index, set);

		Self::deposit_event(RawEvent::NewAccount(who.clone(), index, ret));

	fn reap_account(who: &T::AccountId) {
		<system::AccountNonce<T>>::remove(who);
		Self::deposit_event(RawEvent::ReapedAccount(who.clone()));