From 0555884c91b6b92c3706214232e4902a904234fe Mon Sep 17 00:00:00 2001 From: Dorin Marian Iancu Date: Thu, 14 Nov 2024 15:02:37 +0200 Subject: [PATCH 1/4] impl --- .../src/sc_whitelist_module.rs | 1 + dex/pair/tests/pair_rs_test.rs | 1 + .../weekly-rewards-splitting/src/base_impl.rs | 14 +- .../src/global_info.rs | 7 +- .../src/additional_locked_tokens.rs | 6 +- .../fees-collector/src/claim.rs | 208 ++++++++++++++++++ .../fees-collector/src/config.rs | 30 +-- .../fees-collector/src/events.rs | 8 +- .../fees-collector/src/fees_accumulation.rs | 81 ++++++- energy-integration/fees-collector/src/lib.rs | 150 ++----------- .../tests/fees_collector_test_setup/mod.rs | 2 + .../fees-collector/wasm/src/lib.rs | 16 +- .../governance-v2/tests/gov_test_setup/mod.rs | 3 +- 13 files changed, 356 insertions(+), 171 deletions(-) create mode 100644 energy-integration/fees-collector/src/claim.rs diff --git a/common/modules/sc_whitelist_module/src/sc_whitelist_module.rs b/common/modules/sc_whitelist_module/src/sc_whitelist_module.rs index b04216db2..19de606b0 100644 --- a/common/modules/sc_whitelist_module/src/sc_whitelist_module.rs +++ b/common/modules/sc_whitelist_module/src/sc_whitelist_module.rs @@ -40,6 +40,7 @@ pub trait SCWhitelistModule { match opt_orig_caller { OptionalValue::Some(opt_caller) => { self.require_sc_address_whitelisted(caller); + opt_caller } OptionalValue::None => caller.clone(), diff --git a/dex/pair/tests/pair_rs_test.rs b/dex/pair/tests/pair_rs_test.rs index bd82750d8..7f43373ff 100644 --- a/dex/pair/tests/pair_rs_test.rs +++ b/dex/pair/tests/pair_rs_test.rs @@ -1600,6 +1600,7 @@ fn fees_collector_pair_test() { sc.init( managed_token_id!(LOCKED_TOKEN_ID), managed_address!(&energy_factory_mock_addr), + MultiValueEncoded::new(), ); let _ = sc.known_contracts().insert(managed_address!(&pair_addr)); diff --git a/energy-integration/common-modules/weekly-rewards-splitting/src/base_impl.rs b/energy-integration/common-modules/weekly-rewards-splitting/src/base_impl.rs index de8783686..96de99ce9 100644 --- a/energy-integration/common-modules/weekly-rewards-splitting/src/base_impl.rs +++ b/energy-integration/common-modules/weekly-rewards-splitting/src/base_impl.rs @@ -64,13 +64,15 @@ pub trait WeeklyRewardsSplittingTraitsModule { let total_rewards = self.collect_and_get_rewards_for_week(sc, week); for weekly_reward in &total_rewards { let reward_amount = weekly_reward.amount * energy_amount / total_energy; - if reward_amount > 0 { - user_rewards.push(EsdtTokenPayment::new( - weekly_reward.token_identifier, - 0, - reward_amount, - )); + if reward_amount == 0 { + continue; } + + user_rewards.push(EsdtTokenPayment::new( + weekly_reward.token_identifier, + 0, + reward_amount, + )); } user_rewards diff --git a/energy-integration/common-modules/weekly-rewards-splitting/src/global_info.rs b/energy-integration/common-modules/weekly-rewards-splitting/src/global_info.rs index b05f5aa66..488569bad 100644 --- a/energy-integration/common-modules/weekly-rewards-splitting/src/global_info.rs +++ b/energy-integration/common-modules/weekly-rewards-splitting/src/global_info.rs @@ -1,6 +1,6 @@ multiversx_sc::imports!(); -use common_types::Week; +use common_types::{PaymentsVec, Week}; use energy_query::Energy; use week_timekeeping::EPOCHS_IN_WEEK; @@ -147,10 +147,7 @@ pub trait WeeklyRewardsGlobalInfo: #[view(getTotalRewardsForWeek)] #[storage_mapper("totalRewardsForWeek")] - fn total_rewards_for_week( - &self, - week: Week, - ) -> SingleValueMapper>>; + fn total_rewards_for_week(&self, week: Week) -> SingleValueMapper>; #[view(getTotalEnergyForWeek)] #[storage_mapper("totalEnergyForWeek")] diff --git a/energy-integration/fees-collector/src/additional_locked_tokens.rs b/energy-integration/fees-collector/src/additional_locked_tokens.rs index b74c967f3..fa108c3fb 100644 --- a/energy-integration/fees-collector/src/additional_locked_tokens.rs +++ b/energy-integration/fees-collector/src/additional_locked_tokens.rs @@ -10,6 +10,7 @@ pub trait AdditionalLockedTokensModule: + crate::fees_accumulation::FeesAccumulationModule + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule + + multiversx_sc_modules::only_admin::OnlyAdminModule { #[only_owner] #[endpoint(setLockedTokensPerBlock)] @@ -20,19 +21,18 @@ pub trait AdditionalLockedTokensModule: fn accumulate_additional_locked_tokens(&self) { let last_update_week_mapper = self.last_locked_token_add_week(); - let mut last_update_week = last_update_week_mapper.get(); + let last_update_week = last_update_week_mapper.get(); let current_week = self.get_current_week(); if last_update_week == current_week { return; } - last_update_week = current_week - 1; let blocks_in_week = BLOCKS_IN_WEEK; let amount_per_block = self.locked_tokens_per_block().get(); let new_tokens_amount = amount_per_block * blocks_in_week; let locked_token_id = self.locked_token_id().get(); - self.accumulated_fees(last_update_week, &locked_token_id) + self.accumulated_fees(current_week - 1, &locked_token_id) .update(|fees| *fees += new_tokens_amount); last_update_week_mapper.set(current_week); diff --git a/energy-integration/fees-collector/src/claim.rs b/energy-integration/fees-collector/src/claim.rs new file mode 100644 index 000000000..1b5884410 --- /dev/null +++ b/energy-integration/fees-collector/src/claim.rs @@ -0,0 +1,208 @@ +use core::marker::PhantomData; + +use common_types::{PaymentsVec, Week}; +use weekly_rewards_splitting::base_impl::WeeklyRewardsSplittingTraitsModule; + +multiversx_sc::imports!(); + +#[multiversx_sc::module] +pub trait ClaimModule: + crate::config::ConfigModule + + crate::events::FeesCollectorEventsModule + + weekly_rewards_splitting::WeeklyRewardsSplittingModule + + weekly_rewards_splitting::events::WeeklyRewardsSplittingEventsModule + + weekly_rewards_splitting::global_info::WeeklyRewardsGlobalInfo + + weekly_rewards_splitting::locked_token_buckets::WeeklyRewardsLockedTokenBucketsModule + + weekly_rewards_splitting::update_claim_progress_energy::UpdateClaimProgressEnergyModule + + crate::fees_accumulation::FeesAccumulationModule + + crate::additional_locked_tokens::AdditionalLockedTokensModule + + locking_module::lock_with_energy_module::LockWithEnergyModule + + energy_query::EnergyQueryModule + + week_timekeeping::WeekTimekeepingModule + + multiversx_sc_modules::pause::PauseModule + + utils::UtilsModule + + sc_whitelist_module::SCWhitelistModule + + multiversx_sc_modules::only_admin::OnlyAdminModule +{ + #[endpoint(claimRewards)] + fn claim_rewards_endpoint( + &self, + opt_original_caller: OptionalValue, + ) -> PaymentsVec { + self.require_not_paused(); + + let caller = self.blockchain().get_caller(); + let original_caller = self.get_orig_caller_from_opt(&caller, opt_original_caller); + + self.claim_rewards(caller, original_caller) + } + + #[endpoint(claimBoostedRewards)] + fn claim_boosted_rewards( + &self, + opt_original_caller: OptionalValue, + ) -> PaymentsVec { + self.require_not_paused(); + + let original_caller = match opt_original_caller { + OptionalValue::Some(user) => { + require!( + self.allow_external_claim_rewards(&user).get(), + "Cannot claim rewards for this address" + ); + + user + } + OptionalValue::None => self.blockchain().get_caller(), + }; + + self.claim_rewards(original_caller.clone(), original_caller) + } + + fn claim_rewards( + &self, + caller: ManagedAddress, + original_caller: ManagedAddress, + ) -> PaymentsVec { + self.accumulate_additional_locked_tokens(); + + let wrapper = FeesCollectorWrapper::new(); + let mut rewards = self.claim_multi(&wrapper, &original_caller); + if rewards.is_empty() { + return rewards; + } + + let locked_token_id = self.get_locked_token_id(); + let mut i = 0; + let mut len = rewards.len(); + let mut total_locked_token_rewards_amount = BigUint::zero(); + while i < len { + let rew = rewards.get(i); + if rew.token_identifier != locked_token_id { + i += 1; + continue; + } + + total_locked_token_rewards_amount += rew.amount; + len -= 1; + rewards.remove(i); + } + + if !rewards.is_empty() { + self.send().direct_multi(&caller, &rewards); + } + + if total_locked_token_rewards_amount > 0 { + let locked_rewards = self.lock_virtual( + self.get_base_token_id(), + total_locked_token_rewards_amount, + caller, + original_caller, + ); + + rewards.push(locked_rewards); + } + + rewards + } +} + +pub struct FeesCollectorWrapper { + phantom: PhantomData, +} + +impl Default for FeesCollectorWrapper { + fn default() -> Self { + Self::new() + } +} + +impl FeesCollectorWrapper { + pub fn new() -> FeesCollectorWrapper { + FeesCollectorWrapper { + phantom: PhantomData, + } + } +} + +impl WeeklyRewardsSplittingTraitsModule for FeesCollectorWrapper +where + T: ClaimModule, +{ + type WeeklyRewardsSplittingMod = T; + + fn get_user_rewards_for_week( + &self, + sc: &Self::WeeklyRewardsSplittingMod, + week: Week, + energy_amount: &BigUint<::Api>, + total_energy: &BigUint<::Api>, + ) -> PaymentsVec<::Api> { + let mut user_rewards = ManagedVec::new(); + if energy_amount == &0 || total_energy == &0 { + return user_rewards; + } + + let total_rewards = self.collect_and_get_rewards_for_week(sc, week); + let remaining_rewards_mapper = sc.remaining_rewards(week); + let mut remaining_rewards = remaining_rewards_mapper.get(); + let mut update_storage = false; + + for (i, weekly_reward) in total_rewards.iter().enumerate() { + let reward_amount = weekly_reward.amount * energy_amount / total_energy; + if reward_amount == 0 { + continue; + } + + let mut rem_rew_entry = remaining_rewards.get_mut(i); + rem_rew_entry.amount -= &reward_amount; + update_storage = true; + + user_rewards.push(EsdtTokenPayment::new( + weekly_reward.token_identifier, + 0, + reward_amount, + )); + } + + if update_storage { + remaining_rewards_mapper.set(remaining_rewards); + } + + user_rewards + } + + fn collect_and_get_rewards_for_week( + &self, + sc: &Self::WeeklyRewardsSplittingMod, + week: Week, + ) -> PaymentsVec<::Api> { + let total_rewards_mapper = sc.total_rewards_for_week(week); + if total_rewards_mapper.is_empty() { + let total_rewards = self.collect_rewards_for_week(sc, week); + total_rewards_mapper.set(&total_rewards); + sc.remaining_rewards(week).set(&total_rewards); + + total_rewards + } else { + total_rewards_mapper.get() + } + } + + fn collect_rewards_for_week( + &self, + sc: &Self::WeeklyRewardsSplittingMod, + week: Week, + ) -> PaymentsVec<::Api> { + let mut results = ManagedVec::new(); + let all_tokens = sc.all_tokens().get(); + for token in &all_tokens { + let opt_accumulated_fees = sc.get_and_clear_accumulated_fees(week, &token); + if let Some(accumulated_fees) = opt_accumulated_fees { + results.push(EsdtTokenPayment::new(token, 0, accumulated_fees)); + } + } + + results + } +} diff --git a/energy-integration/fees-collector/src/config.rs b/energy-integration/fees-collector/src/config.rs index 72fa1fc70..0cb2d813b 100644 --- a/energy-integration/fees-collector/src/config.rs +++ b/energy-integration/fees-collector/src/config.rs @@ -11,6 +11,7 @@ pub trait ConfigModule { self.blockchain().is_smart_contract(&sc), "Invalid SC address" ); + let _ = mapper.insert(sc); } } @@ -32,10 +33,12 @@ pub trait ConfigModule { for token in tokens { require!(token.is_valid_esdt_identifier(), "Invalid token ID"); - if !known_tokens_mapper.contains(&token) { - known_tokens_mapper.add(&token); - all_tokens_vec.push(token); + if known_tokens_mapper.contains(&token) { + continue; } + + known_tokens_mapper.add(&token); + all_tokens_vec.push(token); } self.all_tokens().set(&all_tokens_vec); @@ -47,28 +50,29 @@ pub trait ConfigModule { let mut all_tokens_vec = self.all_tokens().get(); let known_tokens_mapper = self.known_tokens(); for token in tokens { - if known_tokens_mapper.contains(&token) { - known_tokens_mapper.remove(&token); + if !known_tokens_mapper.contains(&token) { + continue; + } - unsafe { - let index = all_tokens_vec.find(&token).unwrap_unchecked(); - all_tokens_vec.remove(index); - } + known_tokens_mapper.remove(&token); + unsafe { + let index = all_tokens_vec.find(&token).unwrap_unchecked(); + all_tokens_vec.remove(index); } } self.all_tokens().set(&all_tokens_vec); } - #[view(getLockedTokenId)] - #[storage_mapper("lockedTokenId")] - fn locked_token_id(&self) -> SingleValueMapper; - #[view(getAllTokens)] fn get_all_tokens(&self) -> MultiValueEncoded { self.all_tokens().get().into() } + #[view(getLockedTokenId)] + #[storage_mapper("lockedTokenId")] + fn locked_token_id(&self) -> SingleValueMapper; + #[view(getAllKnownContracts)] #[storage_mapper("knownContracts")] fn known_contracts(&self) -> UnorderedSetMapper; diff --git a/energy-integration/fees-collector/src/events.rs b/energy-integration/fees-collector/src/events.rs index 4962de3e5..7e10ea434 100644 --- a/energy-integration/fees-collector/src/events.rs +++ b/energy-integration/fees-collector/src/events.rs @@ -6,9 +6,9 @@ use common_types::Week; pub trait FeesCollectorEventsModule { fn emit_deposit_swap_fees_event( self, - caller: ManagedAddress, + caller: &ManagedAddress, current_week: Week, - payment: EsdtTokenPayment, + payment: &EsdtTokenPayment, ) { self.deposit_swap_fees_event(caller, current_week, payment); } @@ -16,8 +16,8 @@ pub trait FeesCollectorEventsModule { #[event("deposit_swap_fees_event")] fn deposit_swap_fees_event( &self, - #[indexed] caller: ManagedAddress, + #[indexed] caller: &ManagedAddress, #[indexed] current_week: Week, - #[indexed] payment: EsdtTokenPayment, + #[indexed] payment: &EsdtTokenPayment, ); } diff --git a/energy-integration/fees-collector/src/fees_accumulation.rs b/energy-integration/fees-collector/src/fees_accumulation.rs index f2857407f..f15177c44 100644 --- a/energy-integration/fees-collector/src/fees_accumulation.rs +++ b/energy-integration/fees-collector/src/fees_accumulation.rs @@ -1,13 +1,16 @@ multiversx_sc::imports!(); multiversx_sc::derive_imports!(); +use common_types::PaymentsVec; use week_timekeeping::Week; +use weekly_rewards_splitting::USER_MAX_CLAIM_WEEKS; #[multiversx_sc::module] pub trait FeesAccumulationModule: crate::config::ConfigModule + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule + + multiversx_sc_modules::only_admin::OnlyAdminModule { /// Pair SC will deposit the fees through this endpoint /// Deposits for current week are accessible starting next week @@ -25,23 +28,93 @@ pub trait FeesAccumulationModule: self.known_tokens().contains(&payment.token_identifier), "Invalid payment token" ); - let current_week = self.get_current_week(); if payment.token_nonce > 0 { require!( payment.token_identifier == self.locked_token_id().get(), "Invalid locked token" ); + self.send().esdt_local_burn( &payment.token_identifier, payment.token_nonce, &payment.amount, ); } + + let current_week = self.get_current_week(); self.accumulated_fees(current_week, &payment.token_identifier) .update(|amt| *amt += &payment.amount); - self.emit_deposit_swap_fees_event(caller, current_week, payment); + self.emit_deposit_swap_fees_event(&caller, current_week, &payment); + } + + #[only_admin] + #[endpoint(redistributeRewards)] + fn redistribute_rewards(&self, start_week: Week, end_week: Week) { + let collect_rewards_offset = USER_MAX_CLAIM_WEEKS + 1; + let current_week = self.get_current_week(); + require!( + current_week > collect_rewards_offset, + "Current week must be higher than the week offset" + ); + require!(start_week <= end_week, "Invalid week numbers"); + require!( + end_week <= current_week - collect_rewards_offset, + "Invalid end week" + ); + + let all_tokens = self.all_tokens().get(); + let mut all_rewards = PaymentsVec::new(); + for token_id in &all_tokens { + all_rewards.push(EsdtTokenPayment::new(token_id, 0, BigUint::zero())); + } + + self.accumulate_remaining_rewards(&mut all_rewards, &all_tokens, start_week, end_week); + + for reward_entry in &all_rewards { + if reward_entry.amount == 0 { + continue; + } + + self.accumulated_fees(current_week, &reward_entry.token_identifier) + .update(|acc_fees| *acc_fees += reward_entry.amount); + } + } + + fn accumulate_remaining_rewards( + &self, + all_rewards: &mut PaymentsVec, + all_tokens: &ManagedVec, + start_week: Week, + end_week: Week, + ) { + for week in start_week..=end_week { + self.accumulate_remaining_rewards_single_week(all_rewards, all_tokens, week); + } + } + + fn accumulate_remaining_rewards_single_week( + &self, + all_rewards: &mut PaymentsVec, + all_tokens: &ManagedVec, + week: Week, + ) { + let remaining_rewards = self.remaining_rewards(week).take(); + for rem_rew_entry in &remaining_rewards { + if rem_rew_entry.amount == 0 { + continue; + } + + let opt_index = all_tokens.find(&rem_rew_entry.token_identifier); + if opt_index.is_none() { + continue; + } + + let index = unsafe { opt_index.unwrap_unchecked() }; + let mut rew_entry = all_rewards.get_mut(index); + rew_entry.amount += rem_rew_entry.amount; + } } fn get_and_clear_accumulated_fees( @@ -60,4 +133,8 @@ pub trait FeesAccumulationModule: #[view(getAccumulatedFees)] #[storage_mapper("accumulatedFees")] fn accumulated_fees(&self, week: Week, token: &TokenIdentifier) -> SingleValueMapper; + + #[view(getRemainingRewards)] + #[storage_mapper("remainingRewards")] + fn remaining_rewards(&self, week: Week) -> SingleValueMapper>; } diff --git a/energy-integration/fees-collector/src/lib.rs b/energy-integration/fees-collector/src/lib.rs index e78ed6812..ee9bc0a82 100644 --- a/energy-integration/fees-collector/src/lib.rs +++ b/energy-integration/fees-collector/src/lib.rs @@ -2,11 +2,8 @@ multiversx_sc::imports!(); -use common_types::{PaymentsVec, Week}; -use core::marker::PhantomData; -use weekly_rewards_splitting::base_impl::WeeklyRewardsSplittingTraitsModule; - pub mod additional_locked_tokens; +pub mod claim; pub mod config; pub mod events; pub mod fees_accumulation; @@ -28,145 +25,34 @@ pub trait FeesCollector: + multiversx_sc_modules::pause::PauseModule + utils::UtilsModule + sc_whitelist_module::SCWhitelistModule + + multiversx_sc_modules::only_admin::OnlyAdminModule + + claim::ClaimModule { #[init] - fn init(&self, locked_token_id: TokenIdentifier, energy_factory_address: ManagedAddress) { - let current_epoch = self.blockchain().get_block_epoch(); - self.first_week_start_epoch().set_if_empty(current_epoch); + fn init( + &self, + locked_token_id: TokenIdentifier, + energy_factory_address: ManagedAddress, + admins: MultiValueEncoded, + ) { self.require_valid_token_id(&locked_token_id); self.require_sc_address(&energy_factory_address); + let current_epoch = self.blockchain().get_block_epoch(); + self.first_week_start_epoch().set(current_epoch); + let mut tokens = MultiValueEncoded::new(); tokens.push(locked_token_id.clone()); self.add_known_tokens(tokens); - self.locked_token_id().set_if_empty(locked_token_id); - self.energy_factory_address().set(&energy_factory_address); - } - - #[upgrade] - fn upgrade(&self) {} - - #[endpoint(claimRewards)] - fn claim_rewards_endpoint( - &self, - opt_original_caller: OptionalValue, - ) -> PaymentsVec { - require!(self.not_paused(), "Cannot claim while paused"); - - let caller = self.blockchain().get_caller(); - let original_caller = self.get_orig_caller_from_opt(&caller, opt_original_caller); - - self.claim_rewards(caller, original_caller) - } - - #[endpoint(claimBoostedRewards)] - fn claim_boosted_rewards( - &self, - opt_original_caller: OptionalValue, - ) -> PaymentsVec { - require!(self.not_paused(), "Cannot claim while paused"); - - let original_caller = match opt_original_caller { - OptionalValue::Some(user) => { - require!( - self.allow_external_claim_rewards(&user).get(), - "Cannot claim rewards for this address" - ); - user - } - OptionalValue::None => self.blockchain().get_caller(), - }; - - self.claim_rewards(original_caller.clone(), original_caller) - } - - fn claim_rewards( - &self, - caller: ManagedAddress, - original_caller: ManagedAddress, - ) -> PaymentsVec { - self.accumulate_additional_locked_tokens(); - - let wrapper = FeesCollectorWrapper::new(); - let mut rewards = self.claim_multi(&wrapper, &original_caller); - if rewards.is_empty() { - return rewards; - } - - let locked_token_id = self.get_locked_token_id(); - let mut i = 0; - let mut len = rewards.len(); - let mut total_locked_token_rewards_amount = BigUint::zero(); - while i < len { - let rew = rewards.get(i); - if rew.token_identifier != locked_token_id { - i += 1; - continue; - } + self.locked_token_id().set(locked_token_id); + self.energy_factory_address().set(energy_factory_address); - total_locked_token_rewards_amount += rew.amount; - len -= 1; - rewards.remove(i); - } - - if !rewards.is_empty() { - self.send().direct_multi(&caller, &rewards); - } - - if total_locked_token_rewards_amount > 0 { - let locked_rewards = self.lock_virtual( - self.get_base_token_id(), - total_locked_token_rewards_amount, - caller, - original_caller, - ); - - rewards.push(locked_rewards); - } - - rewards - } -} - -pub struct FeesCollectorWrapper { - phantom: PhantomData, -} - -impl Default for FeesCollectorWrapper { - fn default() -> Self { - Self::new() - } -} - -impl FeesCollectorWrapper { - pub fn new() -> FeesCollectorWrapper { - FeesCollectorWrapper { - phantom: PhantomData, + for admin in admins { + self.add_admin(admin); } } -} - -impl WeeklyRewardsSplittingTraitsModule for FeesCollectorWrapper -where - T: FeesCollector, -{ - type WeeklyRewardsSplittingMod = T; - fn collect_rewards_for_week( - &self, - sc: &Self::WeeklyRewardsSplittingMod, - week: Week, - ) -> PaymentsVec<::Api> { - let mut results = ManagedVec::new(); - let all_tokens = sc.all_tokens().get(); - for token in &all_tokens { - let opt_accumulated_fees = sc.get_and_clear_accumulated_fees(week, &token); - if let Some(accumulated_fees) = opt_accumulated_fees { - results.push(EsdtTokenPayment::new(token, 0, accumulated_fees)); - } - } - - results - } + #[upgrade] + fn upgrade(&self) {} } diff --git a/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs b/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs index 359c4eeb5..a18621604 100644 --- a/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs +++ b/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs @@ -1,5 +1,6 @@ #![allow(deprecated)] +use claim::ClaimModule; use multiversx_sc::{ codec::multi_types::OptionalValue, storage::mappers::StorageTokenWrapper, @@ -151,6 +152,7 @@ where sc.init( managed_token_id!(LOCKED_TOKEN_ID), managed_address!(energy_factory_wrapper.address_ref()), + MultiValueEncoded::new(), ); let _ = sc diff --git a/energy-integration/fees-collector/wasm/src/lib.rs b/energy-integration/fees-collector/wasm/src/lib.rs index 77034bd4a..941bfb1fa 100644 --- a/energy-integration/fees-collector/wasm/src/lib.rs +++ b/energy-integration/fees-collector/wasm/src/lib.rs @@ -6,9 +6,9 @@ // Init: 1 // Upgrade: 1 -// Endpoints: 37 +// Endpoints: 43 // Async Callback (empty): 1 -// Total number of exported functions: 40 +// Total number of exported functions: 46 #![no_std] @@ -20,14 +20,12 @@ multiversx_sc_wasm_adapter::endpoints! { ( init => init upgrade => upgrade - claimRewards => claim_rewards_endpoint - claimBoostedRewards => claim_boosted_rewards addKnownContracts => add_known_contracts removeKnownContracts => remove_known_contracts addKnownTokens => add_known_tokens removeKnownTokens => remove_known_tokens - getLockedTokenId => locked_token_id getAllTokens => get_all_tokens + getLockedTokenId => locked_token_id getAllKnownContracts => known_contracts getAllowExternalClaimRewards => allow_external_claim_rewards getLastActiveWeekForUser => get_last_active_week_for_user_view @@ -39,7 +37,9 @@ multiversx_sc_wasm_adapter::endpoints! { updateEnergyForUser => update_energy_for_user getCurrentClaimProgress => current_claim_progress depositSwapFees => deposit_swap_fees + redistributeRewards => redistribute_rewards getAccumulatedFees => accumulated_fees + getRemainingRewards => remaining_rewards setLockedTokensPerBlock => set_locked_tokens_per_block getLastLockedTokensAddWeek => last_locked_token_add_week getLockedTokensPerBlock => locked_tokens_per_block @@ -57,6 +57,12 @@ multiversx_sc_wasm_adapter::endpoints! { addSCAddressToWhitelist => add_sc_address_to_whitelist removeSCAddressFromWhitelist => remove_sc_address_from_whitelist isSCAddressWhitelisted => is_sc_address_whitelisted + isAdmin => is_admin + addAdmin => add_admin + removeAdmin => remove_admin + getAdmins => admins + claimRewards => claim_rewards_endpoint + claimBoostedRewards => claim_boosted_rewards ) } diff --git a/energy-integration/governance-v2/tests/gov_test_setup/mod.rs b/energy-integration/governance-v2/tests/gov_test_setup/mod.rs index c1d448442..5cc2d851c 100644 --- a/energy-integration/governance-v2/tests/gov_test_setup/mod.rs +++ b/energy-integration/governance-v2/tests/gov_test_setup/mod.rs @@ -2,7 +2,7 @@ use energy_factory_mock::EnergyFactoryMock; use energy_query::Energy; -use fees_collector::FeesCollector; +use fees_collector::{claim::ClaimModule, FeesCollector}; use governance_v2::{ configurable::ConfigurablePropertiesModule, proposal_storage::{ProposalStorageModule, VoteType}, @@ -112,6 +112,7 @@ where sc.init( managed_token_id!(XMEX_TOKEN_ID), managed_address!(energy_factory_wrapper.address_ref()), + MultiValueEncoded::new(), ); }) .assert_ok(); From 0c41aa520d3856eb26026dbf276c2582f81b016d Mon Sep 17 00:00:00 2001 From: Dorin Marian Iancu Date: Fri, 15 Nov 2024 08:50:20 +0200 Subject: [PATCH 2/4] refactor --- .../fees-collector/src/claim.rs | 1 + .../fees-collector/src/fees_accumulation.rs | 74 ------------------ energy-integration/fees-collector/src/lib.rs | 2 + .../src/redistribute_rewards.rs | 75 +++++++++++++++++++ .../fees-collector/wasm/src/lib.rs | 4 +- 5 files changed, 80 insertions(+), 76 deletions(-) create mode 100644 energy-integration/fees-collector/src/redistribute_rewards.rs diff --git a/energy-integration/fees-collector/src/claim.rs b/energy-integration/fees-collector/src/claim.rs index 1b5884410..a4cd5320b 100644 --- a/energy-integration/fees-collector/src/claim.rs +++ b/energy-integration/fees-collector/src/claim.rs @@ -23,6 +23,7 @@ pub trait ClaimModule: + utils::UtilsModule + sc_whitelist_module::SCWhitelistModule + multiversx_sc_modules::only_admin::OnlyAdminModule + + crate::redistribute_rewards::RedistributeRewardsModule { #[endpoint(claimRewards)] fn claim_rewards_endpoint( diff --git a/energy-integration/fees-collector/src/fees_accumulation.rs b/energy-integration/fees-collector/src/fees_accumulation.rs index f15177c44..87865e96f 100644 --- a/energy-integration/fees-collector/src/fees_accumulation.rs +++ b/energy-integration/fees-collector/src/fees_accumulation.rs @@ -1,9 +1,7 @@ multiversx_sc::imports!(); multiversx_sc::derive_imports!(); -use common_types::PaymentsVec; use week_timekeeping::Week; -use weekly_rewards_splitting::USER_MAX_CLAIM_WEEKS; #[multiversx_sc::module] pub trait FeesAccumulationModule: @@ -49,74 +47,6 @@ pub trait FeesAccumulationModule: self.emit_deposit_swap_fees_event(&caller, current_week, &payment); } - #[only_admin] - #[endpoint(redistributeRewards)] - fn redistribute_rewards(&self, start_week: Week, end_week: Week) { - let collect_rewards_offset = USER_MAX_CLAIM_WEEKS + 1; - let current_week = self.get_current_week(); - require!( - current_week > collect_rewards_offset, - "Current week must be higher than the week offset" - ); - require!(start_week <= end_week, "Invalid week numbers"); - require!( - end_week <= current_week - collect_rewards_offset, - "Invalid end week" - ); - - let all_tokens = self.all_tokens().get(); - let mut all_rewards = PaymentsVec::new(); - for token_id in &all_tokens { - all_rewards.push(EsdtTokenPayment::new(token_id, 0, BigUint::zero())); - } - - self.accumulate_remaining_rewards(&mut all_rewards, &all_tokens, start_week, end_week); - - for reward_entry in &all_rewards { - if reward_entry.amount == 0 { - continue; - } - - self.accumulated_fees(current_week, &reward_entry.token_identifier) - .update(|acc_fees| *acc_fees += reward_entry.amount); - } - } - - fn accumulate_remaining_rewards( - &self, - all_rewards: &mut PaymentsVec, - all_tokens: &ManagedVec, - start_week: Week, - end_week: Week, - ) { - for week in start_week..=end_week { - self.accumulate_remaining_rewards_single_week(all_rewards, all_tokens, week); - } - } - - fn accumulate_remaining_rewards_single_week( - &self, - all_rewards: &mut PaymentsVec, - all_tokens: &ManagedVec, - week: Week, - ) { - let remaining_rewards = self.remaining_rewards(week).take(); - for rem_rew_entry in &remaining_rewards { - if rem_rew_entry.amount == 0 { - continue; - } - - let opt_index = all_tokens.find(&rem_rew_entry.token_identifier); - if opt_index.is_none() { - continue; - } - - let index = unsafe { opt_index.unwrap_unchecked() }; - let mut rew_entry = all_rewards.get_mut(index); - rew_entry.amount += rem_rew_entry.amount; - } - } - fn get_and_clear_accumulated_fees( &self, week: Week, @@ -133,8 +63,4 @@ pub trait FeesAccumulationModule: #[view(getAccumulatedFees)] #[storage_mapper("accumulatedFees")] fn accumulated_fees(&self, week: Week, token: &TokenIdentifier) -> SingleValueMapper; - - #[view(getRemainingRewards)] - #[storage_mapper("remainingRewards")] - fn remaining_rewards(&self, week: Week) -> SingleValueMapper>; } diff --git a/energy-integration/fees-collector/src/lib.rs b/energy-integration/fees-collector/src/lib.rs index ee9bc0a82..2ee36602c 100644 --- a/energy-integration/fees-collector/src/lib.rs +++ b/energy-integration/fees-collector/src/lib.rs @@ -7,6 +7,7 @@ pub mod claim; pub mod config; pub mod events; pub mod fees_accumulation; +pub mod redistribute_rewards; #[multiversx_sc::contract] pub trait FeesCollector: @@ -27,6 +28,7 @@ pub trait FeesCollector: + sc_whitelist_module::SCWhitelistModule + multiversx_sc_modules::only_admin::OnlyAdminModule + claim::ClaimModule + + redistribute_rewards::RedistributeRewardsModule { #[init] fn init( diff --git a/energy-integration/fees-collector/src/redistribute_rewards.rs b/energy-integration/fees-collector/src/redistribute_rewards.rs new file mode 100644 index 000000000..28259de17 --- /dev/null +++ b/energy-integration/fees-collector/src/redistribute_rewards.rs @@ -0,0 +1,75 @@ +use common_types::{PaymentsVec, Week}; +use weekly_rewards_splitting::USER_MAX_CLAIM_WEEKS; + +multiversx_sc::imports!(); + +#[multiversx_sc::module] +pub trait RedistributeRewardsModule: + crate::fees_accumulation::FeesAccumulationModule + + crate::config::ConfigModule + + crate::events::FeesCollectorEventsModule + + week_timekeeping::WeekTimekeepingModule + + multiversx_sc_modules::only_admin::OnlyAdminModule +{ + #[only_admin] + #[endpoint(redistributeRewards)] + fn redistribute_rewards(&self, start_week: Week, end_week: Week) { + let collect_rewards_offset = USER_MAX_CLAIM_WEEKS + 1; + let current_week = self.get_current_week(); + require!( + current_week > collect_rewards_offset, + "Current week must be higher than the week offset" + ); + require!(start_week <= end_week, "Invalid week numbers"); + require!( + end_week <= current_week - collect_rewards_offset, + "Invalid end week" + ); + + let all_tokens = self.all_tokens().get(); + let mut all_rewards = PaymentsVec::new(); + for token_id in &all_tokens { + all_rewards.push(EsdtTokenPayment::new(token_id, 0, BigUint::zero())); + } + + for week in start_week..=end_week { + self.accumulate_remaining_rewards_single_week(&mut all_rewards, &all_tokens, week); + } + + for reward_entry in &all_rewards { + if reward_entry.amount == 0 { + continue; + } + + self.accumulated_fees(current_week, &reward_entry.token_identifier) + .update(|acc_fees| *acc_fees += reward_entry.amount); + } + } + + fn accumulate_remaining_rewards_single_week( + &self, + all_rewards: &mut PaymentsVec, + all_tokens: &ManagedVec, + week: Week, + ) { + let remaining_rewards = self.remaining_rewards(week).take(); + for rem_rew_entry in &remaining_rewards { + if rem_rew_entry.amount == 0 { + continue; + } + + let opt_index = all_tokens.find(&rem_rew_entry.token_identifier); + if opt_index.is_none() { + continue; + } + + let index = unsafe { opt_index.unwrap_unchecked() }; + let mut rew_entry = all_rewards.get_mut(index); + rew_entry.amount += rem_rew_entry.amount; + } + } + + #[view(getRemainingRewards)] + #[storage_mapper("remainingRewards")] + fn remaining_rewards(&self, week: Week) -> SingleValueMapper>; +} diff --git a/energy-integration/fees-collector/wasm/src/lib.rs b/energy-integration/fees-collector/wasm/src/lib.rs index 941bfb1fa..4aab5f1d3 100644 --- a/energy-integration/fees-collector/wasm/src/lib.rs +++ b/energy-integration/fees-collector/wasm/src/lib.rs @@ -37,9 +37,7 @@ multiversx_sc_wasm_adapter::endpoints! { updateEnergyForUser => update_energy_for_user getCurrentClaimProgress => current_claim_progress depositSwapFees => deposit_swap_fees - redistributeRewards => redistribute_rewards getAccumulatedFees => accumulated_fees - getRemainingRewards => remaining_rewards setLockedTokensPerBlock => set_locked_tokens_per_block getLastLockedTokensAddWeek => last_locked_token_add_week getLockedTokensPerBlock => locked_tokens_per_block @@ -63,6 +61,8 @@ multiversx_sc_wasm_adapter::endpoints! { getAdmins => admins claimRewards => claim_rewards_endpoint claimBoostedRewards => claim_boosted_rewards + redistributeRewards => redistribute_rewards + getRemainingRewards => remaining_rewards ) } From 696428e4e42468eb3634706daa48900989bd01bf Mon Sep 17 00:00:00 2001 From: Dorin Marian Iancu Date: Fri, 15 Nov 2024 09:31:45 +0200 Subject: [PATCH 3/4] redistribute initial rewards --- .../src/additional_locked_tokens.rs | 1 - .../fees-collector/src/fees_accumulation.rs | 1 - .../src/redistribute_rewards.rs | 83 +++++++++++++++++-- .../fees-collector/wasm/src/lib.rs | 6 +- 4 files changed, 81 insertions(+), 10 deletions(-) diff --git a/energy-integration/fees-collector/src/additional_locked_tokens.rs b/energy-integration/fees-collector/src/additional_locked_tokens.rs index fa108c3fb..9303a408f 100644 --- a/energy-integration/fees-collector/src/additional_locked_tokens.rs +++ b/energy-integration/fees-collector/src/additional_locked_tokens.rs @@ -10,7 +10,6 @@ pub trait AdditionalLockedTokensModule: + crate::fees_accumulation::FeesAccumulationModule + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule - + multiversx_sc_modules::only_admin::OnlyAdminModule { #[only_owner] #[endpoint(setLockedTokensPerBlock)] diff --git a/energy-integration/fees-collector/src/fees_accumulation.rs b/energy-integration/fees-collector/src/fees_accumulation.rs index 87865e96f..e6eb4ba41 100644 --- a/energy-integration/fees-collector/src/fees_accumulation.rs +++ b/energy-integration/fees-collector/src/fees_accumulation.rs @@ -8,7 +8,6 @@ pub trait FeesAccumulationModule: crate::config::ConfigModule + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule - + multiversx_sc_modules::only_admin::OnlyAdminModule { /// Pair SC will deposit the fees through this endpoint /// Deposits for current week are accessible starting next week diff --git a/energy-integration/fees-collector/src/redistribute_rewards.rs b/energy-integration/fees-collector/src/redistribute_rewards.rs index 28259de17..4378a76d9 100644 --- a/energy-integration/fees-collector/src/redistribute_rewards.rs +++ b/energy-integration/fees-collector/src/redistribute_rewards.rs @@ -1,8 +1,12 @@ -use common_types::{PaymentsVec, Week}; +use common_types::{PaymentsVec, TokenAmountPair, Week}; use weekly_rewards_splitting::USER_MAX_CLAIM_WEEKS; multiversx_sc::imports!(); +const INITIAL_REW_DIST: bool = true; +const INITIAL_REW_NOT_DIST: bool = false; +static INVALID_OFFSET_ERR_MSG: &[u8] = b"Current week must be higher than the week offset"; + #[multiversx_sc::module] pub trait RedistributeRewardsModule: crate::fees_accumulation::FeesAccumulationModule @@ -10,7 +14,32 @@ pub trait RedistributeRewardsModule: + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule + multiversx_sc_modules::only_admin::OnlyAdminModule + + crate::additional_locked_tokens::AdditionalLockedTokensModule { + #[only_admin] + #[endpoint(redistributeInitialRewards)] + fn redistribute_initial_rewards(&self) { + require!( + self.redistributed_initial_rewards().get() == INITIAL_REW_NOT_DIST, + "Initial rewards already distributed" + ); + + let collect_rewards_offset = USER_MAX_CLAIM_WEEKS + 1; + let current_week = self.get_current_week(); + require!( + current_week > collect_rewards_offset, + INVALID_OFFSET_ERR_MSG + ); + + let all_rem_rewards = self.redist_initial_rew(current_week, collect_rewards_offset); + for reward_entry in &all_rem_rewards { + self.accumulated_fees(current_week, &reward_entry.token) + .update(|acc_fees| *acc_fees += reward_entry.amount); + } + + self.redistributed_initial_rewards().set(INITIAL_REW_DIST); + } + #[only_admin] #[endpoint(redistributeRewards)] fn redistribute_rewards(&self, start_week: Week, end_week: Week) { @@ -18,7 +47,7 @@ pub trait RedistributeRewardsModule: let current_week = self.get_current_week(); require!( current_week > collect_rewards_offset, - "Current week must be higher than the week offset" + INVALID_OFFSET_ERR_MSG ); require!(start_week <= end_week, "Invalid week numbers"); require!( @@ -27,9 +56,9 @@ pub trait RedistributeRewardsModule: ); let all_tokens = self.all_tokens().get(); - let mut all_rewards = PaymentsVec::new(); + let mut all_rewards = ManagedVec::new(); for token_id in &all_tokens { - all_rewards.push(EsdtTokenPayment::new(token_id, 0, BigUint::zero())); + all_rewards.push(TokenAmountPair::new(token_id, BigUint::zero())); } for week in start_week..=end_week { @@ -41,14 +70,14 @@ pub trait RedistributeRewardsModule: continue; } - self.accumulated_fees(current_week, &reward_entry.token_identifier) + self.accumulated_fees(current_week, &reward_entry.token) .update(|acc_fees| *acc_fees += reward_entry.amount); } } fn accumulate_remaining_rewards_single_week( &self, - all_rewards: &mut PaymentsVec, + all_rewards: &mut ManagedVec>, all_tokens: &ManagedVec, week: Week, ) { @@ -69,6 +98,48 @@ pub trait RedistributeRewardsModule: } } + fn redist_initial_rew( + &self, + current_week: Week, + collect_rewards_offset: Week, + ) -> ManagedVec> { + let all_tokens = self.all_tokens().get(); + let locked_token_id = self.locked_token_id().get(); + let start_week = current_week - collect_rewards_offset; + let end_week = current_week; + + let mut all_rem_rewards = ManagedVec::new(); + for token_id in &all_tokens { + if token_id == locked_token_id { + continue; + } + + let mut token_balance = self + .blockchain() + .get_sc_balance(&EgldOrEsdtTokenIdentifier::esdt(token_id.clone()), 0); + if token_balance == 0 { + continue; + } + + for week in start_week..=end_week { + let to_dist_week = self.accumulated_fees(week, &token_id).get(); + token_balance -= to_dist_week; + } + + if token_balance == 0 { + continue; + } + + all_rem_rewards.push(TokenAmountPair::new(token_id, token_balance)); + } + + all_rem_rewards + } + + #[view(wereInitialRewardsRedistributed)] + #[storage_mapper("redistributedInitialRewards")] + fn redistributed_initial_rewards(&self) -> SingleValueMapper; + #[view(getRemainingRewards)] #[storage_mapper("remainingRewards")] fn remaining_rewards(&self, week: Week) -> SingleValueMapper>; diff --git a/energy-integration/fees-collector/wasm/src/lib.rs b/energy-integration/fees-collector/wasm/src/lib.rs index 4aab5f1d3..348ce6a33 100644 --- a/energy-integration/fees-collector/wasm/src/lib.rs +++ b/energy-integration/fees-collector/wasm/src/lib.rs @@ -6,9 +6,9 @@ // Init: 1 // Upgrade: 1 -// Endpoints: 43 +// Endpoints: 45 // Async Callback (empty): 1 -// Total number of exported functions: 46 +// Total number of exported functions: 48 #![no_std] @@ -61,7 +61,9 @@ multiversx_sc_wasm_adapter::endpoints! { getAdmins => admins claimRewards => claim_rewards_endpoint claimBoostedRewards => claim_boosted_rewards + redistributeInitialRewards => redistribute_initial_rewards redistributeRewards => redistribute_rewards + wereInitialRewardsRedistributed => redistributed_initial_rewards getRemainingRewards => remaining_rewards ) } From f3f689a88ce3c14fa50f401f8719df233a1e14ec Mon Sep 17 00:00:00 2001 From: Dorin Marian Iancu Date: Fri, 15 Nov 2024 13:17:28 +0200 Subject: [PATCH 4/4] remove feature + test --- .../fees-collector/src/claim.rs | 7 +- .../src/redistribute_rewards.rs | 73 +------ .../tests/fees_collector_rust_test.rs | 194 ++++++++++++++++++ .../tests/fees_collector_test_setup/mod.rs | 5 +- .../fees-collector/wasm/src/lib.rs | 6 +- 5 files changed, 202 insertions(+), 83 deletions(-) diff --git a/energy-integration/fees-collector/src/claim.rs b/energy-integration/fees-collector/src/claim.rs index a4cd5320b..b0994d08b 100644 --- a/energy-integration/fees-collector/src/claim.rs +++ b/energy-integration/fees-collector/src/claim.rs @@ -147,8 +147,6 @@ where let total_rewards = self.collect_and_get_rewards_for_week(sc, week); let remaining_rewards_mapper = sc.remaining_rewards(week); let mut remaining_rewards = remaining_rewards_mapper.get(); - let mut update_storage = false; - for (i, weekly_reward) in total_rewards.iter().enumerate() { let reward_amount = weekly_reward.amount * energy_amount / total_energy; if reward_amount == 0 { @@ -157,7 +155,6 @@ where let mut rem_rew_entry = remaining_rewards.get_mut(i); rem_rew_entry.amount -= &reward_amount; - update_storage = true; user_rewards.push(EsdtTokenPayment::new( weekly_reward.token_identifier, @@ -166,9 +163,7 @@ where )); } - if update_storage { - remaining_rewards_mapper.set(remaining_rewards); - } + remaining_rewards_mapper.set(remaining_rewards); user_rewards } diff --git a/energy-integration/fees-collector/src/redistribute_rewards.rs b/energy-integration/fees-collector/src/redistribute_rewards.rs index 4378a76d9..d3c5d35ce 100644 --- a/energy-integration/fees-collector/src/redistribute_rewards.rs +++ b/energy-integration/fees-collector/src/redistribute_rewards.rs @@ -3,10 +3,6 @@ use weekly_rewards_splitting::USER_MAX_CLAIM_WEEKS; multiversx_sc::imports!(); -const INITIAL_REW_DIST: bool = true; -const INITIAL_REW_NOT_DIST: bool = false; -static INVALID_OFFSET_ERR_MSG: &[u8] = b"Current week must be higher than the week offset"; - #[multiversx_sc::module] pub trait RedistributeRewardsModule: crate::fees_accumulation::FeesAccumulationModule @@ -14,32 +10,7 @@ pub trait RedistributeRewardsModule: + crate::events::FeesCollectorEventsModule + week_timekeeping::WeekTimekeepingModule + multiversx_sc_modules::only_admin::OnlyAdminModule - + crate::additional_locked_tokens::AdditionalLockedTokensModule { - #[only_admin] - #[endpoint(redistributeInitialRewards)] - fn redistribute_initial_rewards(&self) { - require!( - self.redistributed_initial_rewards().get() == INITIAL_REW_NOT_DIST, - "Initial rewards already distributed" - ); - - let collect_rewards_offset = USER_MAX_CLAIM_WEEKS + 1; - let current_week = self.get_current_week(); - require!( - current_week > collect_rewards_offset, - INVALID_OFFSET_ERR_MSG - ); - - let all_rem_rewards = self.redist_initial_rew(current_week, collect_rewards_offset); - for reward_entry in &all_rem_rewards { - self.accumulated_fees(current_week, &reward_entry.token) - .update(|acc_fees| *acc_fees += reward_entry.amount); - } - - self.redistributed_initial_rewards().set(INITIAL_REW_DIST); - } - #[only_admin] #[endpoint(redistributeRewards)] fn redistribute_rewards(&self, start_week: Week, end_week: Week) { @@ -47,7 +18,7 @@ pub trait RedistributeRewardsModule: let current_week = self.get_current_week(); require!( current_week > collect_rewards_offset, - INVALID_OFFSET_ERR_MSG + "Current week must be higher than the week offset" ); require!(start_week <= end_week, "Invalid week numbers"); require!( @@ -98,48 +69,6 @@ pub trait RedistributeRewardsModule: } } - fn redist_initial_rew( - &self, - current_week: Week, - collect_rewards_offset: Week, - ) -> ManagedVec> { - let all_tokens = self.all_tokens().get(); - let locked_token_id = self.locked_token_id().get(); - let start_week = current_week - collect_rewards_offset; - let end_week = current_week; - - let mut all_rem_rewards = ManagedVec::new(); - for token_id in &all_tokens { - if token_id == locked_token_id { - continue; - } - - let mut token_balance = self - .blockchain() - .get_sc_balance(&EgldOrEsdtTokenIdentifier::esdt(token_id.clone()), 0); - if token_balance == 0 { - continue; - } - - for week in start_week..=end_week { - let to_dist_week = self.accumulated_fees(week, &token_id).get(); - token_balance -= to_dist_week; - } - - if token_balance == 0 { - continue; - } - - all_rem_rewards.push(TokenAmountPair::new(token_id, token_balance)); - } - - all_rem_rewards - } - - #[view(wereInitialRewardsRedistributed)] - #[storage_mapper("redistributedInitialRewards")] - fn redistributed_initial_rewards(&self) -> SingleValueMapper; - #[view(getRemainingRewards)] #[storage_mapper("remainingRewards")] fn remaining_rewards(&self, week: Week) -> SingleValueMapper>; diff --git a/energy-integration/fees-collector/tests/fees_collector_rust_test.rs b/energy-integration/fees-collector/tests/fees_collector_rust_test.rs index 11151d2dc..538c04e3e 100644 --- a/energy-integration/fees-collector/tests/fees_collector_rust_test.rs +++ b/energy-integration/fees-collector/tests/fees_collector_rust_test.rs @@ -5,6 +5,7 @@ mod fees_collector_test_setup; use energy_query::Energy; use fees_collector::additional_locked_tokens::{AdditionalLockedTokensModule, BLOCKS_IN_WEEK}; use fees_collector::fees_accumulation::FeesAccumulationModule; +use fees_collector::redistribute_rewards::RedistributeRewardsModule; use fees_collector_test_setup::*; use multiversx_sc::types::{BigInt, EsdtTokenPayment, ManagedVec}; use multiversx_sc_scenario::{ @@ -1419,3 +1420,196 @@ fn additional_locked_tokens_test() { }) .assert_ok(); } + +#[test] +fn redistribute_rewards_test() { + let rust_zero = rust_biguint!(0); + let mut fc_setup = + FeesCollectorSetup::new(fees_collector::contract_obj, energy_factory::contract_obj); + + let first_user = fc_setup.b_mock.create_user_account(&rust_zero); + let second_user = fc_setup.b_mock.create_user_account(&rust_zero); + let third_user = fc_setup.b_mock.create_user_account(&rust_zero); + + fc_setup.set_energy(&first_user, 50, 3_000); + fc_setup.set_energy(&second_user, 50, 9_000); + fc_setup.set_energy(&third_user, 1, 1); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.claim(&first_user).assert_ok(); + fc_setup.claim(&second_user).assert_ok(); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 2 (inactive week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + fc_setup + .b_mock + .execute_query(&fc_setup.fc_wrapper, |sc| { + let mut expected_total_rewards = ManagedVec::new(); + expected_total_rewards.push(EsdtTokenPayment::new( + managed_token_id!(FIRST_TOKEN_ID), + 0, + managed_biguint!(USER_BALANCE / 10), + )); + expected_total_rewards.push(EsdtTokenPayment::new( + managed_token_id!(SECOND_TOKEN_ID), + 0, + managed_biguint!(USER_BALANCE / 20), + )); + assert_eq!(expected_total_rewards, sc.total_rewards_for_week(1).get()); + }) + .assert_ok(); + + // advance to week 3 (inactive week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 4 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 5 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 6 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 7 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 8 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 9 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // advance to week 10 (active week) + fc_setup.advance_week(); + + fc_setup + .deposit(FIRST_TOKEN_ID, USER_BALANCE / 10) + .assert_ok(); + fc_setup + .deposit(SECOND_TOKEN_ID, USER_BALANCE / 20) + .assert_ok(); + + fc_setup.set_energy(&third_user, 1, 1); + fc_setup.claim(&third_user).assert_ok(); + + // redist rewards + fc_setup + .b_mock + .execute_tx( + &fc_setup.owner_address, + &fc_setup.fc_wrapper, + &rust_zero, + |sc| { + sc.redistribute_rewards(1, 5); + + // Rewards were put in current_week storage (i.e. 10) + + let first_token_balance = sc + .accumulated_fees(10, &managed_token_id!(FIRST_TOKEN_ID)) + .get(); + let second_token_balance = sc + .accumulated_fees(10, &managed_token_id!(SECOND_TOKEN_ID)) + .get(); + + // i.e. 6 weeks worth of rewards minus what the third user claimed + assert_eq!( + first_token_balance, + managed_biguint!(599_952_417_140_485_515u64) + ); + assert_eq!( + second_token_balance, + managed_biguint!(299_976_208_570_242_758) + ); + }, + ) + .assert_ok(); +} diff --git a/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs b/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs index a18621604..3bc7d4bca 100644 --- a/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs +++ b/energy-integration/fees-collector/tests/fees_collector_test_setup/mod.rs @@ -149,10 +149,13 @@ where b_mock .execute_tx(&owner_address, &fc_wrapper, &rust_zero, |sc| { + let mut admins = MultiValueEncoded::new(); + admins.push(managed_address!(&owner_address)); + sc.init( managed_token_id!(LOCKED_TOKEN_ID), managed_address!(energy_factory_wrapper.address_ref()), - MultiValueEncoded::new(), + admins, ); let _ = sc diff --git a/energy-integration/fees-collector/wasm/src/lib.rs b/energy-integration/fees-collector/wasm/src/lib.rs index 348ce6a33..4aab5f1d3 100644 --- a/energy-integration/fees-collector/wasm/src/lib.rs +++ b/energy-integration/fees-collector/wasm/src/lib.rs @@ -6,9 +6,9 @@ // Init: 1 // Upgrade: 1 -// Endpoints: 45 +// Endpoints: 43 // Async Callback (empty): 1 -// Total number of exported functions: 48 +// Total number of exported functions: 46 #![no_std] @@ -61,9 +61,7 @@ multiversx_sc_wasm_adapter::endpoints! { getAdmins => admins claimRewards => claim_rewards_endpoint claimBoostedRewards => claim_boosted_rewards - redistributeInitialRewards => redistribute_initial_rewards redistributeRewards => redistribute_rewards - wereInitialRewardsRedistributed => redistributed_initial_rewards getRemainingRewards => remaining_rewards ) }