pallet_vesting/
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//! # Vesting Pallet
19//!
20//! - [`Config`]
21//! - [`Call`]
22//!
23//! ## Overview
24//!
25//! A simple pallet providing a means of placing a linear curve on an account's locked balance. This
26//! pallet ensures that there is a lock in place preventing the balance to drop below the *unvested*
27//! amount for any reason other than the ones specified in `UnvestedFundsAllowedWithdrawReasons`
28//! configuration value.
29//!
30//! As the amount vested increases over time, the amount unvested reduces. However, locks remain in
31//! place and explicit action is needed on behalf of the user to ensure that the amount locked is
32//! equivalent to the amount remaining to be vested. This is done through a dispatchable function,
33//! either `vest` (in typical case where the sender is calling on their own behalf) or `vest_other`
34//! in case the sender is calling on another account's behalf.
35//!
36//! ## Interface
37//!
38//! This pallet implements the `VestingSchedule` trait.
39//!
40//! ### Dispatchable Functions
41//!
42//! - `vest` - Update the lock, reducing it in line with the amount "vested" so far.
43//! - `vest_other` - Update the lock of another account, reducing it in line with the amount
44//!   "vested" so far.
45
46#![cfg_attr(not(feature = "std"), no_std)]
47
48mod benchmarking;
49
50#[cfg(test)]
51mod mock;
52#[cfg(test)]
53mod tests;
54mod vesting_info;
55
56pub mod migrations;
57pub mod weights;
58
59extern crate alloc;
60
61use alloc::vec::Vec;
62use codec::{Decode, DecodeWithMemTracking, Encode, MaxEncodedLen};
63use core::{fmt::Debug, marker::PhantomData};
64use frame_support::{
65	dispatch::DispatchResult,
66	ensure,
67	storage::bounded_vec::BoundedVec,
68	traits::{
69		Currency, ExistenceRequirement, Get, LockIdentifier, LockableCurrency, VestedTransfer,
70		VestingSchedule, WithdrawReasons,
71	},
72	weights::Weight,
73};
74use frame_system::pallet_prelude::BlockNumberFor;
75use scale_info::TypeInfo;
76use sp_runtime::{
77	traits::{
78		AtLeast32BitUnsigned, BlockNumberProvider, Bounded, Convert, MaybeSerializeDeserialize,
79		One, Saturating, StaticLookup, Zero,
80	},
81	DispatchError, RuntimeDebug,
82};
83
84pub use pallet::*;
85pub use vesting_info::*;
86pub use weights::WeightInfo;
87
88type BalanceOf<T> =
89	<<T as Config>::Currency as Currency<<T as frame_system::Config>::AccountId>>::Balance;
90type MaxLocksOf<T> =
91	<<T as Config>::Currency as LockableCurrency<<T as frame_system::Config>::AccountId>>::MaxLocks;
92type AccountIdLookupOf<T> = <<T as frame_system::Config>::Lookup as StaticLookup>::Source;
93
94const VESTING_ID: LockIdentifier = *b"vesting ";
95
96// A value placed in storage that represents the current version of the Vesting storage.
97// This value is used by `on_runtime_upgrade` to determine whether we run storage migration logic.
98#[derive(Encode, Decode, Clone, Copy, PartialEq, Eq, RuntimeDebug, MaxEncodedLen, TypeInfo)]
99pub enum Releases {
100	V0,
101	V1,
102}
103
104impl Default for Releases {
105	fn default() -> Self {
106		Releases::V0
107	}
108}
109
110/// Actions to take against a user's `Vesting` storage entry.
111#[derive(Clone, Copy)]
112enum VestingAction {
113	/// Do not actively remove any schedules.
114	Passive,
115	/// Remove the schedule specified by the index.
116	Remove { index: usize },
117	/// Remove the two schedules, specified by index, so they can be merged.
118	Merge { index1: usize, index2: usize },
119}
120
121impl VestingAction {
122	/// Whether or not the filter says the schedule index should be removed.
123	fn should_remove(&self, index: usize) -> bool {
124		match self {
125			Self::Passive => false,
126			Self::Remove { index: index1 } => *index1 == index,
127			Self::Merge { index1, index2 } => *index1 == index || *index2 == index,
128		}
129	}
130
131	/// Pick the schedules that this action dictates should continue vesting undisturbed.
132	fn pick_schedules<T: Config>(
133		&self,
134		schedules: Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>,
135	) -> impl Iterator<Item = VestingInfo<BalanceOf<T>, BlockNumberFor<T>>> + '_ {
136		schedules.into_iter().enumerate().filter_map(move |(index, schedule)| {
137			if self.should_remove(index) {
138				None
139			} else {
140				Some(schedule)
141			}
142		})
143	}
144}
145
146// Wrapper for `T::MAX_VESTING_SCHEDULES` to satisfy `trait Get`.
147pub struct MaxVestingSchedulesGet<T>(PhantomData<T>);
148impl<T: Config> Get<u32> for MaxVestingSchedulesGet<T> {
149	fn get() -> u32 {
150		T::MAX_VESTING_SCHEDULES
151	}
152}
153
154#[frame_support::pallet]
155pub mod pallet {
156	use super::*;
157	use frame_support::pallet_prelude::*;
158	use frame_system::pallet_prelude::*;
159
160	#[pallet::config]
161	pub trait Config: frame_system::Config {
162		/// The overarching event type.
163		type RuntimeEvent: From<Event<Self>> + IsType<<Self as frame_system::Config>::RuntimeEvent>;
164
165		/// The currency trait.
166		type Currency: LockableCurrency<Self::AccountId>;
167
168		/// Convert the block number into a balance.
169		type BlockNumberToBalance: Convert<BlockNumberFor<Self>, BalanceOf<Self>>;
170
171		/// The minimum amount transferred to call `vested_transfer`.
172		#[pallet::constant]
173		type MinVestedTransfer: Get<BalanceOf<Self>>;
174
175		/// Weight information for extrinsics in this pallet.
176		type WeightInfo: WeightInfo;
177
178		/// Reasons that determine under which conditions the balance may drop below
179		/// the unvested amount.
180		type UnvestedFundsAllowedWithdrawReasons: Get<WithdrawReasons>;
181
182		/// Query the current block number.
183		///
184		/// Must return monotonically increasing values when called from consecutive blocks.
185		/// Can be configured to return either:
186		/// - the local block number of the runtime via `frame_system::Pallet`
187		/// - a remote block number, eg from the relay chain through `RelaychainDataProvider`
188		/// - an arbitrary value through a custom implementation of the trait
189		///
190		/// There is currently no migration provided to "hot-swap" block number providers and it may
191		/// result in undefined behavior when doing so. Parachains are therefore best off setting
192		/// this to their local block number provider if they have the pallet already deployed.
193		///
194		/// Suggested values:
195		/// - Solo- and Relay-chains: `frame_system::Pallet`
196		/// - Parachains that may produce blocks sparingly or only when needed (on-demand):
197		///   - already have the pallet deployed: `frame_system::Pallet`
198		///   - are freshly deploying this pallet: `RelaychainDataProvider`
199		/// - Parachains with a reliably block production rate (PLO or bulk-coretime):
200		///   - already have the pallet deployed: `frame_system::Pallet`
201		///   - are freshly deploying this pallet: no strong recommendation. Both local and remote
202		///     providers can be used. Relay provider can be a bit better in cases where the
203		///     parachain is lagging its block production to avoid clock skew.
204		type BlockNumberProvider: BlockNumberProvider<BlockNumber = BlockNumberFor<Self>>;
205
206		/// Maximum number of vesting schedules an account may have at a given moment.
207		const MAX_VESTING_SCHEDULES: u32;
208	}
209
210	#[pallet::extra_constants]
211	impl<T: Config> Pallet<T> {
212		#[pallet::constant_name(MaxVestingSchedules)]
213		fn max_vesting_schedules() -> u32 {
214			T::MAX_VESTING_SCHEDULES
215		}
216	}
217
218	#[pallet::hooks]
219	impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
220		fn integrity_test() {
221			assert!(T::MAX_VESTING_SCHEDULES > 0, "`MaxVestingSchedules` must ge greater than 0");
222		}
223	}
224
225	/// Information regarding the vesting of a given account.
226	#[pallet::storage]
227	pub type Vesting<T: Config> = StorageMap<
228		_,
229		Blake2_128Concat,
230		T::AccountId,
231		BoundedVec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>, MaxVestingSchedulesGet<T>>,
232	>;
233
234	/// Storage version of the pallet.
235	///
236	/// New networks start with latest version, as determined by the genesis build.
237	#[pallet::storage]
238	pub type StorageVersion<T: Config> = StorageValue<_, Releases, ValueQuery>;
239
240	#[pallet::pallet]
241	pub struct Pallet<T>(_);
242
243	#[pallet::genesis_config]
244	#[derive(frame_support::DefaultNoBound)]
245	pub struct GenesisConfig<T: Config> {
246		pub vesting: Vec<(T::AccountId, BlockNumberFor<T>, BlockNumberFor<T>, BalanceOf<T>)>,
247	}
248
249	#[pallet::genesis_build]
250	impl<T: Config> BuildGenesisConfig for GenesisConfig<T> {
251		fn build(&self) {
252			use sp_runtime::traits::Saturating;
253
254			// Genesis uses the latest storage version.
255			StorageVersion::<T>::put(Releases::V1);
256
257			// Generate initial vesting configuration
258			// * who - Account which we are generating vesting configuration for
259			// * begin - Block when the account will start to vest
260			// * length - Number of blocks from `begin` until fully vested
261			// * liquid - Number of units which can be spent before vesting begins
262			for &(ref who, begin, length, liquid) in self.vesting.iter() {
263				let balance = T::Currency::free_balance(who);
264				assert!(!balance.is_zero(), "Currencies must be init'd before vesting");
265				// Total genesis `balance` minus `liquid` equals funds locked for vesting
266				let locked = balance.saturating_sub(liquid);
267				let length_as_balance = T::BlockNumberToBalance::convert(length);
268				let per_block = locked / length_as_balance.max(sp_runtime::traits::One::one());
269				let vesting_info = VestingInfo::new(locked, per_block, begin);
270				if !vesting_info.is_valid() {
271					panic!("Invalid VestingInfo params at genesis")
272				};
273
274				Vesting::<T>::try_append(who, vesting_info)
275					.expect("Too many vesting schedules at genesis.");
276
277				let reasons =
278					WithdrawReasons::except(T::UnvestedFundsAllowedWithdrawReasons::get());
279
280				T::Currency::set_lock(VESTING_ID, who, locked, reasons);
281			}
282		}
283	}
284
285	#[pallet::event]
286	#[pallet::generate_deposit(pub(super) fn deposit_event)]
287	pub enum Event<T: Config> {
288		/// The amount vested has been updated. This could indicate a change in funds available.
289		/// The balance given is the amount which is left unvested (and thus locked).
290		VestingUpdated { account: T::AccountId, unvested: BalanceOf<T> },
291		/// An \[account\] has become fully vested.
292		VestingCompleted { account: T::AccountId },
293	}
294
295	/// Error for the vesting pallet.
296	#[pallet::error]
297	pub enum Error<T> {
298		/// The account given is not vesting.
299		NotVesting,
300		/// The account already has `MaxVestingSchedules` count of schedules and thus
301		/// cannot add another one. Consider merging existing schedules in order to add another.
302		AtMaxVestingSchedules,
303		/// Amount being transferred is too low to create a vesting schedule.
304		AmountLow,
305		/// An index was out of bounds of the vesting schedules.
306		ScheduleIndexOutOfBounds,
307		/// Failed to create a new schedule because some parameter was invalid.
308		InvalidScheduleParams,
309	}
310
311	#[pallet::call]
312	impl<T: Config> Pallet<T> {
313		/// Unlock any vested funds of the sender account.
314		///
315		/// The dispatch origin for this call must be _Signed_ and the sender must have funds still
316		/// locked under this pallet.
317		///
318		/// Emits either `VestingCompleted` or `VestingUpdated`.
319		///
320		/// ## Complexity
321		/// - `O(1)`.
322		#[pallet::call_index(0)]
323		#[pallet::weight(T::WeightInfo::vest_locked(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
324			.max(T::WeightInfo::vest_unlocked(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES))
325		)]
326		pub fn vest(origin: OriginFor<T>) -> DispatchResult {
327			let who = ensure_signed(origin)?;
328			Self::do_vest(who)
329		}
330
331		/// Unlock any vested funds of a `target` account.
332		///
333		/// The dispatch origin for this call must be _Signed_.
334		///
335		/// - `target`: The account whose vested funds should be unlocked. Must have funds still
336		/// locked under this pallet.
337		///
338		/// Emits either `VestingCompleted` or `VestingUpdated`.
339		///
340		/// ## Complexity
341		/// - `O(1)`.
342		#[pallet::call_index(1)]
343		#[pallet::weight(T::WeightInfo::vest_other_locked(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
344			.max(T::WeightInfo::vest_other_unlocked(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES))
345		)]
346		pub fn vest_other(origin: OriginFor<T>, target: AccountIdLookupOf<T>) -> DispatchResult {
347			ensure_signed(origin)?;
348			let who = T::Lookup::lookup(target)?;
349			Self::do_vest(who)
350		}
351
352		/// Create a vested transfer.
353		///
354		/// The dispatch origin for this call must be _Signed_.
355		///
356		/// - `target`: The account receiving the vested funds.
357		/// - `schedule`: The vesting schedule attached to the transfer.
358		///
359		/// Emits `VestingCreated`.
360		///
361		/// NOTE: This will unlock all schedules through the current block.
362		///
363		/// ## Complexity
364		/// - `O(1)`.
365		#[pallet::call_index(2)]
366		#[pallet::weight(
367			T::WeightInfo::vested_transfer(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
368		)]
369		pub fn vested_transfer(
370			origin: OriginFor<T>,
371			target: AccountIdLookupOf<T>,
372			schedule: VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
373		) -> DispatchResult {
374			let transactor = ensure_signed(origin)?;
375			let target = T::Lookup::lookup(target)?;
376			Self::do_vested_transfer(&transactor, &target, schedule)
377		}
378
379		/// Force a vested transfer.
380		///
381		/// The dispatch origin for this call must be _Root_.
382		///
383		/// - `source`: The account whose funds should be transferred.
384		/// - `target`: The account that should be transferred the vested funds.
385		/// - `schedule`: The vesting schedule attached to the transfer.
386		///
387		/// Emits `VestingCreated`.
388		///
389		/// NOTE: This will unlock all schedules through the current block.
390		///
391		/// ## Complexity
392		/// - `O(1)`.
393		#[pallet::call_index(3)]
394		#[pallet::weight(
395			T::WeightInfo::force_vested_transfer(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
396		)]
397		pub fn force_vested_transfer(
398			origin: OriginFor<T>,
399			source: AccountIdLookupOf<T>,
400			target: AccountIdLookupOf<T>,
401			schedule: VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
402		) -> DispatchResult {
403			ensure_root(origin)?;
404			let target = T::Lookup::lookup(target)?;
405			let source = T::Lookup::lookup(source)?;
406			Self::do_vested_transfer(&source, &target, schedule)
407		}
408
409		/// Merge two vesting schedules together, creating a new vesting schedule that unlocks over
410		/// the highest possible start and end blocks. If both schedules have already started the
411		/// current block will be used as the schedule start; with the caveat that if one schedule
412		/// is finished by the current block, the other will be treated as the new merged schedule,
413		/// unmodified.
414		///
415		/// NOTE: If `schedule1_index == schedule2_index` this is a no-op.
416		/// NOTE: This will unlock all schedules through the current block prior to merging.
417		/// NOTE: If both schedules have ended by the current block, no new schedule will be created
418		/// and both will be removed.
419		///
420		/// Merged schedule attributes:
421		/// - `starting_block`: `MAX(schedule1.starting_block, scheduled2.starting_block,
422		///   current_block)`.
423		/// - `ending_block`: `MAX(schedule1.ending_block, schedule2.ending_block)`.
424		/// - `locked`: `schedule1.locked_at(current_block) + schedule2.locked_at(current_block)`.
425		///
426		/// The dispatch origin for this call must be _Signed_.
427		///
428		/// - `schedule1_index`: index of the first schedule to merge.
429		/// - `schedule2_index`: index of the second schedule to merge.
430		#[pallet::call_index(4)]
431		#[pallet::weight(
432			T::WeightInfo::not_unlocking_merge_schedules(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
433			.max(T::WeightInfo::unlocking_merge_schedules(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES))
434		)]
435		pub fn merge_schedules(
436			origin: OriginFor<T>,
437			schedule1_index: u32,
438			schedule2_index: u32,
439		) -> DispatchResult {
440			let who = ensure_signed(origin)?;
441			if schedule1_index == schedule2_index {
442				return Ok(())
443			};
444			let schedule1_index = schedule1_index as usize;
445			let schedule2_index = schedule2_index as usize;
446
447			let schedules = Vesting::<T>::get(&who).ok_or(Error::<T>::NotVesting)?;
448			let merge_action =
449				VestingAction::Merge { index1: schedule1_index, index2: schedule2_index };
450
451			let (schedules, locked_now) = Self::exec_action(schedules.to_vec(), merge_action)?;
452
453			Self::write_vesting(&who, schedules)?;
454			Self::write_lock(&who, locked_now);
455
456			Ok(())
457		}
458
459		/// Force remove a vesting schedule
460		///
461		/// The dispatch origin for this call must be _Root_.
462		///
463		/// - `target`: An account that has a vesting schedule
464		/// - `schedule_index`: The vesting schedule index that should be removed
465		#[pallet::call_index(5)]
466		#[pallet::weight(
467			T::WeightInfo::force_remove_vesting_schedule(MaxLocksOf::<T>::get(), T::MAX_VESTING_SCHEDULES)
468		)]
469		pub fn force_remove_vesting_schedule(
470			origin: OriginFor<T>,
471			target: <T::Lookup as StaticLookup>::Source,
472			schedule_index: u32,
473		) -> DispatchResultWithPostInfo {
474			ensure_root(origin)?;
475			let who = T::Lookup::lookup(target)?;
476
477			let schedules_count = Vesting::<T>::decode_len(&who).unwrap_or_default();
478			ensure!(schedule_index < schedules_count as u32, Error::<T>::InvalidScheduleParams);
479
480			Self::remove_vesting_schedule(&who, schedule_index)?;
481
482			Ok(Some(T::WeightInfo::force_remove_vesting_schedule(
483				MaxLocksOf::<T>::get(),
484				schedules_count as u32,
485			))
486			.into())
487		}
488	}
489}
490
491impl<T: Config> Pallet<T> {
492	// Public function for accessing vesting storage
493	pub fn vesting(
494		account: T::AccountId,
495	) -> Option<BoundedVec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>, MaxVestingSchedulesGet<T>>>
496	{
497		Vesting::<T>::get(account)
498	}
499
500	// Create a new `VestingInfo`, based off of two other `VestingInfo`s.
501	// NOTE: We assume both schedules have had funds unlocked up through the current block.
502	fn merge_vesting_info(
503		now: BlockNumberFor<T>,
504		schedule1: VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
505		schedule2: VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
506	) -> Option<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>> {
507		let schedule1_ending_block = schedule1.ending_block_as_balance::<T::BlockNumberToBalance>();
508		let schedule2_ending_block = schedule2.ending_block_as_balance::<T::BlockNumberToBalance>();
509		let now_as_balance = T::BlockNumberToBalance::convert(now);
510
511		// Check if one or both schedules have ended.
512		match (schedule1_ending_block <= now_as_balance, schedule2_ending_block <= now_as_balance) {
513			// If both schedules have ended, we don't merge and exit early.
514			(true, true) => return None,
515			// If one schedule has ended, we treat the one that has not ended as the new
516			// merged schedule.
517			(true, false) => return Some(schedule2),
518			(false, true) => return Some(schedule1),
519			// If neither schedule has ended don't exit early.
520			_ => {},
521		}
522
523		let locked = schedule1
524			.locked_at::<T::BlockNumberToBalance>(now)
525			.saturating_add(schedule2.locked_at::<T::BlockNumberToBalance>(now));
526		// This shouldn't happen because we know at least one ending block is greater than now,
527		// thus at least a schedule a some locked balance.
528		debug_assert!(
529			!locked.is_zero(),
530			"merge_vesting_info validation checks failed to catch a locked of 0"
531		);
532
533		let ending_block = schedule1_ending_block.max(schedule2_ending_block);
534		let starting_block = now.max(schedule1.starting_block()).max(schedule2.starting_block());
535
536		let per_block = {
537			let duration = ending_block
538				.saturating_sub(T::BlockNumberToBalance::convert(starting_block))
539				.max(One::one());
540			(locked / duration).max(One::one())
541		};
542
543		let schedule = VestingInfo::new(locked, per_block, starting_block);
544		debug_assert!(schedule.is_valid(), "merge_vesting_info schedule validation check failed");
545
546		Some(schedule)
547	}
548
549	// Execute a vested transfer from `source` to `target` with the given `schedule`.
550	fn do_vested_transfer(
551		source: &T::AccountId,
552		target: &T::AccountId,
553		schedule: VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
554	) -> DispatchResult {
555		// Validate user inputs.
556		ensure!(schedule.locked() >= T::MinVestedTransfer::get(), Error::<T>::AmountLow);
557		if !schedule.is_valid() {
558			return Err(Error::<T>::InvalidScheduleParams.into())
559		};
560
561		// Check we can add to this account prior to any storage writes.
562		Self::can_add_vesting_schedule(
563			target,
564			schedule.locked(),
565			schedule.per_block(),
566			schedule.starting_block(),
567		)?;
568
569		T::Currency::transfer(source, target, schedule.locked(), ExistenceRequirement::AllowDeath)?;
570
571		// We can't let this fail because the currency transfer has already happened.
572		// Must be successful as it has been checked before.
573		// Better to return error on failure anyway.
574		let res = Self::add_vesting_schedule(
575			target,
576			schedule.locked(),
577			schedule.per_block(),
578			schedule.starting_block(),
579		);
580		debug_assert!(res.is_ok(), "Failed to add a schedule when we had to succeed.");
581
582		Ok(())
583	}
584
585	/// Iterate through the schedules to track the current locked amount and
586	/// filter out completed and specified schedules.
587	///
588	/// Returns a tuple that consists of:
589	/// - Vec of vesting schedules, where completed schedules and those specified
590	/// 	by filter are removed. (Note the vec is not checked for respecting
591	/// 	bounded length.)
592	/// - The amount locked at the current block number based on the given schedules.
593	///
594	/// NOTE: the amount locked does not include any schedules that are filtered out via `action`.
595	fn report_schedule_updates(
596		schedules: Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>,
597		action: VestingAction,
598	) -> (Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>, BalanceOf<T>) {
599		let now = T::BlockNumberProvider::current_block_number();
600
601		let mut total_locked_now: BalanceOf<T> = Zero::zero();
602		let filtered_schedules = action
603			.pick_schedules::<T>(schedules)
604			.filter(|schedule| {
605				let locked_now = schedule.locked_at::<T::BlockNumberToBalance>(now);
606				let keep = !locked_now.is_zero();
607				if keep {
608					total_locked_now = total_locked_now.saturating_add(locked_now);
609				}
610				keep
611			})
612			.collect::<Vec<_>>();
613
614		(filtered_schedules, total_locked_now)
615	}
616
617	/// Write an accounts updated vesting lock to storage.
618	fn write_lock(who: &T::AccountId, total_locked_now: BalanceOf<T>) {
619		if total_locked_now.is_zero() {
620			T::Currency::remove_lock(VESTING_ID, who);
621			Self::deposit_event(Event::<T>::VestingCompleted { account: who.clone() });
622		} else {
623			let reasons = WithdrawReasons::except(T::UnvestedFundsAllowedWithdrawReasons::get());
624			T::Currency::set_lock(VESTING_ID, who, total_locked_now, reasons);
625			Self::deposit_event(Event::<T>::VestingUpdated {
626				account: who.clone(),
627				unvested: total_locked_now,
628			});
629		};
630	}
631
632	/// Write an accounts updated vesting schedules to storage.
633	fn write_vesting(
634		who: &T::AccountId,
635		schedules: Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>,
636	) -> Result<(), DispatchError> {
637		let schedules: BoundedVec<
638			VestingInfo<BalanceOf<T>, BlockNumberFor<T>>,
639			MaxVestingSchedulesGet<T>,
640		> = schedules.try_into().map_err(|_| Error::<T>::AtMaxVestingSchedules)?;
641
642		if schedules.len() == 0 {
643			Vesting::<T>::remove(&who);
644		} else {
645			Vesting::<T>::insert(who, schedules)
646		}
647
648		Ok(())
649	}
650
651	/// Unlock any vested funds of `who`.
652	fn do_vest(who: T::AccountId) -> DispatchResult {
653		let schedules = Vesting::<T>::get(&who).ok_or(Error::<T>::NotVesting)?;
654
655		let (schedules, locked_now) =
656			Self::exec_action(schedules.to_vec(), VestingAction::Passive)?;
657
658		Self::write_vesting(&who, schedules)?;
659		Self::write_lock(&who, locked_now);
660
661		Ok(())
662	}
663
664	/// Execute a `VestingAction` against the given `schedules`. Returns the updated schedules
665	/// and locked amount.
666	fn exec_action(
667		schedules: Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>,
668		action: VestingAction,
669	) -> Result<(Vec<VestingInfo<BalanceOf<T>, BlockNumberFor<T>>>, BalanceOf<T>), DispatchError> {
670		let (schedules, locked_now) = match action {
671			VestingAction::Merge { index1: idx1, index2: idx2 } => {
672				// The schedule index is based off of the schedule ordering prior to filtering out
673				// any schedules that may be ending at this block.
674				let schedule1 = *schedules.get(idx1).ok_or(Error::<T>::ScheduleIndexOutOfBounds)?;
675				let schedule2 = *schedules.get(idx2).ok_or(Error::<T>::ScheduleIndexOutOfBounds)?;
676
677				// The length of `schedules` decreases by 2 here since we filter out 2 schedules.
678				// Thus we know below that we can push the new merged schedule without error
679				// (assuming initial state was valid).
680				let (mut schedules, mut locked_now) =
681					Self::report_schedule_updates(schedules.to_vec(), action);
682
683				let now = T::BlockNumberProvider::current_block_number();
684				if let Some(new_schedule) = Self::merge_vesting_info(now, schedule1, schedule2) {
685					// Merging created a new schedule so we:
686					// 1) need to add it to the accounts vesting schedule collection,
687					schedules.push(new_schedule);
688					// (we use `locked_at` in case this is a schedule that started in the past)
689					let new_schedule_locked =
690						new_schedule.locked_at::<T::BlockNumberToBalance>(now);
691					// and 2) update the locked amount to reflect the schedule we just added.
692					locked_now = locked_now.saturating_add(new_schedule_locked);
693				} // In the None case there was no new schedule to account for.
694
695				(schedules, locked_now)
696			},
697			_ => Self::report_schedule_updates(schedules.to_vec(), action),
698		};
699
700		debug_assert!(
701			locked_now > Zero::zero() && schedules.len() > 0 ||
702				locked_now == Zero::zero() && schedules.len() == 0
703		);
704
705		Ok((schedules, locked_now))
706	}
707}
708
709impl<T: Config> VestingSchedule<T::AccountId> for Pallet<T>
710where
711	BalanceOf<T>: MaybeSerializeDeserialize + Debug,
712{
713	type Currency = T::Currency;
714	type Moment = BlockNumberFor<T>;
715
716	/// Get the amount that is currently being vested and cannot be transferred out of this account.
717	fn vesting_balance(who: &T::AccountId) -> Option<BalanceOf<T>> {
718		if let Some(v) = Vesting::<T>::get(who) {
719			let now = T::BlockNumberProvider::current_block_number();
720			let total_locked_now = v.iter().fold(Zero::zero(), |total, schedule| {
721				schedule.locked_at::<T::BlockNumberToBalance>(now).saturating_add(total)
722			});
723			Some(T::Currency::free_balance(who).min(total_locked_now))
724		} else {
725			None
726		}
727	}
728
729	/// Adds a vesting schedule to a given account.
730	///
731	/// If the account has `MaxVestingSchedules`, an Error is returned and nothing
732	/// is updated.
733	///
734	/// On success, a linearly reducing amount of funds will be locked. In order to realise any
735	/// reduction of the lock over time as it diminishes, the account owner must use `vest` or
736	/// `vest_other`.
737	///
738	/// Is a no-op if the amount to be vested is zero.
739	///
740	/// NOTE: This doesn't alter the free balance of the account.
741	fn add_vesting_schedule(
742		who: &T::AccountId,
743		locked: BalanceOf<T>,
744		per_block: BalanceOf<T>,
745		starting_block: BlockNumberFor<T>,
746	) -> DispatchResult {
747		if locked.is_zero() {
748			return Ok(())
749		}
750
751		let vesting_schedule = VestingInfo::new(locked, per_block, starting_block);
752		// Check for `per_block` or `locked` of 0.
753		if !vesting_schedule.is_valid() {
754			return Err(Error::<T>::InvalidScheduleParams.into())
755		};
756
757		let mut schedules = Vesting::<T>::get(who).unwrap_or_default();
758
759		// NOTE: we must push the new schedule so that `exec_action`
760		// will give the correct new locked amount.
761		ensure!(schedules.try_push(vesting_schedule).is_ok(), Error::<T>::AtMaxVestingSchedules);
762
763		let (schedules, locked_now) =
764			Self::exec_action(schedules.to_vec(), VestingAction::Passive)?;
765
766		Self::write_vesting(who, schedules)?;
767		Self::write_lock(who, locked_now);
768
769		Ok(())
770	}
771
772	/// Ensure we can call `add_vesting_schedule` without error. This should always
773	/// be called prior to `add_vesting_schedule`.
774	fn can_add_vesting_schedule(
775		who: &T::AccountId,
776		locked: BalanceOf<T>,
777		per_block: BalanceOf<T>,
778		starting_block: BlockNumberFor<T>,
779	) -> DispatchResult {
780		// Check for `per_block` or `locked` of 0.
781		if !VestingInfo::new(locked, per_block, starting_block).is_valid() {
782			return Err(Error::<T>::InvalidScheduleParams.into())
783		}
784
785		ensure!(
786			(Vesting::<T>::decode_len(who).unwrap_or_default() as u32) < T::MAX_VESTING_SCHEDULES,
787			Error::<T>::AtMaxVestingSchedules
788		);
789
790		Ok(())
791	}
792
793	/// Remove a vesting schedule for a given account.
794	fn remove_vesting_schedule(who: &T::AccountId, schedule_index: u32) -> DispatchResult {
795		let schedules = Vesting::<T>::get(who).ok_or(Error::<T>::NotVesting)?;
796		let remove_action = VestingAction::Remove { index: schedule_index as usize };
797
798		let (schedules, locked_now) = Self::exec_action(schedules.to_vec(), remove_action)?;
799
800		Self::write_vesting(who, schedules)?;
801		Self::write_lock(who, locked_now);
802		Ok(())
803	}
804}
805
806/// An implementation that allows the Vesting Pallet to handle a vested transfer
807/// on behalf of another Pallet.
808impl<T: Config> VestedTransfer<T::AccountId> for Pallet<T>
809where
810	BalanceOf<T>: MaybeSerializeDeserialize + Debug,
811{
812	type Currency = T::Currency;
813	type Moment = BlockNumberFor<T>;
814
815	fn vested_transfer(
816		source: &T::AccountId,
817		target: &T::AccountId,
818		locked: BalanceOf<T>,
819		per_block: BalanceOf<T>,
820		starting_block: BlockNumberFor<T>,
821	) -> DispatchResult {
822		use frame_support::storage::{with_transaction, TransactionOutcome};
823		let schedule = VestingInfo::new(locked, per_block, starting_block);
824		with_transaction(|| -> TransactionOutcome<DispatchResult> {
825			let result = Self::do_vested_transfer(source, target, schedule);
826
827			match &result {
828				Ok(()) => TransactionOutcome::Commit(result),
829				_ => TransactionOutcome::Rollback(result),
830			}
831		})
832	}
833}