diff --git a/src/lib.rs b/src/lib.rs index 63013bf..6481718 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -323,6 +323,11 @@ pub mod pallet { pub type Candidates = CountedStorageMap<_, Blake2_128Concat, T::AccountId, CandidateInfoOf, OptionQuery>; + /// Map of Candidates that have been removed in the current session. + #[pallet::storage] + pub type SessionRemovedCandidates = + StorageMap<_, Blake2_128Concat, T::AccountId, CandidateInfoOf, OptionQuery>; + /// Last block authored by a collator. #[pallet::storage] pub type LastAuthoredBlock = @@ -575,6 +580,8 @@ pub mod pallet { NoStakeOnCandidate, /// No rewards to claim as previous claim happened on the same session. NoPendingClaim, + /// Candidate has not been removed in the current session. + NotRemovedCandidate, } #[pallet::hooks] @@ -1321,6 +1328,11 @@ pub mod pallet { } let info = CandidateInfo { stake, stakers }; *maybe_candidate_info = Some(info.clone()); + + // If the candidate left in the current session and is now rejoining + // remove it from the SessionRemovedCandidates + SessionRemovedCandidates::::remove(who); + T::Currency::set_freeze(&FreezeReason::CandidacyBond.into(), who, bond)?; Ok(info) }, @@ -1569,6 +1581,10 @@ pub mod pallet { } Self::release_candidacy_bond(who)?; + // Store removed candidate in SessionRemovedCandidates to properly reward + // the candidate and its stakers at the end of the session. + SessionRemovedCandidates::::insert(who, candidate.clone()); + Self::deposit_event(Event::CandidateRemoved { account: who.clone() }); *maybe_candidate = None; Ok(candidate) @@ -1700,7 +1716,17 @@ pub mod pallet { if !rewardable_blocks.is_zero() && !total_rewards.is_zero() { let collator_percentage = CollatorRewardPercentage::::get(); for (collator, blocks) in ProducedBlocks::::drain() { - if let Ok(collator_info) = Self::get_candidate(&collator) { + // Get the collator info of a candidate, in the case that the collator was removed from the + // candidate list during the session, the collator and its stakers must still be rewarded + // for the produced blocks in the session so the info can be obtained from SessionRemovedCandidates. + let info = Self::get_candidate(&collator) + .or_else(|_| { + SessionRemovedCandidates::::take(&collator) + .ok_or(Error::::NotRemovedCandidate) + }) + .ok(); + + if let Some(collator_info) = info { if blocks > rewardable_blocks { // The only case this could happen is if the candidate was an invulnerable during the session. // Since blocks produced by invulnerables are not currently stored in ProducedBlocks this error @@ -1892,6 +1918,7 @@ pub mod pallet { let (candidate, worst_bond) = Self::get_worst_candidate()?; ensure!(bond > worst_bond, Error::::InvalidCandidacyBond); Self::try_remove_candidate(&candidate, false)?; + Ok(candidate) } @@ -2001,6 +2028,12 @@ pub mod pallet { let removed = candidates_len_before.saturating_sub(active_candidates_count); let result = Self::assemble_collators(); + // Although the removed candidates are passively deleted from SessionRemovedCandidates + // during the distribution of session rewards, it is possible that a removed candidate + // is not removed if the candidate didn't produce and blocks during the session. For that + // reason the leftover keys in the SessionRemovedCandidates StorageMap must be cleared. + let _ = SessionRemovedCandidates::::clear(T::MaxCandidates::get(), None); + frame_system::Pallet::::register_extra_weight_unchecked( T::WeightInfo::new_session(removed, candidates_len_before), DispatchClass::Mandatory, diff --git a/src/tests.rs b/src/tests.rs index 0a6a58f..43d4e7e 100644 --- a/src/tests.rs +++ b/src/tests.rs @@ -18,8 +18,8 @@ use crate::{ mock::*, AutoCompound, BalanceOf, CandidacyBondReleases, CandidateInfo, CandidateStakeInfo, Candidates, ClaimableRewards, CollatorRewardPercentage, Config, CurrentSession, DesiredCandidates, Error, Event, ExtraReward, FreezeReason, Invulnerables, LastAuthoredBlock, - MinCandidacyBond, MinStake, PerSessionRewards, ProducedBlocks, ReleaseQueues, StakeTarget, - TotalBlocks, UserStake, UserStakeInfo, + MinCandidacyBond, MinStake, PerSessionRewards, ProducedBlocks, ReleaseQueues, + SessionRemovedCandidates, StakeTarget, TotalBlocks, UserStake, UserStakeInfo, }; use crate::{CandidateStake, ReleaseRequest}; use frame_support::pallet_prelude::TypedGet; @@ -823,6 +823,10 @@ mod register_as_candidate { // the candidate leaves assert_ok!(CollatorStaking::leave_intent(RuntimeOrigin::signed(3))); + assert_eq!( + SessionRemovedCandidates::::get(3), + Some(CandidateInfo { stake: 60, stakers: 1 }) + ); // the stake remains the same assert_eq!( CandidateStake::::get(3, 4), @@ -929,6 +933,10 @@ mod leave_intent { CandidateStakeInfo { stake: 0, session: 0 } ); assert_eq!(LastAuthoredBlock::::get(3), 0); + assert_eq!( + SessionRemovedCandidates::::get(3), + Some(CandidateInfo { stake: 0, stakers: 0 }) + ); }); } @@ -2506,6 +2514,7 @@ mod collator_rewards { })); }); } + #[test] fn should_reward_collator() { new_test_ext().execute_with(|| { @@ -2572,6 +2581,10 @@ mod collator_rewards { Balances::free_balance(CollatorStaking::account_id()) - Balances::minimum_balance(), 18 ); + // we can safely remove the collator, as rewards will be delivered anyway to both + // the collator itself and its stakers. + assert_ok!(CollatorStaking::leave_intent(RuntimeOrigin::signed(4))); + initialize_to_block(20); System::assert_has_event(RuntimeEvent::CollatorStaking(Event::SessionEnded { index: 1,