diff --git a/pallets/admin-utils/src/lib.rs b/pallets/admin-utils/src/lib.rs index 47799f30d..e17aefbe8 100644 --- a/pallets/admin-utils/src/lib.rs +++ b/pallets/admin-utils/src/lib.rs @@ -26,7 +26,7 @@ pub mod pallet { /// Configure the pallet by specifying the parameters and types on which it depends. #[pallet::config] - pub trait Config: frame_system::Config { + pub trait Config: frame_system::Config + pallet_subtensor::Config { /// Because this pallet emits events, it depends on the runtime's definition of an event. type RuntimeEvent: From> + IsType<::RuntimeEvent>; @@ -996,6 +996,18 @@ pub mod pallet { log::info!("ToggleSetWeightsCommitReveal( netuid: {:?} ) ", netuid); Ok(()) } + + /// Sets the [`pallet_subtensor::TotalSubnetLocked`] value. + #[pallet::call_index(50)] + #[pallet::weight(T::WeightInfo::sudo_set_commit_reveal_weights_enabled())] + pub fn sudo_set_total_subnet_locked(origin: OriginFor, amount: u64) -> DispatchResult { + ensure_root(origin)?; + + pallet_subtensor::TotalSubnetLocked::::put(amount); + + log::info!("Set pallet_subtensor::TotalSubnetLocked to {}", amount); + Ok(()) + } } } diff --git a/pallets/admin-utils/tests/tests.rs b/pallets/admin-utils/tests/tests.rs index f87b43e74..8dca4bb34 100644 --- a/pallets/admin-utils/tests/tests.rs +++ b/pallets/admin-utils/tests/tests.rs @@ -1,3 +1,4 @@ +use frame_support::assert_err; use frame_support::assert_ok; use frame_support::sp_runtime::DispatchError; use frame_system::Config; @@ -1178,3 +1179,34 @@ fn test_sudo_set_target_stakes_per_interval() { assert_eq!(SubtensorModule::get_target_stakes_per_interval(), to_be_set); }); } + +#[test] +fn test_set_total_subnet_locked_ok() { + new_test_ext().execute_with(|| { + // Setup + let before = pallet_subtensor::TotalSubnetLocked::::get(); + let new = 1000u64; + assert_ne!(before, new); + assert_ok!(AdminUtils::sudo_set_total_subnet_locked( + RuntimeOrigin::root(), + new + )); + let after = pallet_subtensor::TotalSubnetLocked::::get(); + assert_eq!(after, new); + }); +} + +#[test] +fn test_set_total_subnet_locked_not_sudo() { + new_test_ext().execute_with(|| { + // Setup + let before = pallet_subtensor::TotalSubnetLocked::::get(); + let new = 1000u64; + let who = U256::from(1); + assert_ne!(before, new); + assert_err!( + AdminUtils::sudo_set_total_subnet_locked(RuntimeOrigin::signed(who), new), + DispatchError::BadOrigin + ); + }); +} diff --git a/pallets/subtensor/src/events.rs b/pallets/subtensor/src/events.rs index 47cc9973b..7bafe337e 100644 --- a/pallets/subtensor/src/events.rs +++ b/pallets/subtensor/src/events.rs @@ -132,5 +132,20 @@ mod events { MinDelegateTakeSet(u16), /// the target stakes per interval is set by sudo/admin transaction TargetStakesPerIntervalSet(u64), + /// Total issuance was rejigged + TotalIssuanceRejigged { + /// If Some a signed account, or root + who: Option, + /// The previous pallet total issuance + prev_total_issuance: u64, + /// The new pallet total issuance + new_total_issuance: u64, + /// The total amount of tokens staked + total_stake: u64, + /// The total amount of tokens in accounts + total_account_balances: u64, + /// The total amount of tokens locked in subnets + total_subnet_locked: u64, + }, } } diff --git a/pallets/subtensor/src/lib.rs b/pallets/subtensor/src/lib.rs index b590781f5..504375b54 100644 --- a/pallets/subtensor/src/lib.rs +++ b/pallets/subtensor/src/lib.rs @@ -103,7 +103,8 @@ pub mod pallet { /// Currency type that will be used to place deposits on neurons type Currency: fungible::Balanced - + fungible::Mutate; + + fungible::Mutate + + fungible::Inspect; /// Senate members with members management functions. type SenateMembers: crate::MemberManagement; @@ -322,6 +323,16 @@ pub mod pallet { pub type MinTake = StorageValue<_, u16, ValueQuery, DefaultMinTake>; #[pallet::storage] // --- ITEM ( global_block_emission ) pub type BlockEmission = StorageValue<_, u64, ValueQuery, DefaultBlockEmission>; + + /// The Subtensor [`TotalIssuance`] represents the total issuance of tokens on the Bittensor network. + /// + /// It comprised of three parts: + /// - The total amount of issued tokens, tracked in the TotalIssuance of the Balances pallet + /// - The total amount of tokens staked in the system, tracked in [`TotalStake`] + /// - The total amount of tokens locked up for subnet reg, tracked in [`TotalSubnetLocked`] + /// + /// Eventually, Bittensor should migrate to using Holds afterwhich time we will not require this + /// seperate accounting. #[pallet::storage] // --- ITEM ( total_issuance ) pub type TotalIssuance = StorageValue<_, u64, ValueQuery, DefaultTotalIssuance>; #[pallet::storage] // --- ITEM (target_stakes_per_interval) @@ -655,6 +666,9 @@ pub mod pallet { #[pallet::storage] // --- MAP ( netuid ) --> subnet_locked pub type SubnetLocked = StorageMap<_, Identity, u16, u64, ValueQuery, DefaultSubnetLocked>; + /// The total amount of locked tokens in subnets for subnet registration. + #[pallet::storage] + pub type TotalSubnetLocked = StorageValue<_, u64, ValueQuery>; /// ================================= /// ==== Axon / Promo Endpoints ===== @@ -1287,6 +1301,40 @@ pub mod pallet { weight } + + #[cfg(feature = "try-runtime")] + fn try_state(_n: BlockNumberFor) -> Result<(), sp_runtime::TryRuntimeError> { + use frame_support::traits::fungible::Inspect; + + // Assert [`TotalStake`] accounting is correct + let mut total_staked = 0; + for stake in Stake::::iter() { + total_staked += stake.2; + } + ensure!( + total_staked == TotalStake::::get(), + "TotalStake does not match total staked" + ); + + // Assert [`TotalSubnetLocked`] accounting is correct + let mut total_subnet_locked = 0; + for (_, locked) in SubnetLocked::::iter() { + total_subnet_locked += locked; + } + ensure!( + total_subnet_locked == TotalSubnetLocked::::get(), + "TotalSubnetLocked does not match total subnet locked" + ); + + // Assert [`TotalIssuance`] accounting is correct + let currency_issuance = T::Currency::total_issuance(); + ensure!( + TotalIssuance::::get() == currency_issuance + total_staked + total_subnet_locked, + "TotalIssuance accounting discrepancy" + ); + + Ok(()) + } } /// Dispatchable functions allow users to interact with the pallet and invoke state changes. @@ -2051,6 +2099,42 @@ pub mod pallet { pub fn dissolve_network(origin: OriginFor, netuid: u16) -> DispatchResult { Self::user_remove_network(origin, netuid) } + + /// Set the [`TotalIssuance`] storage value to the total account balances issued + the + /// total amount staked + the total amount locked in subnets. + #[pallet::call_index(63)] + #[pallet::weight(( + Weight::default() + .saturating_add(T::DbWeight::get().reads(3)) + .saturating_add(T::DbWeight::get().writes(1)), + DispatchClass::Normal, + Pays::Yes + ))] + pub fn rejig_total_issuance(origin: OriginFor) -> DispatchResult { + let who = ensure_signed_or_root(origin)?; + + let total_account_balances = + >::total_issuance(); + let total_stake = TotalStake::::get(); + let total_subnet_locked = TotalSubnetLocked::::get(); + + let prev_total_issuance = TotalIssuance::::get(); + let new_total_issuance = total_account_balances + .saturating_add(total_stake) + .saturating_add(total_subnet_locked); + TotalIssuance::::put(new_total_issuance); + + Self::deposit_event(Event::TotalIssuanceRejigged { + who, + prev_total_issuance, + new_total_issuance, + total_stake, + total_account_balances, + total_subnet_locked, + }); + + Ok(()) + } } // ---- Subtensor helper functions. diff --git a/pallets/subtensor/src/utils.rs b/pallets/subtensor/src/utils.rs index 54b7818c9..00648258a 100644 --- a/pallets/subtensor/src/utils.rs +++ b/pallets/subtensor/src/utils.rs @@ -324,6 +324,13 @@ impl Pallet { } pub fn set_subnet_locked_balance(netuid: u16, amount: u64) { + let prev_total = TotalSubnetLocked::::get(); + let prev = SubnetLocked::::get(netuid); + + // Deduct the prev amount and add the new amount to the total + TotalSubnetLocked::::put(prev_total.saturating_sub(prev).saturating_add(amount)); + + // Finally, set the new amount SubnetLocked::::insert(netuid, amount); } diff --git a/pallets/subtensor/tests/root.rs b/pallets/subtensor/tests/root.rs index 7958c9c81..cfedf8038 100644 --- a/pallets/subtensor/tests/root.rs +++ b/pallets/subtensor/tests/root.rs @@ -1,4 +1,5 @@ use crate::mock::*; +use frame_support::traits::fungible::Mutate; use frame_support::{assert_err, assert_ok}; use frame_system::Config; use frame_system::{EventRecord, Phase}; @@ -972,3 +973,82 @@ fn test_dissolve_network_does_not_exist_err() { ); }); } + +#[test] +fn test_rejig_total_issuance_ok() { + new_test_ext(1).execute_with(|| { + // Setup + let who = U256::from(1); + Balances::set_balance(&who, 100); + let balances_total_issuance = Balances::total_issuance(); + assert!(balances_total_issuance > 0); + let total_stake = 100; + let total_subnet_locked = 1000; + pallet_subtensor::TotalSubnetLocked::::put(total_subnet_locked); + pallet_subtensor::TotalStake::::put(total_stake); + + let expected_total_issuance = balances_total_issuance + total_stake + total_subnet_locked; + + // Rejig total issuance + let total_issuance_before = pallet_subtensor::TotalIssuance::::get(); + assert_ne!(total_issuance_before, expected_total_issuance); + assert_ok!(SubtensorModule::rejig_total_issuance( + RuntimeOrigin::signed(who) + )); + let total_issuance_after = pallet_subtensor::TotalIssuance::::get(); + + // Rejigged + assert_eq!(total_issuance_after, expected_total_issuance); + System::assert_last_event(RuntimeEvent::SubtensorModule( + pallet_subtensor::Event::TotalIssuanceRejigged { + who: Some(who), + new_total_issuance: total_issuance_after, + prev_total_issuance: total_issuance_before, + total_account_balances: balances_total_issuance, + total_stake, + total_subnet_locked, + }, + )); + + // Works with root + assert_ok!(SubtensorModule::rejig_total_issuance(RuntimeOrigin::root())); + System::assert_last_event(RuntimeEvent::SubtensorModule( + pallet_subtensor::Event::TotalIssuanceRejigged { + who: None, + new_total_issuance: total_issuance_after, + prev_total_issuance: total_issuance_after, + total_account_balances: balances_total_issuance, + total_stake, + total_subnet_locked, + }, + )); + }); +} + +#[test] +fn test_set_subnet_locked_balance_adjusts_total_subnet_locked_correctly() { + new_test_ext(1).execute_with(|| { + // Asset starting from zero + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 0u64); + + // Locking on one subnet replaces its lock + SubtensorModule::set_subnet_locked_balance(0u16, 100u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 100u64); + SubtensorModule::set_subnet_locked_balance(0u16, 1500u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 1500u64); + SubtensorModule::set_subnet_locked_balance(0u16, 1000u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 1000u64); + + // Locking on an additional subnet is additive + SubtensorModule::set_subnet_locked_balance(1u16, 100u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 1100u64); + SubtensorModule::set_subnet_locked_balance(1u16, 1000u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 2000u64); + + // We can go all the way back to zero + SubtensorModule::set_subnet_locked_balance(0u16, 0u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 1000u64); + SubtensorModule::set_subnet_locked_balance(1u16, 0u64); + assert_eq!(pallet_subtensor::TotalSubnetLocked::::get(), 0u64); + }); +} diff --git a/runtime/src/lib.rs b/runtime/src/lib.rs index 2364008fd..63d17103b 100644 --- a/runtime/src/lib.rs +++ b/runtime/src/lib.rs @@ -10,13 +10,15 @@ pub mod check_nonce; mod migrations; use codec::{Decode, Encode, MaxEncodedLen}; +use frame_support::traits::Imbalance; use frame_support::{ dispatch::DispatchResultWithPostInfo, genesis_builder_helper::{build_config, create_default_config}, pallet_prelude::{DispatchError, Get}, - traits::{fungible::HoldConsideration, LinearStoragePrice}, + traits::{fungible::HoldConsideration, LinearStoragePrice, OnUnbalanced}, }; use frame_system::{EnsureNever, EnsureRoot, RawOrigin}; +use pallet_balances::NegativeImbalance; use pallet_commitments::CanCommit; use pallet_grandpa::{ fg_primitives, AuthorityId as GrandpaId, AuthorityList as GrandpaAuthorityList, @@ -333,10 +335,21 @@ parameter_types! { pub FeeMultiplier: Multiplier = Multiplier::one(); } +/// Deduct the transaction fee from the Subtensor Pallet TotalIssuance when dropping the transaction +/// fee. +pub struct TransactionFeeHandler; +impl OnUnbalanced> for TransactionFeeHandler { + fn on_nonzero_unbalanced(credit: NegativeImbalance) { + let ti_before = pallet_subtensor::TotalIssuance::::get(); + pallet_subtensor::TotalIssuance::::put(ti_before.saturating_sub(credit.peek())); + drop(credit); + } +} + impl pallet_transaction_payment::Config for Runtime { type RuntimeEvent = RuntimeEvent; - type OnChargeTransaction = CurrencyAdapter; + type OnChargeTransaction = CurrencyAdapter; //type TransactionByteFee = TransactionByteFee; // Convert dispatch weight to a chargeable fee.