From 836eab509167d481024f34c04d051bf98dc5efc6 Mon Sep 17 00:00:00 2001 From: Greg Zaitsev Date: Thu, 19 Mar 2026 15:43:26 -0400 Subject: [PATCH 1/4] Introduce palletId-based subnet account Ids --- chain-extensions/src/mock.rs | 2 ++ docs/subnet_account_ids.md | 25 +++++++++++++++++++++++ pallets/admin-utils/src/tests/mock.rs | 2 ++ pallets/subtensor/rpc/src/lib.rs | 18 ++++++++++++++++ pallets/subtensor/runtime-api/src/lib.rs | 1 + pallets/subtensor/src/macros/config.rs | 4 ++++ pallets/subtensor/src/subnets/subnet.rs | 9 ++++++++ pallets/subtensor/src/tests/mock.rs | 2 ++ pallets/subtensor/src/tests/subnet.rs | 24 ++++++++++++++++++++++ pallets/transaction-fee/src/tests/mock.rs | 2 ++ runtime/src/lib.rs | 6 ++++++ 11 files changed, 95 insertions(+) create mode 100644 docs/subnet_account_ids.md diff --git a/chain-extensions/src/mock.rs b/chain-extensions/src/mock.rs index 5c9729250f..e00254a603 100644 --- a/chain-extensions/src/mock.rs +++ b/chain-extensions/src/mock.rs @@ -345,6 +345,7 @@ parameter_types! { pub const LeaseDividendsDistributionInterval: u32 = 100; pub const MaxImmuneUidsPercentage: Percent = Percent::from_percent(80); pub const EvmKeyAssociateRateLimit: u64 = 10; + pub const SubtensorPalletId: PalletId = PalletId(*b"subtensr"); } impl pallet_subtensor::Config for Test { @@ -420,6 +421,7 @@ impl pallet_subtensor::Config for Test { type CommitmentsInterface = CommitmentsI; type EvmKeyAssociateRateLimit = EvmKeyAssociateRateLimit; type AuthorshipProvider = MockAuthorshipProvider; + type SubtensorPalletId = SubtensorPalletId; } // Swap-related parameter types diff --git a/docs/subnet_account_ids.md b/docs/subnet_account_ids.md new file mode 100644 index 0000000000..a0e2984081 --- /dev/null +++ b/docs/subnet_account_ids.md @@ -0,0 +1,25 @@ +# The subnet account IDs + + 0: 5EYCAe5jLQhn6ofDSvqF6iY53erXNkwhyE1aCEgvi1NNs91F + 1: 5EYCAe5jLQhn6ofDSvqWqk5fA9XiqK3ahtx5kBNmAqF78mqL + 2: 5EYCAe5jLQhn6ofDSvqnamdFGeCvHs9TSZtbJ84bdf7qQRc6 + 3: 5EYCAe5jLQhn6ofDSvr4KoAqP8t7kRFLBEq6r4kS6UzZgCb5 + 4: 5EYCAe5jLQhn6ofDSvrL4piRVdZKCyMCuumcQ1SGZJsHwmeE + 5: 5EYCAe5jLQhn6ofDSvrborG1c8EWfXT5eai7wx8728k2DHK7 + 6: 5EYCAe5jLQhn6ofDSvrsYsobicui85YxPFedVtowUxckUuF8 + 7: 5EYCAe5jLQhn6ofDSvs9HuMBq7auadeq7vb93qVmwnVUkg5A + 8: 5EYCAe5jLQhn6ofDSvsR2vtmwcG73BkhrbXebnBcQcND2Bdh + 9: 5EYCAe5jLQhn6ofDSvsgmxSN46wJVjrabGUA9isSsSEwHnFy + 10: 5EYCAe5jLQhn6ofDSvsxWyyxAbcVxHxTKwQfhfZHLG7fZUJG + 11: 5EYCAe5jLQhn6ofDSvtEG1XYH6HhQr4L4cMBFcF7o5zPpyA3 + 12: 5EYCAe5jLQhn6ofDSvtW1358PaxtsQACoHHgoYvxFus86kJK + 13: 5EYCAe5jLQhn6ofDSvtmk4ciW5e6KxG5XxECMVcnijjrN8rz + 14: 5EYCAe5jLQhn6ofDSvu3V6AJcaKHnWMxGdAhuSJdBZcadwDn + 15: 5EYCAe5jLQhn6ofDSvuKE7htj4zVF4Tq1J7DTNzTePVJucfX + 16: 5EYCAe5jLQhn6ofDSvuay9FUqZfghcZhjy3j1KgJ7DN3BDc2 + 17: 5EYCAe5jLQhn6ofDSvuriAo4x4LtAAfaUdzEZGN8a3EmSncG + 18: 5EYCAe5jLQhn6ofDSvv8TCLf4Z25cimTDJvk7D3y2s7ViZEm + 19: 5EYCAe5jLQhn6ofDSvvQCDtFB3hH5GsKwysFf9joVgzDytnb + 20: 5EYCAe5jLQhn6ofDSvvfwFRqHYNUXpyCgeomD6RdxWrxFpQR + + diff --git a/pallets/admin-utils/src/tests/mock.rs b/pallets/admin-utils/src/tests/mock.rs index f744a4ea39..6c19176225 100644 --- a/pallets/admin-utils/src/tests/mock.rs +++ b/pallets/admin-utils/src/tests/mock.rs @@ -156,6 +156,7 @@ parameter_types! { pub const LeaseDividendsDistributionInterval: u32 = 100; // 100 blocks pub const MaxImmuneUidsPercentage: Percent = Percent::from_percent(80); pub const EvmKeyAssociateRateLimit: u64 = 0; + pub const SubtensorPalletId: PalletId = PalletId(*b"subtensr"); } impl pallet_subtensor::Config for Test { @@ -231,6 +232,7 @@ impl pallet_subtensor::Config for Test { type CommitmentsInterface = CommitmentsI; type EvmKeyAssociateRateLimit = EvmKeyAssociateRateLimit; type AuthorshipProvider = MockAuthorshipProvider; + type SubtensorPalletId = SubtensorPalletId; } parameter_types! { diff --git a/pallets/subtensor/rpc/src/lib.rs b/pallets/subtensor/rpc/src/lib.rs index 98e2df2f62..c5320bcafa 100644 --- a/pallets/subtensor/rpc/src/lib.rs +++ b/pallets/subtensor/rpc/src/lib.rs @@ -109,6 +109,8 @@ pub trait SubtensorCustomApi { ) -> RpcResult>; #[method(name = "subnetInfo_getSubnetToPrune")] fn get_subnet_to_prune(&self, at: Option) -> RpcResult>; + #[method(name = "subnetInfo_getSubnetAccountId")] + fn get_subnet_account_id(&self, netuid: NetUid, at: Option) -> RpcResult>; } pub struct SubtensorCustom { @@ -531,4 +533,20 @@ where } } } + + fn get_subnet_account_id( + &self, + netuid: NetUid, + at: Option<::Hash>, + ) -> RpcResult> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| self.client.info().best_hash); + + match api.get_subnet_account_id(at, netuid) { + Ok(result) => Ok(result.encode()), + Err(_) => { + Err(Error::RuntimeError(format!("Subnet does not exist")).into()) + } + } + } } diff --git a/pallets/subtensor/runtime-api/src/lib.rs b/pallets/subtensor/runtime-api/src/lib.rs index 84da95cd36..ece243e9cb 100644 --- a/pallets/subtensor/runtime-api/src/lib.rs +++ b/pallets/subtensor/runtime-api/src/lib.rs @@ -48,6 +48,7 @@ sp_api::decl_runtime_apis! { fn get_coldkey_auto_stake_hotkey(coldkey: AccountId32, netuid: NetUid) -> Option; fn get_selective_mechagraph(netuid: NetUid, subid: MechId, metagraph_indexes: Vec) -> Option>; fn get_subnet_to_prune() -> Option; + fn get_subnet_account_id(netuid: NetUid) -> Option; } pub trait StakeInfoRuntimeApi { diff --git a/pallets/subtensor/src/macros/config.rs b/pallets/subtensor/src/macros/config.rs index 1537de65b3..5cb9bb02aa 100644 --- a/pallets/subtensor/src/macros/config.rs +++ b/pallets/subtensor/src/macros/config.rs @@ -7,6 +7,7 @@ use frame_support::pallet_macros::pallet_section; mod config { use crate::{CommitmentsInterface, GetAlphaForTao, GetTaoForAlpha}; + use frame_support::PalletId; use pallet_commitments::GetCommitments; use subtensor_runtime_common::AuthorshipInfo; use subtensor_swap_interface::{SwapEngine, SwapHandler}; @@ -250,5 +251,8 @@ mod config { /// Maximum percentage of immune UIDs. #[pallet::constant] type MaxImmuneUidsPercentage: Get; + /// Pallet account ID + #[pallet::constant] + type SubtensorPalletId: Get; } } diff --git a/pallets/subtensor/src/subnets/subnet.rs b/pallets/subtensor/src/subnets/subnet.rs index 769db17ebe..489c07031c 100644 --- a/pallets/subtensor/src/subnets/subnet.rs +++ b/pallets/subtensor/src/subnets/subnet.rs @@ -1,5 +1,6 @@ use super::*; use sp_core::Get; +use sp_runtime::traits::AccountIdConversion; use subtensor_runtime_common::{NetUid, TaoBalance}; impl Pallet { /// Returns true if the subnetwork exists. @@ -438,4 +439,12 @@ impl Pallet { pub fn is_valid_subnet_for_emission(netuid: NetUid) -> bool { FirstEmissionBlockNumber::::get(netuid).is_some() } + + pub fn get_subnet_account_id(netuid: NetUid) -> Option { + if NetworksAdded::::contains_key(netuid) { + Some(T::SubtensorPalletId::get().into_sub_account_truncating(u16::from(netuid))) + } else { + None + } + } } diff --git a/pallets/subtensor/src/tests/mock.rs b/pallets/subtensor/src/tests/mock.rs index 772909638b..10dd50338d 100644 --- a/pallets/subtensor/src/tests/mock.rs +++ b/pallets/subtensor/src/tests/mock.rs @@ -242,6 +242,7 @@ parameter_types! { pub const LeaseDividendsDistributionInterval: u32 = 100; pub const MaxImmuneUidsPercentage: Percent = Percent::from_percent(80); pub const EvmKeyAssociateRateLimit: u64 = 10; + pub const SubtensorPalletId: PalletId = PalletId(*b"subtensr"); } impl crate::Config for Test { @@ -317,6 +318,7 @@ impl crate::Config for Test { type CommitmentsInterface = CommitmentsI; type EvmKeyAssociateRateLimit = EvmKeyAssociateRateLimit; type AuthorshipProvider = MockAuthorshipProvider; + type SubtensorPalletId = SubtensorPalletId; } // Swap-related parameter types diff --git a/pallets/subtensor/src/tests/subnet.rs b/pallets/subtensor/src/tests/subnet.rs index dd5016a32f..098955ae46 100644 --- a/pallets/subtensor/src/tests/subnet.rs +++ b/pallets/subtensor/src/tests/subnet.rs @@ -888,3 +888,27 @@ fn test_update_symbol_fails_if_symbol_already_in_use() { ); }); } + +// cargo test --package pallet-subtensor --lib -- tests::subnet::test_get_subnet_account_id_exists_and_is_distinct_for_257_consecutive_subnets --exact --nocapture +#[test] +fn test_get_subnet_account_id_exists_and_is_distinct_for_257_consecutive_subnets() { + new_test_ext(1).execute_with(|| { + use std::collections::BTreeSet; + + let mut account_ids = BTreeSet::new(); + + for raw_netuid in 0u16..=256u16 { + let netuid = NetUid::from(raw_netuid); + add_network(netuid, 10, 0); + + let account_id = SubtensorModule::get_subnet_account_id(netuid); + assert!( + account_ids.insert(account_id), + "duplicate subnet account id for netuid {:?}", + netuid + ); + } + + assert_eq!(account_ids.len(), 257); + }); +} diff --git a/pallets/transaction-fee/src/tests/mock.rs b/pallets/transaction-fee/src/tests/mock.rs index 1b4eac0706..2129220165 100644 --- a/pallets/transaction-fee/src/tests/mock.rs +++ b/pallets/transaction-fee/src/tests/mock.rs @@ -229,6 +229,7 @@ parameter_types! { pub const LeaseDividendsDistributionInterval: u32 = 100; // 100 blocks pub const MaxImmuneUidsPercentage: Percent = Percent::from_percent(80); pub const EvmKeyAssociateRateLimit: u64 = 0; + pub const SubtensorPalletId: PalletId = PalletId(*b"subtensr"); } impl pallet_subtensor::Config for Test { @@ -304,6 +305,7 @@ impl pallet_subtensor::Config for Test { type CommitmentsInterface = CommitmentsI; type EvmKeyAssociateRateLimit = EvmKeyAssociateRateLimit; type AuthorshipProvider = MockAuthorshipProvider; + type SubtensorPalletId = SubtensorPalletId; } parameter_types! { diff --git a/runtime/src/lib.rs b/runtime/src/lib.rs index 9c9697bed6..1ed433a0f4 100644 --- a/runtime/src/lib.rs +++ b/runtime/src/lib.rs @@ -1121,6 +1121,7 @@ parameter_types! { pub const LeaseDividendsDistributionInterval: BlockNumber = 100; // 100 blocks pub const MaxImmuneUidsPercentage: Percent = Percent::from_percent(80); pub const EvmKeyAssociateRateLimit: u64 = EVM_KEY_ASSOCIATE_RATELIMIT; + pub const SubtensorPalletId: PalletId = PalletId(*b"subtensr"); } impl pallet_subtensor::Config for Runtime { @@ -1196,6 +1197,7 @@ impl pallet_subtensor::Config for Runtime { type CommitmentsInterface = CommitmentsI; type EvmKeyAssociateRateLimit = EvmKeyAssociateRateLimit; type AuthorshipProvider = BlockAuthorFromAura; + type SubtensorPalletId = SubtensorPalletId; } parameter_types! { @@ -2486,6 +2488,10 @@ impl_runtime_apis! { fn get_selective_mechagraph(netuid: NetUid, mecid: MechId, metagraph_indexes: Vec) -> Option> { SubtensorModule::get_selective_mechagraph(netuid, mecid, metagraph_indexes) } + + fn get_subnet_account_id(netuid: NetUid) -> Option { + SubtensorModule::get_subnet_account_id(netuid) + } } impl subtensor_custom_rpc_runtime_api::StakeInfoRuntimeApi for Runtime { From e97adb8a50177b50cb6152a4ae54f49c601edffe Mon Sep 17 00:00:00 2001 From: Greg Zaitsev Date: Fri, 20 Mar 2026 18:38:30 -0400 Subject: [PATCH 2/4] Forbid use of subnet account IDs as hotkey and as subnet owner hotkey --- pallets/subtensor/rpc/src/lib.rs | 4 +- pallets/subtensor/src/benchmarks.rs | 10 +-- pallets/subtensor/src/coinbase/root.rs | 2 +- pallets/subtensor/src/macros/dispatches.rs | 2 +- .../subtensor/src/migrations/migrate_rao.rs | 2 +- pallets/subtensor/src/staking/account.rs | 2 +- pallets/subtensor/src/staking/helpers.rs | 22 ++++- pallets/subtensor/src/subnets/leasing.rs | 2 +- pallets/subtensor/src/subnets/registration.rs | 4 +- pallets/subtensor/src/subnets/subnet.rs | 22 ++++- pallets/subtensor/src/swap/swap_coldkey.rs | 10 ++- pallets/subtensor/src/swap/swap_hotkey.rs | 6 +- pallets/subtensor/src/tests/children.rs | 2 +- pallets/subtensor/src/tests/evm.rs | 10 +-- pallets/subtensor/src/tests/leasing.rs | 8 +- pallets/subtensor/src/tests/move_stake.rs | 85 ++++++++++--------- pallets/subtensor/src/tests/recycle_alpha.rs | 16 ++-- pallets/subtensor/src/tests/staking.rs | 14 +-- pallets/subtensor/src/tests/subnet.rs | 41 ++++++++- pallets/subtensor/src/utils/misc.rs | 9 +- runtime/tests/account_conversion.rs | 53 ++++++++++++ 21 files changed, 231 insertions(+), 95 deletions(-) create mode 100644 runtime/tests/account_conversion.rs diff --git a/pallets/subtensor/rpc/src/lib.rs b/pallets/subtensor/rpc/src/lib.rs index c5320bcafa..6feff774ad 100644 --- a/pallets/subtensor/rpc/src/lib.rs +++ b/pallets/subtensor/rpc/src/lib.rs @@ -544,9 +544,7 @@ where match api.get_subnet_account_id(at, netuid) { Ok(result) => Ok(result.encode()), - Err(_) => { - Err(Error::RuntimeError(format!("Subnet does not exist")).into()) - } + Err(_) => Err(Error::RuntimeError("Subnet does not exist".to_string()).into()), } } } diff --git a/pallets/subtensor/src/benchmarks.rs b/pallets/subtensor/src/benchmarks.rs index 98bb64c263..4d29708332 100644 --- a/pallets/subtensor/src/benchmarks.rs +++ b/pallets/subtensor/src/benchmarks.rs @@ -857,7 +857,7 @@ mod pallet_benchmarks { let alpha_to_move = Subtensor::::get_stake_for_hotkey_and_coldkey_on_subnet(&origin, &coldkey, netuid); - Subtensor::::create_account_if_non_existent(&coldkey, &destination); + let _ = Subtensor::::create_account_if_non_existent(&coldkey, &destination); // Remove stake limit for benchmark StakingOperationRateLimiter::::remove((origin.clone(), coldkey.clone(), netuid)); @@ -1038,7 +1038,7 @@ mod pallet_benchmarks { let alpha_to_transfer = Subtensor::::get_stake_for_hotkey_and_coldkey_on_subnet(&hot, &coldkey, netuid); - Subtensor::::create_account_if_non_existent(&dest, &hot); + let _ = Subtensor::::create_account_if_non_existent(&dest, &hot); // Remove stake limit for benchmark StakingOperationRateLimiter::::remove((hot.clone(), coldkey.clone(), netuid)); @@ -1278,7 +1278,7 @@ mod pallet_benchmarks { let descr = vec![]; let add = vec![]; - Subtensor::::create_account_if_non_existent(&coldkey, &hotkey); + let _ = Subtensor::::create_account_if_non_existent(&coldkey, &hotkey); Subtensor::::init_new_network(1.into(), 1); let deposit: u64 = 1_000_000_000u64.saturating_mul(2); Subtensor::::add_balance_to_coldkey_account(&coldkey, deposit.into()); @@ -1365,7 +1365,7 @@ mod pallet_benchmarks { fn unstake_all() { let coldkey: T::AccountId = whitelisted_caller(); let hotkey: T::AccountId = account("A", 0, 14); - Subtensor::::create_account_if_non_existent(&coldkey, &hotkey); + let _ = Subtensor::::create_account_if_non_existent(&coldkey, &hotkey); #[extrinsic_call] _(RawOrigin::Signed(coldkey.clone()), hotkey.clone()); @@ -1608,7 +1608,7 @@ mod pallet_benchmarks { let lease_id = 0; let lease = SubnetLeases::::get(0).unwrap(); let hotkey = account::("beneficiary_hotkey", 0, 0); - Subtensor::::create_account_if_non_existent(&beneficiary, &hotkey); + let _ = Subtensor::::create_account_if_non_existent(&beneficiary, &hotkey); #[extrinsic_call] _( RawOrigin::Signed(beneficiary.clone()), diff --git a/pallets/subtensor/src/coinbase/root.rs b/pallets/subtensor/src/coinbase/root.rs index e2714fba1b..5f5d2878f4 100644 --- a/pallets/subtensor/src/coinbase/root.rs +++ b/pallets/subtensor/src/coinbase/root.rs @@ -110,7 +110,7 @@ impl Pallet { ); // --- 6. Create a network account for the user if it doesn't exist. - Self::create_account_if_non_existent(&coldkey, &hotkey); + Self::create_account_if_non_existent(&coldkey, &hotkey)?; // --- 7. Fetch the current size of the subnetwork. let current_num_root_validators: u16 = Self::get_num_root_validators(); diff --git a/pallets/subtensor/src/macros/dispatches.rs b/pallets/subtensor/src/macros/dispatches.rs index 2ba2c87e16..ddb41b211c 100644 --- a/pallets/subtensor/src/macros/dispatches.rs +++ b/pallets/subtensor/src/macros/dispatches.rs @@ -1859,7 +1859,7 @@ mod dispatches { ) -> DispatchResult { let coldkey = ensure_signed(origin)?; - let _ = Self::do_try_associate_hotkey(&coldkey, &hotkey); + Self::do_try_associate_hotkey(&coldkey, &hotkey)?; Ok(()) } diff --git a/pallets/subtensor/src/migrations/migrate_rao.rs b/pallets/subtensor/src/migrations/migrate_rao.rs index 6092d41c6f..53b0312139 100644 --- a/pallets/subtensor/src/migrations/migrate_rao.rs +++ b/pallets/subtensor/src/migrations/migrate_rao.rs @@ -108,7 +108,7 @@ pub fn migrate_rao() -> Weight { // Set Owner as the coldkey. SubnetOwnerHotkey::::insert(netuid, owner_coldkey.clone()); // Associate the coldkey to coldkey. - Pallet::::create_account_if_non_existent(&owner_coldkey, &owner_coldkey); + let _ = Pallet::::create_account_if_non_existent(&owner_coldkey, &owner_coldkey); // Only register the owner coldkey if it's not already a hotkey on the subnet. if !Uids::::contains_key(*netuid, &owner_coldkey) { diff --git a/pallets/subtensor/src/staking/account.rs b/pallets/subtensor/src/staking/account.rs index 20a6ff3036..3252c0836f 100644 --- a/pallets/subtensor/src/staking/account.rs +++ b/pallets/subtensor/src/staking/account.rs @@ -6,7 +6,7 @@ impl Pallet { hotkey: &T::AccountId, ) -> DispatchResult { // Ensure the hotkey is not already associated with a coldkey - Self::create_account_if_non_existent(coldkey, hotkey); + Self::create_account_if_non_existent(coldkey, hotkey)?; Ok(()) } diff --git a/pallets/subtensor/src/staking/helpers.rs b/pallets/subtensor/src/staking/helpers.rs index 5c785f199b..42344faed5 100644 --- a/pallets/subtensor/src/staking/helpers.rs +++ b/pallets/subtensor/src/staking/helpers.rs @@ -132,7 +132,16 @@ impl Pallet { // Creates a cold - hot pairing account if the hotkey is not already an active account. // - pub fn create_account_if_non_existent(coldkey: &T::AccountId, hotkey: &T::AccountId) { + pub fn create_account_if_non_existent( + coldkey: &T::AccountId, + hotkey: &T::AccountId, + ) -> DispatchResult { + // Only allow to register non-system hotkeys + ensure!( + Self::is_subnet_account_id(hotkey).is_none(), + Error::::NonAssociatedColdKey + ); + if !Self::hotkey_account_exists(hotkey) { Owner::::insert(hotkey, coldkey); @@ -150,6 +159,17 @@ impl Pallet { StakingHotkeys::::insert(coldkey, staking_hotkeys); } } + Ok(()) + } + + pub fn set_hotkey_owner(coldkey: &T::AccountId, hotkey: &T::AccountId) -> DispatchResult { + // Only allow to register non-system hotkeys + ensure!( + Self::is_subnet_account_id(hotkey).is_none(), + Error::::NonAssociatedColdKey + ); + Owner::::insert(hotkey, coldkey); + Ok(()) } //// If the hotkey is not a delegate, make it a delegate. diff --git a/pallets/subtensor/src/subnets/leasing.rs b/pallets/subtensor/src/subnets/leasing.rs index e7c10bf8ff..9e29931d8c 100644 --- a/pallets/subtensor/src/subnets/leasing.rs +++ b/pallets/subtensor/src/subnets/leasing.rs @@ -218,7 +218,7 @@ impl Pallet { Error::::BeneficiaryDoesNotOwnHotkey ); SubnetOwner::::insert(lease.netuid, lease.beneficiary.clone()); - Self::set_subnet_owner_hotkey(lease.netuid, &hotkey); + Self::set_subnet_owner_hotkey(lease.netuid, &hotkey)?; // Stop tracking the lease coldkey and hotkey let _ = frame_system::Pallet::::dec_providers(&lease.coldkey).defensive(); diff --git a/pallets/subtensor/src/subnets/registration.rs b/pallets/subtensor/src/subnets/registration.rs index ccf32f6ac7..17f6b5da05 100644 --- a/pallets/subtensor/src/subnets/registration.rs +++ b/pallets/subtensor/src/subnets/registration.rs @@ -111,7 +111,7 @@ impl Pallet { ); // If the network account does not exist we will create it here. - Self::create_account_if_non_existent(&coldkey, &hotkey); + Self::create_account_if_non_existent(&coldkey, &hotkey)?; // --- 8. Ensure that the pairing is correct. ensure!( @@ -294,7 +294,7 @@ impl Pallet { UsedWork::::insert(work.clone(), current_block_number); // --- 10. If the network account does not exist we will create it here. - Self::create_account_if_non_existent(&coldkey, &hotkey); + Self::create_account_if_non_existent(&coldkey, &hotkey)?; // --- 11. Ensure that the pairing is correct. ensure!( diff --git a/pallets/subtensor/src/subnets/subnet.rs b/pallets/subtensor/src/subnets/subnet.rs index 489c07031c..81bb3fc80a 100644 --- a/pallets/subtensor/src/subnets/subnet.rs +++ b/pallets/subtensor/src/subnets/subnet.rs @@ -1,4 +1,5 @@ use super::*; +use frame_support::PalletId; use sp_core::Get; use sp_runtime::traits::AccountIdConversion; use subtensor_runtime_common::{NetUid, TaoBalance}; @@ -124,6 +125,12 @@ impl Pallet { Error::::NonAssociatedColdKey ); + // Ensure that hotkey is not a special account + ensure!( + Self::is_subnet_account_id(hotkey).is_none(), + Error::::NonAssociatedColdKey + ); + // --- 3. Ensure the mechanism is Dynamic. ensure!(mechid == 1, Error::::MechanismDoesNotExist); @@ -190,7 +197,7 @@ impl Pallet { log::debug!("init_new_network: {netuid_to_register:?}"); // --- 12. Add the caller to the neuron set. - Self::create_account_if_non_existent(&coldkey, hotkey); + Self::create_account_if_non_existent(&coldkey, hotkey)?; Self::append_neuron(netuid_to_register, hotkey, current_block); log::debug!("Appended neuron for netuid {netuid_to_register:?}, hotkey: {hotkey:?}"); @@ -216,7 +223,7 @@ impl Pallet { SubnetTAO::::insert(netuid_to_register, pool_initial_tao); SubnetAlphaIn::::insert(netuid_to_register, pool_initial_alpha); SubnetOwner::::insert(netuid_to_register, coldkey.clone()); - SubnetOwnerHotkey::::insert(netuid_to_register, hotkey.clone()); + Self::set_subnet_owner_hotkey(netuid_to_register, hotkey)?; SubnetLocked::::insert(netuid_to_register, actual_tao_lock_amount); SubnetTaoProvided::::insert(netuid_to_register, TaoBalance::ZERO); SubnetAlphaInProvided::::insert(netuid_to_register, AlphaBalance::ZERO); @@ -430,7 +437,7 @@ impl Pallet { ); // Insert/update the hotkey - SubnetOwnerHotkey::::insert(netuid, hotkey); + Self::set_subnet_owner_hotkey(netuid, hotkey)?; // Return success. Ok(()) @@ -447,4 +454,13 @@ impl Pallet { None } } + + pub fn is_subnet_account_id(account: &T::AccountId) -> Option { + let pallet_id = T::SubtensorPalletId::get(); + + match PalletId::try_from_sub_account::(account) { + Some((decoded_pallet_id, netuid)) if decoded_pallet_id == pallet_id => Some(netuid), + _ => None, + } + } } diff --git a/pallets/subtensor/src/swap/swap_coldkey.rs b/pallets/subtensor/src/swap/swap_coldkey.rs index 27fef995b2..d48313f8a9 100644 --- a/pallets/subtensor/src/swap/swap_coldkey.rs +++ b/pallets/subtensor/src/swap/swap_coldkey.rs @@ -30,7 +30,7 @@ impl Pallet { Self::transfer_coldkey_stake(netuid, old_coldkey, new_coldkey); } Self::transfer_staking_hotkeys(old_coldkey, new_coldkey); - Self::transfer_hotkeys_ownership(old_coldkey, new_coldkey); + Self::transfer_hotkeys_ownership(old_coldkey, new_coldkey)?; // Transfer any remaining balance from old_coldkey to new_coldkey let remaining_balance = Self::get_coldkey_balance(old_coldkey); @@ -164,14 +164,17 @@ impl Pallet { } /// Transfer the ownership of the hotkeys owned by the old coldkey to the new coldkey. - fn transfer_hotkeys_ownership(old_coldkey: &T::AccountId, new_coldkey: &T::AccountId) { + fn transfer_hotkeys_ownership( + old_coldkey: &T::AccountId, + new_coldkey: &T::AccountId, + ) -> DispatchResult { let old_owned_hotkeys: Vec = OwnedHotkeys::::get(old_coldkey); let mut new_owned_hotkeys: Vec = OwnedHotkeys::::get(new_coldkey); for owned_hotkey in old_owned_hotkeys.iter() { // Remove the hotkey from the old coldkey. Owner::::remove(owned_hotkey); // Add the hotkey to the new coldkey. - Owner::::insert(owned_hotkey, new_coldkey.clone()); + Self::set_hotkey_owner(new_coldkey, owned_hotkey)?; // Addd the owned hotkey to the new set of owned hotkeys. if !new_owned_hotkeys.contains(owned_hotkey) { new_owned_hotkeys.push(owned_hotkey.clone()); @@ -179,5 +182,6 @@ impl Pallet { } OwnedHotkeys::::remove(old_coldkey); OwnedHotkeys::::insert(new_coldkey, new_owned_hotkeys); + Ok(()) } } diff --git a/pallets/subtensor/src/swap/swap_hotkey.rs b/pallets/subtensor/src/swap/swap_hotkey.rs index 8cb12a974a..30b9947ca4 100644 --- a/pallets/subtensor/src/swap/swap_hotkey.rs +++ b/pallets/subtensor/src/swap/swap_hotkey.rs @@ -192,7 +192,7 @@ impl Pallet { // 2. Swap owner. // Owner( hotkey ) -> coldkey -- the coldkey that owns the hotkey. Owner::::remove(old_hotkey); - Owner::::insert(new_hotkey, coldkey.clone()); + Self::set_hotkey_owner(coldkey, new_hotkey)?; weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1)); // 3. Swap OwnedHotkeys. @@ -314,7 +314,7 @@ impl Pallet { // 7. Swap owner. // Owner( hotkey ) -> coldkey -- the coldkey that owns the hotkey. // Owner::::remove(old_hotkey); - Owner::::insert(new_hotkey, coldkey.clone()); + Self::create_account_if_non_existent(coldkey, new_hotkey)?; weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1)); // 8. Swap OwnedHotkeys. @@ -513,7 +513,7 @@ impl Pallet { if let Ok(old_subnet_owner_hotkey) = SubnetOwnerHotkey::::try_get(netuid) { weight.saturating_accrue(T::DbWeight::get().reads(1)); if old_subnet_owner_hotkey == *old_hotkey { - SubnetOwnerHotkey::::insert(netuid, new_hotkey); + Self::set_subnet_owner_hotkey(netuid, new_hotkey)?; weight.saturating_accrue(T::DbWeight::get().writes(1)); } } diff --git a/pallets/subtensor/src/tests/children.rs b/pallets/subtensor/src/tests/children.rs index 1558c4cb6c..18bf23cb1f 100644 --- a/pallets/subtensor/src/tests/children.rs +++ b/pallets/subtensor/src/tests/children.rs @@ -341,7 +341,7 @@ fn test_add_singular_child() { ), Err(Error::::NonAssociatedColdKey.into()) ); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); step_rate_limit(&TransactionType::SetChildren, netuid); assert_eq!( SubtensorModule::do_schedule_children( diff --git a/pallets/subtensor/src/tests/evm.rs b/pallets/subtensor/src/tests/evm.rs index ae0acde27a..d692a72f72 100644 --- a/pallets/subtensor/src/tests/evm.rs +++ b/pallets/subtensor/src/tests/evm.rs @@ -44,7 +44,7 @@ fn test_associate_evm_key_success() { let coldkey = U256::from(1); let hotkey = U256::from(2); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); @@ -93,7 +93,7 @@ fn test_associate_evm_key_different_block_number_success() { let coldkey = U256::from(1); let hotkey = U256::from(2); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); @@ -182,7 +182,7 @@ fn test_associate_evm_key_hotkey_not_registered_in_subnet() { let coldkey = U256::from(1); let hotkey = U256::from(2); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); let pair = ecdsa::Pair::generate().0; let public = pair.public(); @@ -224,7 +224,7 @@ fn test_associate_evm_key_using_wrong_hash_function() { let coldkey = U256::from(1); let hotkey = U256::from(2); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); @@ -268,7 +268,7 @@ fn test_associate_evm_key_rate_limit_exceeded() { let coldkey = U256::from(1); let hotkey = U256::from(2); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); diff --git a/pallets/subtensor/src/tests/leasing.rs b/pallets/subtensor/src/tests/leasing.rs index 0c6ae629c3..c479bc4db5 100644 --- a/pallets/subtensor/src/tests/leasing.rs +++ b/pallets/subtensor/src/tests/leasing.rs @@ -264,7 +264,7 @@ fn test_terminate_lease_works() { // Create a hotkey for the beneficiary let hotkey = U256::from(3); - SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); // Terminate the lease assert_ok!(SubtensorModule::terminate_lease( @@ -356,7 +356,7 @@ fn test_terminate_lease_fails_if_origin_is_not_beneficiary() { // Create a hotkey for the beneficiary let hotkey = U256::from(3); - SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); // Terminate the lease assert_err!( @@ -389,7 +389,7 @@ fn test_terminate_lease_fails_if_lease_has_no_end_block() { // Create a hotkey for the beneficiary let hotkey = U256::from(3); - SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); // Terminate the lease assert_err!( @@ -427,7 +427,7 @@ fn test_terminate_lease_fails_if_lease_has_not_ended() { // Create a hotkey for the beneficiary let hotkey = U256::from(3); - SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&beneficiary, &hotkey); // Terminate the lease assert_err!( diff --git a/pallets/subtensor/src/tests/move_stake.rs b/pallets/subtensor/src/tests/move_stake.rs index 294dc79661..ee52dca246 100644 --- a/pallets/subtensor/src/tests/move_stake.rs +++ b/pallets/subtensor/src/tests/move_stake.rs @@ -26,8 +26,8 @@ fn test_do_move_success() { let stake_amount = DefaultMinStake::::get() * 10.into(); // Set up initial stake - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); SubtensorModule::stake_into_subnet( &origin_hotkey, &coldkey, @@ -103,8 +103,8 @@ fn test_do_move_different_subnets() { ); // Set up initial stake and subnets - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); SubtensorModule::stake_into_subnet( &origin_hotkey, &coldkey, @@ -276,7 +276,7 @@ fn test_do_move_nonexistent_destination_hotkey() { mock::setup_reserves(netuid, reserve.into(), reserve.into()); // Set up initial stake - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); let alpha = SubtensorModule::stake_into_subnet( &origin_hotkey, &coldkey, @@ -360,8 +360,9 @@ fn test_do_move_partial_stake() { // Move partial stake let alpha_moved = AlphaBalance::from(alpha.to_u64() * portion_moved / 10); - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = + SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); assert_ok!(SubtensorModule::do_move_stake( RuntimeOrigin::signed(coldkey), origin_hotkey, @@ -409,8 +410,8 @@ fn test_do_move_multiple_times() { let initial_stake = DefaultMinStake::::get().to_u64() * 10; // Set up initial stake - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey1); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey2); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey1); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey2); SubtensorModule::stake_into_subnet( &hotkey1, &coldkey, @@ -501,8 +502,8 @@ fn test_do_move_wrong_origin() { // Attempt to move stake with wrong origin add_network(netuid, 1, 0); - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); assert_err!( SubtensorModule::do_move_stake( RuntimeOrigin::signed(wrong_coldkey), @@ -549,7 +550,7 @@ fn test_do_move_same_hotkey_fails() { let stake_amount = DefaultMinStake::::get().to_u64() * 10; // Set up initial stake - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -599,8 +600,8 @@ fn test_do_move_event_emission() { let stake_amount = DefaultMinStake::::get().to_u64() * 10; // Set up initial stake - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); SubtensorModule::stake_into_subnet( &origin_hotkey, &coldkey, @@ -674,8 +675,8 @@ fn test_do_move_storage_updates() { .unwrap(); // Move stake - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); let alpha = SubtensorModule::get_stake_for_hotkey_and_coldkey_on_subnet( &origin_hotkey, &coldkey, @@ -725,8 +726,8 @@ fn test_move_full_amount_same_netuid() { let origin_hotkey = U256::from(2); let destination_hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); // Set up initial stake SubtensorModule::stake_into_subnet( @@ -790,8 +791,8 @@ fn test_do_move_max_values() { let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); // Set up initial stake with maximum value - SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &origin_hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &destination_hotkey); // Add lots of liquidity to bypass low liquidity check let reserve = u64::MAX / 1000; @@ -901,8 +902,8 @@ fn test_do_transfer_success() { let stake_amount = DefaultMinStake::::get().to_u64() * 10; // 3. Set up initial stake: (origin_coldkey, hotkey) on netuid. - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); - SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, @@ -1011,7 +1012,7 @@ fn test_do_transfer_insufficient_stake() { let hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, @@ -1051,7 +1052,7 @@ fn test_do_transfer_wrong_origin() { let stake_amount = DefaultMinStake::::get().to_u64() * 10; let fee: u64 = 0; // FIXME: DefaultStakingFee is deprecated - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); SubtensorModule::add_balance_to_coldkey_account( &origin_coldkey, (stake_amount + fee).into(), @@ -1093,7 +1094,7 @@ fn test_do_transfer_minimum_stake_check() { let hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get(); - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, @@ -1135,8 +1136,8 @@ fn test_do_transfer_different_subnets() { let stake_amount = DefaultMinStake::::get().to_u64() * 10; // 3. Create accounts if needed. - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); - SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); // 4. Deposit free balance so transaction fees do not reduce staked funds. SubtensorModule::add_balance_to_coldkey_account(&origin_coldkey, 1_000_000_000.into()); @@ -1207,7 +1208,7 @@ fn test_do_swap_success() { let hotkey = U256::from(2); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1262,7 +1263,7 @@ fn test_do_swap_nonexistent_subnet() { let nonexistent_netuid2 = NetUid::from(9999); let stake_amount = 1_000_000; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); assert_noop!( SubtensorModule::do_swap_stake( @@ -1315,7 +1316,7 @@ fn test_do_swap_insufficient_stake() { let stake_amount = DefaultMinStake::::get().to_u64() * 5; let attempted_swap = stake_amount * 2; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1350,7 +1351,7 @@ fn test_do_swap_wrong_origin() { let hotkey = U256::from(3); let stake_amount = 100_000; - SubtensorModule::create_account_if_non_existent(&real_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&real_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &real_coldkey, @@ -1388,7 +1389,7 @@ fn test_do_swap_minimum_stake_check() { let total_stake = DefaultMinStake::::get(); let swap_amount = 1; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1424,7 +1425,7 @@ fn test_do_swap_same_subnet() { let hotkey = U256::from(2); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1469,7 +1470,7 @@ fn test_do_swap_partial_stake() { let hotkey = U256::from(2); let total_stake_tao = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1521,7 +1522,7 @@ fn test_do_swap_storage_updates() { let hotkey = U256::from(2); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1581,7 +1582,7 @@ fn test_do_swap_multiple_times() { let hotkey = U256::from(2); let initial_stake = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1652,7 +1653,7 @@ fn test_do_swap_allows_non_owned_hotkey() { let foreign_coldkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&foreign_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&foreign_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &coldkey, @@ -1799,8 +1800,8 @@ fn test_transfer_stake_rate_limited() { let hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); - SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, @@ -1844,8 +1845,8 @@ fn test_transfer_stake_doesnt_limit_destination_coldkey() { let hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); - SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&destination_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, @@ -1891,7 +1892,7 @@ fn test_swap_stake_limits_destination_netuid() { let hotkey = U256::from(3); let stake_amount = DefaultMinStake::::get().to_u64() * 10; - SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&origin_coldkey, &hotkey); SubtensorModule::stake_into_subnet( &hotkey, &origin_coldkey, diff --git a/pallets/subtensor/src/tests/recycle_alpha.rs b/pallets/subtensor/src/tests/recycle_alpha.rs index 404967dc74..94b5e98279 100644 --- a/pallets/subtensor/src/tests/recycle_alpha.rs +++ b/pallets/subtensor/src/tests/recycle_alpha.rs @@ -23,7 +23,7 @@ fn test_recycle_success() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -79,7 +79,7 @@ fn test_recycle_two_stakers() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -149,7 +149,7 @@ fn test_recycle_staker_is_nominator() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -222,7 +222,7 @@ fn test_burn_success() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -278,7 +278,7 @@ fn test_burn_staker_is_nominator() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -348,7 +348,7 @@ fn test_burn_two_stakers() { Balances::make_free_balance_be(&coldkey, initial_balance.into()); // associate coldkey and hotkey - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); assert!(SubtensorModule::if_subnet_exist(netuid)); @@ -419,7 +419,7 @@ fn test_recycle_errors() { let initial_balance = 1_000_000_000; Balances::make_free_balance_be(&coldkey, initial_balance.into()); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); let stake_amount = 200_000; @@ -491,7 +491,7 @@ fn test_burn_errors() { let initial_balance = 1_000_000_000; Balances::make_free_balance_be(&coldkey, initial_balance.into()); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); register_ok_neuron(netuid, hotkey, coldkey, 0); let stake_amount = 200_000; diff --git a/pallets/subtensor/src/tests/staking.rs b/pallets/subtensor/src/tests/staking.rs index 28d5353ce7..80574ec786 100644 --- a/pallets/subtensor/src/tests/staking.rs +++ b/pallets/subtensor/src/tests/staking.rs @@ -792,7 +792,7 @@ fn test_add_stake_insufficient_liquidity() { let amount_staked = DefaultMinStake::::get().to_u64() * 10; let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::add_balance_to_coldkey_account(&coldkey, amount_staked.into()); // Set the liquidity at lowest possible value so that all staking requests fail @@ -823,7 +823,7 @@ fn test_add_stake_insufficient_liquidity_one_side_ok() { let amount_staked = DefaultMinStake::::get().to_u64() * 10; let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::add_balance_to_coldkey_account(&coldkey, amount_staked.into()); // Set the liquidity at lowest possible value so that all staking requests fail @@ -852,7 +852,7 @@ fn test_add_stake_insufficient_liquidity_one_side_fail() { let amount_staked = DefaultMinStake::::get().to_u64() * 10; let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::add_balance_to_coldkey_account(&coldkey, amount_staked.into()); // Set the liquidity at lowest possible value so that all staking requests fail @@ -883,7 +883,7 @@ fn test_remove_stake_insufficient_liquidity() { let amount_staked = DefaultMinStake::::get().to_u64() * 10; let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); SubtensorModule::add_balance_to_coldkey_account(&coldkey, amount_staked.into()); // Simulate stake for hotkey @@ -2572,7 +2572,7 @@ fn test_add_stake_fee_goes_to_subnet_tao() { let tao_to_stake = DefaultMinStake::::get() * 10.into(); let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); let subnet_tao_before = SubnetTAO::::get(netuid); // Add stake @@ -2618,7 +2618,7 @@ fn test_remove_stake_fee_goes_to_subnet_tao() { let tao_to_stake = DefaultMinStake::::get() * 10.into(); let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); let subnet_tao_before = SubnetTAO::::get(netuid); // Add stake @@ -2671,7 +2671,7 @@ fn test_remove_stake_fee_realistic_values() { let alpha_divs = AlphaBalance::from(2_816_190); let netuid = add_dynamic_network(&subnet_owner_hotkey, &subnet_owner_coldkey); - SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); + let _ = SubtensorModule::create_account_if_non_existent(&coldkey, &hotkey); // Mock a realistic scenario: // Subnet 1 has 3896 TAO and 128_011 Alpha in reserves, which diff --git a/pallets/subtensor/src/tests/subnet.rs b/pallets/subtensor/src/tests/subnet.rs index 098955ae46..8567d02407 100644 --- a/pallets/subtensor/src/tests/subnet.rs +++ b/pallets/subtensor/src/tests/subnet.rs @@ -5,6 +5,7 @@ use crate::*; use frame_support::{assert_err, assert_noop, assert_ok}; use frame_system::Config; use sp_core::U256; +use std::collections::BTreeSet; use subtensor_runtime_common::{AlphaBalance, TaoBalance}; use super::mock; @@ -893,8 +894,6 @@ fn test_update_symbol_fails_if_symbol_already_in_use() { #[test] fn test_get_subnet_account_id_exists_and_is_distinct_for_257_consecutive_subnets() { new_test_ext(1).execute_with(|| { - use std::collections::BTreeSet; - let mut account_ids = BTreeSet::new(); for raw_netuid in 0u16..=256u16 { @@ -912,3 +911,41 @@ fn test_get_subnet_account_id_exists_and_is_distinct_for_257_consecutive_subnets assert_eq!(account_ids.len(), 257); }); } + +// cargo test --package pallet-subtensor --lib -- tests::subnet::test_is_subnet_account_id --exact --nocapture +#[test] +fn test_is_subnet_account_id() { + new_test_ext(1).execute_with(|| { + for raw_netuid in 0u16..=2048u16 { + let netuid = NetUid::from(raw_netuid); + add_network(netuid, 10, 0); + + let account_id = SubtensorModule::get_subnet_account_id(netuid).unwrap(); + let roudtrip_netuid = SubtensorModule::is_subnet_account_id(&account_id); + assert_eq!(netuid, roudtrip_netuid.unwrap()); + } + + // Not a subnet account + let not_subnet_account_id = U256::from(1); + assert!(SubtensorModule::is_subnet_account_id(¬_subnet_account_id).is_none()); + }); +} + +// cargo test --package pallet-subtensor --lib -- tests::subnet::test_cannot_register_system_hotkey --exact --nocapture +#[test] +fn test_cannot_register_system_hotkey() { + new_test_ext(1).execute_with(|| { + for raw_netuid in 0u16..=2048u16 { + let coldkey = U256::from(1); + let netuid = NetUid::from(raw_netuid); + add_network(netuid, 10, 0); + + let account_id = SubtensorModule::get_subnet_account_id(netuid).unwrap(); + assert_err!( + SubtensorModule::create_account_if_non_existent(&coldkey, &account_id), + Error::::NonAssociatedColdKey + ); + assert!(!SubtensorModule::coldkey_owns_hotkey(&coldkey, &account_id),); + } + }); +} diff --git a/pallets/subtensor/src/utils/misc.rs b/pallets/subtensor/src/utils/misc.rs index 75f2d6695d..5e53b56a34 100644 --- a/pallets/subtensor/src/utils/misc.rs +++ b/pallets/subtensor/src/utils/misc.rs @@ -844,9 +844,16 @@ impl Pallet { /// /// * Update the SubnetOwnerHotkey storage. /// * Emits a SubnetOwnerHotkeySet event. - pub fn set_subnet_owner_hotkey(netuid: NetUid, hotkey: &T::AccountId) { + pub fn set_subnet_owner_hotkey(netuid: NetUid, hotkey: &T::AccountId) -> DispatchResult { + // Ensure that hotkey is not a special account + ensure!( + Self::is_subnet_account_id(hotkey).is_none(), + Error::::NonAssociatedColdKey + ); + SubnetOwnerHotkey::::insert(netuid, hotkey.clone()); Self::deposit_event(Event::SubnetOwnerHotkeySet(netuid, hotkey.clone())); + Ok(()) } // Get the uid of the Owner Hotkey for a subnet. diff --git a/runtime/tests/account_conversion.rs b/runtime/tests/account_conversion.rs new file mode 100644 index 0000000000..11aad3da85 --- /dev/null +++ b/runtime/tests/account_conversion.rs @@ -0,0 +1,53 @@ +#![allow(clippy::unwrap_used)] + +use node_subtensor_runtime::{BuildStorage, RuntimeGenesisConfig, SubtensorModule, System}; +use subtensor_runtime_common::NetUid; + +fn new_test_ext() -> sp_io::TestExternalities { + sp_tracing::try_init_simple(); + let mut ext: sp_io::TestExternalities = RuntimeGenesisConfig { + balances: pallet_balances::GenesisConfig { + balances: vec![], + dev_accounts: None, + }, + ..Default::default() + } + .build_storage() + .unwrap() + .into(); + ext.execute_with(|| System::set_block_number(1)); + ext +} + +/// Test full-range netuids on real ss58 accounts to ensure no panics +/// cargo test --package node-subtensor-runtime --test account_conversion -- test_subnet_account_id_no_panics --exact --nocapture +#[test] +#[ignore] +fn test_subnet_account_id_no_panics() { + new_test_ext().execute_with(|| { + for raw_netuid in 0u16..=u16::MAX { + let netuid = NetUid::from(raw_netuid); + SubtensorModule::init_new_network(netuid, 10); + + let account_id = SubtensorModule::get_subnet_account_id(netuid).unwrap(); + let roudtrip_netuid = SubtensorModule::is_subnet_account_id(&account_id); + assert_eq!(netuid, roudtrip_netuid.unwrap()); + } + }); +} + +/// Quick sanity test +/// cargo test --package node-subtensor-runtime --test account_conversion -- test_subnet_account_id_no_panics_quick --exact --nocapture +#[test] +fn test_subnet_account_id_no_panics_quick() { + new_test_ext().execute_with(|| { + for raw_netuid in 0u16..=1024u16 { + let netuid = NetUid::from(raw_netuid); + SubtensorModule::init_new_network(netuid, 10); + + let account_id = SubtensorModule::get_subnet_account_id(netuid).unwrap(); + let roudtrip_netuid = SubtensorModule::is_subnet_account_id(&account_id); + assert_eq!(netuid, roudtrip_netuid.unwrap()); + } + }); +} From b51fe9c4ac44bf5cb1fb02442cfde597aa953e94 Mon Sep 17 00:00:00 2001 From: Greg Zaitsev Date: Mon, 23 Mar 2026 11:24:51 -0400 Subject: [PATCH 3/4] Document subnet account IDs for subnets 0-1024 --- docs/subnet_account_ids.md | 1048 +++++++++++++++++++++++++++++++++++- 1 file changed, 1026 insertions(+), 22 deletions(-) diff --git a/docs/subnet_account_ids.md b/docs/subnet_account_ids.md index a0e2984081..a3fc180548 100644 --- a/docs/subnet_account_ids.md +++ b/docs/subnet_account_ids.md @@ -1,25 +1,1029 @@ # The subnet account IDs - 0: 5EYCAe5jLQhn6ofDSvqF6iY53erXNkwhyE1aCEgvi1NNs91F - 1: 5EYCAe5jLQhn6ofDSvqWqk5fA9XiqK3ahtx5kBNmAqF78mqL - 2: 5EYCAe5jLQhn6ofDSvqnamdFGeCvHs9TSZtbJ84bdf7qQRc6 - 3: 5EYCAe5jLQhn6ofDSvr4KoAqP8t7kRFLBEq6r4kS6UzZgCb5 - 4: 5EYCAe5jLQhn6ofDSvrL4piRVdZKCyMCuumcQ1SGZJsHwmeE - 5: 5EYCAe5jLQhn6ofDSvrborG1c8EWfXT5eai7wx8728k2DHK7 - 6: 5EYCAe5jLQhn6ofDSvrsYsobicui85YxPFedVtowUxckUuF8 - 7: 5EYCAe5jLQhn6ofDSvs9HuMBq7auadeq7vb93qVmwnVUkg5A - 8: 5EYCAe5jLQhn6ofDSvsR2vtmwcG73BkhrbXebnBcQcND2Bdh - 9: 5EYCAe5jLQhn6ofDSvsgmxSN46wJVjrabGUA9isSsSEwHnFy - 10: 5EYCAe5jLQhn6ofDSvsxWyyxAbcVxHxTKwQfhfZHLG7fZUJG - 11: 5EYCAe5jLQhn6ofDSvtEG1XYH6HhQr4L4cMBFcF7o5zPpyA3 - 12: 5EYCAe5jLQhn6ofDSvtW1358PaxtsQACoHHgoYvxFus86kJK - 13: 5EYCAe5jLQhn6ofDSvtmk4ciW5e6KxG5XxECMVcnijjrN8rz - 14: 5EYCAe5jLQhn6ofDSvu3V6AJcaKHnWMxGdAhuSJdBZcadwDn - 15: 5EYCAe5jLQhn6ofDSvuKE7htj4zVF4Tq1J7DTNzTePVJucfX - 16: 5EYCAe5jLQhn6ofDSvuay9FUqZfghcZhjy3j1KgJ7DN3BDc2 - 17: 5EYCAe5jLQhn6ofDSvuriAo4x4LtAAfaUdzEZGN8a3EmSncG - 18: 5EYCAe5jLQhn6ofDSvv8TCLf4Z25cimTDJvk7D3y2s7ViZEm - 19: 5EYCAe5jLQhn6ofDSvvQCDtFB3hH5GsKwysFf9joVgzDytnb - 20: 5EYCAe5jLQhn6ofDSvvfwFRqHYNUXpyCgeomD6RdxWrxFpQR - +netuid: ss58 account ID +0: 5EYCAe5jLQhn6ofDSvqF6iY53erXNkwhyE1aCEgvi1NNs91F +1: 5EYCAe5jLQhn6ofDSvqWqk5fA9XiqK3ahtx5kBNmAqF78mqL +2: 5EYCAe5jLQhn6ofDSvqnamdFGeCvHs9TSZtbJ84bdf7qQRc6 +3: 5EYCAe5jLQhn6ofDSvr4KoAqP8t7kRFLBEq6r4kS6UzZgCb5 +4: 5EYCAe5jLQhn6ofDSvrL4piRVdZKCyMCuumcQ1SGZJsHwmeE +5: 5EYCAe5jLQhn6ofDSvrborG1c8EWfXT5eai7wx8728k2DHK7 +6: 5EYCAe5jLQhn6ofDSvrsYsobicui85YxPFedVtowUxckUuF8 +7: 5EYCAe5jLQhn6ofDSvs9HuMBq7auadeq7vb93qVmwnVUkg5A +8: 5EYCAe5jLQhn6ofDSvsR2vtmwcG73BkhrbXebnBcQcND2Bdh +9: 5EYCAe5jLQhn6ofDSvsgmxSN46wJVjrabGUA9isSsSEwHnFy +10: 5EYCAe5jLQhn6ofDSvsxWyyxAbcVxHxTKwQfhfZHLG7fZUJG +11: 5EYCAe5jLQhn6ofDSvtEG1XYH6HhQr4L4cMBFcF7o5zPpyA3 +12: 5EYCAe5jLQhn6ofDSvtW1358PaxtsQACoHHgoYvxFus86kJK +13: 5EYCAe5jLQhn6ofDSvtmk4ciW5e6KxG5XxECMVcnijjrN8rz +14: 5EYCAe5jLQhn6ofDSvu3V6AJcaKHnWMxGdAhuSJdBZcadwDn +15: 5EYCAe5jLQhn6ofDSvuKE7htj4zVF4Tq1J7DTNzTePVJucfX +16: 5EYCAe5jLQhn6ofDSvuay9FUqZfghcZhjy3j1KgJ7DN3BDc2 +17: 5EYCAe5jLQhn6ofDSvuriAo4x4LtAAfaUdzEZGN8a3EmSncG +18: 5EYCAe5jLQhn6ofDSvv8TCLf4Z25cimTDJvk7D3y2s7ViZEm +19: 5EYCAe5jLQhn6ofDSvvQCDtFB3hH5GsKwysFf9joVgzDytnb +20: 5EYCAe5jLQhn6ofDSvvfwFRqHYNUXpyCgeomD6RdxWrxFpQR +21: 5EYCAe5jLQhn6ofDSvvwgGyRQ33fzP55RKkGm37URLjgXG7M +22: 5EYCAe5jLQhn6ofDSvwDRJX1WXisSwAx9zgnJyoJtAcQo59Y +23: 5EYCAe5jLQhn6ofDSvwVAL4bd2Q4uVGptfdHrvV9LzV94VBb +24: 5EYCAe5jLQhn6ofDSvwkuMcBjX5GN3NhdLZoQsAyopMsL7A7 +25: 5EYCAe5jLQhn6ofDSvx2eP9mr1kTpbUaN1WJxorpGeEbbfgG +26: 5EYCAe5jLQhn6ofDSvxJPQhMxWRfH9aT6gSpWkYejU7KsbGp +27: 5EYCAe5jLQhn6ofDSvxa8SEx516rjhgKqMPL4hEVCHz49DPw +28: 5EYCAe5jLQhn6ofDSvxqsTnYBVn4CFnCa2KqcdvKf7rnQo7f +29: 5EYCAe5jLQhn6ofDSvy7cVL8HzTFeot5JhGMAacA7wjWgPix +30: 5EYCAe5jLQhn6ofDSvyPMWsiQV8T7Myx3NCriXHzamcEwyqa +31: 5EYCAe5jLQhn6ofDSvyf6YRJWyoeZv5pn39NGTyq3bUyDc8k +32: 5EYCAe5jLQhn6ofDSvyvqZxtdUUr2UBhWi5spQffWRMhV5hU +33: 5EYCAe5jLQhn6ofDSvzCabWUjyA3V2HaFP2PNMMVyFERkxPm +34: 5EYCAe5jLQhn6ofDSvzUKd44rTqEwaPSz3xtvJ3LS57A2Td3 +35: 5EYCAe5jLQhn6ofDSvzk4ebexxWSQ8VKiiuQUEjAttytJ8Nx +36: 5EYCAe5jLQhn6ofDSw11og9F5TBdrgbCTPqv2BR1MircZp68 +37: 5EYCAe5jLQhn6ofDSw1HYhgqBwrqKEh5C4nRa86qpYjLqCQd +38: 5EYCAe5jLQhn6ofDSw1ZHjERJSY2mnnwvjiw84ngHNc56t9n +39: 5EYCAe5jLQhn6ofDSw1q2kn1QwDEELtpfQfSg1UWkCUoNVFh +40: 5EYCAe5jLQhn6ofDSw26mnKbXRtRgtzhQ5bxDxAMD2MXeL6A +41: 5EYCAe5jLQhn6ofDSw2NWosBdvZd9T6a8kYTmtrBfrEFusmX +42: 5EYCAe5jLQhn6ofDSw2eFqQmkREpc1CSsRUyKqY28g6zBbxD +43: 5EYCAe5jLQhn6ofDSw2uzrxMruv24ZJKc6RUsnDrbVyiT4uZ +44: 5EYCAe5jLQhn6ofDSw3BjtVwyQbDX7QCLmMzRiuh4KrSienC +45: 5EYCAe5jLQhn6ofDSw3TUv3Y5uGQyfW55SJVyfbXX9jAzHBc +46: 5EYCAe5jLQhn6ofDSw3jDwb8CPwcSDbwp7F1XcHMyybuFsV6 +47: 5EYCAe5jLQhn6ofDSw3zxy8iJtcotmhpYnBX5YyCSoUdXS9C +48: 5EYCAe5jLQhn6ofDSw4GhzgJRPJ1MKohHT82dVf2udMMo854 +49: 5EYCAe5jLQhn6ofDSw4YT2DtXsyCosua284YBSLsNTE64sjn +50: 5EYCAe5jLQhn6ofDSw4pC3mUeNeQGS1Sko13jP2hqH6pLJc1 +51: 5EYCAe5jLQhn6ofDSw55w5K4ksKbiz7KVTwZHKiYJ6yYc62p +52: 5EYCAe5jLQhn6ofDSw5Mg6resMzoBYDCE8t4qGQNkvrGsYLi +53: 5EYCAe5jLQhn6ofDSw5dR8QEyrfze6K4xopaPD6DDkj19BcH +54: 5EYCAe5jLQhn6ofDSw5uA9wq6MMC6eQwhUm5w9n3gabjR243 +55: 5EYCAe5jLQhn6ofDSw6AuBVRCr2PZCWpS9hbV6Tt9QUTghER +56: 5EYCAe5jLQhn6ofDSw6SeD31KLhb1kchApe7339icEMBxKr7 +57: 5EYCAe5jLQhn6ofDSw6iPEabRqNnUJiZuVacayqZ54DvDhGB +58: 5EYCAe5jLQhn6ofDSw6z8G8BYL3yvrpSeAX88vXPXt6eVRoY +59: 5EYCAe5jLQhn6ofDSw7FsHfmepjBPQvKNqTdgsDDzhyNky3e +60: 5EYCAe5jLQhn6ofDSw7XcKDMmKQNqy2C7WQ9Eou4TXr72f1B +61: 5EYCAe5jLQhn6ofDSw7oMLkwsp5aJX84rBLenkatvMiqJC2W +62: 5EYCAe5jLQhn6ofDSw856NJXzJkmm5DwarHALhGjPBbZa5wW +63: 5EYCAe5jLQhn6ofDSw8LqPr86oRyDdKpKXDftdxZr1UHqWzq +64: 5EYCAe5jLQhn6ofDSw8caRPiDJ7AgBRh4CABSaeQJqM278gq +65: 5EYCAe5jLQhn6ofDSw8tKSwJKnnN8jXZns6gzXLEmfDkNtXu +66: 5EYCAe5jLQhn6ofDSw9A4UUtSHTZbHdSXY3CYU25EV6UeLH2 +67: 5EYCAe5jLQhn6ofDSw9RoW2UYn8m3qjKGCyi6QhuhJyCv9nu +68: 5EYCAe5jLQhn6ofDSw9hYXa4fGoxWPqBzsvDeMPkA8qwBecQ +69: 5EYCAe5jLQhn6ofDSw9yHZ7emmV9xww4jYrjCJ5acxifTH7b +70: 5EYCAe5jLQhn6ofDSwAF2afEtGAMRW2wUDoEkEmR5nbPiuFf +71: 5EYCAe5jLQhn6ofDSwAWmcCpzkqYt48pCtjkJBTFYcU7ziWG +72: 5EYCAe5jLQhn6ofDSwAnWdkR7FWkLcEgwZgFr8961SLrGPJp +73: 5EYCAe5jLQhn6ofDSwB4FfJ1DkBwoALZgEcmQ4pvUGDaXxGw +74: 5EYCAe5jLQhn6ofDSwBKzgqbLEs9FiSSQuZGx1Wkw66JoNQY +75: 5EYCAe5jLQhn6ofDSwBbjiPBSjYLiGYK9aVnVxCbPuy357eQ +76: 5EYCAe5jLQhn6ofDSwBsUjvmZEDYApeBtFSJ3ttRrjqmLmRP +77: 5EYCAe5jLQhn6ofDSwC9DmUMfitjdNk4cvNobqaGKZiVcSd4 +78: 5EYCAe5jLQhn6ofDSwCQxo1wnDZw5vqwMbKK9nG6nPbDsr3v +79: 5EYCAe5jLQhn6ofDSwCghpZXtiF8YUwp6GFphiwwFDTx9ZXw +80: 5EYCAe5jLQhn6ofDSwCxSr781CvL133gpwCLFfdmi3LgRGUs +81: 5EYCAe5jLQhn6ofDSwDEBsei7hbXTb9ZZc8qocKcAsDQgmDH +82: 5EYCAe5jLQhn6ofDSwDVvuCJECGiv9FSJH5MMZ1Sdh68xe6G +83: 5EYCAe5jLQhn6ofDSwDmfvjtLgwvNhMK2x1ruVhH6WxsE2Rh +84: 5EYCAe5jLQhn6ofDSwE3QxHUTBd7qFTBmcxNTSP7ZLqbVqHX +85: 5EYCAe5jLQhn6ofDSwEK9yq4ZgJKHoZ4WHtt1P4x2AiKmP2V +86: 5EYCAe5jLQhn6ofDSwEau1NegAyWkMewExqPZKknUzb42r36 +87: 5EYCAe5jLQhn6ofDSwEre2vEnfeiCukoydmu7GScwpTnJa5d +88: 5EYCAe5jLQhn6ofDSwF8P4TpuAKufTrgiJiQfD8TQeLWaGop +89: 5EYCAe5jLQhn6ofDSwFQ861R1f1781xZSyevD9pHsUDEqiBR +90: 5EYCAe5jLQhn6ofDSwFfs7Z189gJaa4SBebRm6W8LJ5y7dfH +91: 5EYCAe5jLQhn6ofDSwFwc96bEeMW38AJvKXwK3Bxo7xhP3yn +92: 5EYCAe5jLQhn6ofDSwGDMAeBM92hVgGBezUSrysoFwqReqrS +93: 5EYCAe5jLQhn6ofDSwGV6CBmTdhtxEN4PfQxQvZdimi9vW9r +94: 5EYCAe5jLQhn6ofDSwGkqDjMa8P6QnTw8LMTxsFUBbatC8C5 +95: 5EYCAe5jLQhn6ofDSwH2aFGwgd4HsLZos1HyWowJeRTcTVsg +96: 5EYCAe5jLQhn6ofDSwHJKGpXo7jVKtfgbgEV4kd97FLLjBeJ +97: 5EYCAe5jLQhn6ofDSwHa4JN7ucQgnSmZLMAzchJya5D4zq8v +98: 5EYCAe5jLQhn6ofDSwHqoKui275tEzsS527WAdzp2u5oGNSd +99: 5EYCAe5jLQhn6ofDSwJ7YMTJ8bm5hYyJoh41iageVixXYH59 +100: 5EYCAe5jLQhn6ofDSwJPHNztF6SHA75BYMzXGXNUxYqFoj9g +101: 5EYCAe5jLQhn6ofDSwJf2QYUMb7UcfB4H2w2pU4KRNhz5GP5 +102: 5EYCAe5jLQhn6ofDSwJvmS64U5ng5DGw1hsYNQk9tCaiLvoS +103: 5EYCAe5jLQhn6ofDSwKCWTdeaaTsXmNokNp3vMRzM2TScknA +104: 5EYCAe5jLQhn6ofDSwKUFVBEh594zKUgV3kZUJ7porLAtE76 +105: 5EYCAe5jLQhn6ofDSwKjzWipoZpGSsaZDih52EofGgCu9mbP +106: 5EYCAe5jLQhn6ofDSwL1jYGQv4VTuRgRxPdaaBVVjW5dRU9u +107: 5EYCAe5jLQhn6ofDSwLHUZp12ZAfMynJh4a688BLCKxMhEMq +108: 5EYCAe5jLQhn6ofDSwLZDbMb93qrpXtBRjWbg4sAf9q5xtB8 +109: 5EYCAe5jLQhn6ofDSwLpxcuBFYX4H5z4AQT7E1Z17yhpELLK +110: 5EYCAe5jLQhn6ofDSwM6heSmN3CFje5vu5PcmxEqaoaYW1KP +111: 5EYCAe5jLQhn6ofDSwMNSfzMUXsTCCBodkL8Ktvg3dTGmYbX +112: 5EYCAe5jLQhn6ofDSwMeBhXwb2YeekHgNRGdsqcWWTL13NLP +113: 5EYCAe5jLQhn6ofDSwMuvj5XhXDr7JPZ76D9RnJLyHCjK2Zy +114: 5EYCAe5jLQhn6ofDSwNBfkd7p1u3ZrVRqm9eyizBS75TaPgK +115: 5EYCAe5jLQhn6ofDSwNTQnAhvWaF2QbJaS6AXfg1tvxBrDUN +116: 5EYCAe5jLQhn6ofDSwNj9oiJ31FSUxhBK72g5cMrMkpv7iJx +117: 5EYCAe5jLQhn6ofDSwNztqFt9VvdwWo43myBdZ3gpahePQpf +118: 5EYCAe5jLQhn6ofDSwPGdroUFzbqQ4tvnSuhBVjXHQaNet2o +119: 5EYCAe5jLQhn6ofDSwPYNtM4NVH2rczoX7rCjSRMkET6vioH +120: 5EYCAe5jLQhn6ofDSwPp7uteUyxEKB6gFnniHP7CD4KqCQDN +121: 5EYCAe5jLQhn6ofDSwQ5rwSEbUdRmjCYzTjDqKo2ftCZTubr +122: 5EYCAe5jLQhn6ofDSwQMbxyphyJdEHJRj8fjPGUs8i5HjcA3 +123: 5EYCAe5jLQhn6ofDSwQdLzXQpTypgqQJTocEwDAhbXx21Awy +124: 5EYCAe5jLQhn6ofDSwQu624zvxf29PWBCUYkV9rY4MpkGu1f +125: 5EYCAe5jLQhn6ofDSwRAq3cb3TLDbwc3w9VG36YNXBhUYKDi +126: 5EYCAe5jLQhn6ofDSwRSa5AB9x1R4VhvfpRmb3ECz1aCp2ze +127: 5EYCAe5jLQhn6ofDSwRiK6hmGSgcX3ooQVNH8yv3SqSw5mpH +128: 5EYCAe5jLQhn6ofDSwRz48FMNwMoybug9AJngvbsufKfME2t +129: 5EYCAe5jLQhn6ofDSwSFo9nwVS31SA1YsqFJEsHiNVCPcuZ9 +130: 5EYCAe5jLQhn6ofDSwSXYBLXbviCti7RcWBonoyYqK57tgCT +131: 5EYCAe5jLQhn6ofDSwSoHCt7iRPQMGDJMB8KLkfPJ8wrAGyP +132: 5EYCAe5jLQhn6ofDSwT52ERhpv4bopKB5r4pthMDkxpaRs97 +133: 5EYCAe5jLQhn6ofDSwTLmFyHwQjoGNR3pX1LSe34DnhJhU9A +134: 5EYCAe5jLQhn6ofDSwTcWHWt3uQzivWvZBwqzaitgca2xvCA +135: 5EYCAe5jLQhn6ofDSwTtFK4UAQ6CBUcoHrtMYXQj9SSmEgDM +136: 5EYCAe5jLQhn6ofDSwU9zLc4GtmPe2ig2Xps6U6ZcGKVWLNs +137: 5EYCAe5jLQhn6ofDSwURjN9ePPSb6apYmCmNeQnQ56CDn1SN +138: 5EYCAe5jLQhn6ofDSwUhUPhEVt7nZ8vRVshtCMUEXv4x3U6G +139: 5EYCAe5jLQhn6ofDSwUyDREpcNnz1h2JEYePkJA4zjwgK8dv +140: 5EYCAe5jLQhn6ofDSwVExSnQisUBUF8AyDauJEquTZpQaoue +141: 5EYCAe5jLQhn6ofDSwVWhUKzqN9NvoE3htXQrBXjvPh8rQgX +142: 5EYCAe5jLQhn6ofDSwVnSVsawrpaPMKvSZTvQ8DaPDZs8C7o +143: 5EYCAe5jLQhn6ofDSwW4BXRB4MVmquRoBEQRx4uQr3SbPfUD +144: 5EYCAe5jLQhn6ofDSwWKvYxmArAyJTXfuuLwW1bFJsKKf8Ax +145: 5EYCAe5jLQhn6ofDSwWbfaWMHLrAm1dYeaHT3xH5mhC3vmAe +146: 5EYCAe5jLQhn6ofDSwWsQc3wPqXNDZjRPFDxbtxvEX4nCgWj +147: 5EYCAe5jLQhn6ofDSwX99dbXWLCZg7qJ7vAU9qekhLwWUAUr +148: 5EYCAe5jLQhn6ofDSwXQtf97cpsm8fwArb6yhnLbAApEjeXz +149: 5EYCAe5jLQhn6ofDSwXgdgghjKYxbE33bG3VFj2Rczgy1Vmr +150: 5EYCAe5jLQhn6ofDSwXxNiEHqpEA3n8vKvyzofiG5pZhGuX1 +151: 5EYCAe5jLQhn6ofDSwYE7jmsxJuMWLEo4bvWMcQ6YeSRYYra +152: 5EYCAe5jLQhn6ofDSwYVrmKU4oaYxtLfoGs1uZ5w1UK9pR2x +153: 5EYCAe5jLQhn6ofDSwYmbns4BJFkRSSYXwoXTVmmUJBt5u26 +154: 5EYCAe5jLQhn6ofDSwZ3LpQeHnvwszYRGck31STbw84cMhaR +155: 5EYCAe5jLQhn6ofDSwZK5qxEQHc9LYeJ1HgYZP9SPwwLdHWw +156: 5EYCAe5jLQhn6ofDSwZapsVpWnHLo6kAjxd47KqGrmp4tuDS +157: 5EYCAe5jLQhn6ofDSwZrZu3QdGxYFer3UdZZfGX7KbgoAWJ3 +158: 5EYCAe5jLQhn6ofDSwa8JvazjmdjiCwvDJW5DDCwnRZXRvL1 +159: 5EYCAe5jLQhn6ofDSwaQ3x8arGJwAm3nwySam9tnFFSFhYDW +160: 5EYCAe5jLQhn6ofDSwafnygAxkz8dK9fgeP6K6aci5JyyTiu +161: 5EYCAe5jLQhn6ofDSwawY1Dm5FfL5sFYRKKbs3GTAuBiEuDo +162: 5EYCAe5jLQhn6ofDSwbDH2mMBkLXYRMR9zG7QyxHdj4SWa3R +163: 5EYCAe5jLQhn6ofDSwbV24JwJF1izyTHtfCcxve86YwAnHW1 +164: 5EYCAe5jLQhn6ofDSwbkm5rXQjgvTXZAdL98WsKxZNou3ejC +165: 5EYCAe5jLQhn6ofDSwc2W7Q7XEN7v5f3N15e4p1o2CgdKWRW +166: 5EYCAe5jLQhn6ofDSwcJF8whdj3KNdkv6g29ckhdV2ZMbAew +167: 5EYCAe5jLQhn6ofDSwcZzAVHkDiWqBrnqLxfAhPTwrS5rhzb +168: 5EYCAe5jLQhn6ofDSwcqjC2sriPiHjxfa1uAie5JQgJp8CWe +169: 5EYCAe5jLQhn6ofDSwd7UDaTyD4ukJ4YJgqgGam8sWBYQ2H8 +170: 5EYCAe5jLQhn6ofDSwdPDF845hk7CrAR3MnBpXSyLL4GfbVT +171: 5EYCAe5jLQhn6ofDSwdexGfeCCRJfQGHn2ihNU8oo9vzw6s9 +172: 5EYCAe5jLQhn6ofDSwdvhJDEJh6W7xNAWhfCvQpeFyojCkDK +173: 5EYCAe5jLQhn6ofDSweCSKkpRBmhaWU3FNbiUMWUiogTUUax +174: 5EYCAe5jLQhn6ofDSweUBMJQXgSu34Zuz3YE2JCKBdZBjxx9 +175: 5EYCAe5jLQhn6ofDSwejvNqzeB86VcfniiUjaEt9eTRv1hEQ +176: 5EYCAe5jLQhn6ofDSwf1fQPakfoHxAmfTPRF8BZz7HJeHEFa +177: 5EYCAe5jLQhn6ofDSwfHQRwAsAUVQisYC4Mkg8Fpa7BNZ735 +178: 5EYCAe5jLQhn6ofDSwfZ9TUkyf9gsGyQvjJGE4wf2w46pjEb +179: 5EYCAe5jLQhn6ofDSwfptV2M69ptKq5HfQEmn1dVVkvq6Jdm +180: 5EYCAe5jLQhn6ofDSwg6dWZwCeW5nPBAQ5BHKxKKxaoZMqLc +181: 5EYCAe5jLQhn6ofDSwgNNY7XK9BHEwH38k7nsu1ARQgHdMC3 +182: 5EYCAe5jLQhn6ofDSwge7Zf7RdrUhVNusR4JRqgztEZ1uEhR +183: 5EYCAe5jLQhn6ofDSwgurbChY8XgA3Unc5zoynNqM4RkAppF +184: 5EYCAe5jLQhn6ofDSwhBbckHedCscbafLkwKXj4fotJUSCzn +185: 5EYCAe5jLQhn6ofDSwhTLeHsm7t559gY5Rsq5fkWGiBChwy2 +186: 5EYCAe5jLQhn6ofDSwhj5fqTscZGXhnQp6pLdcSLjY3vyjeX +187: 5EYCAe5jLQhn6ofDSwhzphP3z7ETzFtHYmkrBZ8BCMvfFF73 +188: 5EYCAe5jLQhn6ofDSwiGZive6bufSozAHShMjVp1fBoPWsGG +189: 5EYCAe5jLQhn6ofDSwiYJkUED6aruN6327dsHSVr81g7nZdr +190: 5EYCAe5jLQhn6ofDSwip3n1pKbG4MvBuknaNqPBgaqYr4Epy +191: 5EYCAe5jLQhn6ofDSwj5noZQS5wFpUHnVTWtPKsX3fRaKfPX +192: 5EYCAe5jLQhn6ofDSwjMXq6zYacTH2PfE8TPwGZMWVJJbVUj +193: 5EYCAe5jLQhn6ofDSwjdGreaf5HejaVXxoPuVDFByKB2ryMD +194: 5EYCAe5jLQhn6ofDSwju1tCAmZxrC8bQhULR39w2S93m8YwD +195: 5EYCAe5jLQhn6ofDSwkAkujkt4e3eghHS9Gvb6crtxvVQ8TS +196: 5EYCAe5jLQhn6ofDSwkSVwHLzZKF7EoAApDS93JhMnoDfqYs +197: 5EYCAe5jLQhn6ofDSwkiExpw73zSZnu2uV9wgyzXpcfwwNtd +198: 5EYCAe5jLQhn6ofDSwkyyzNXDYfe2LzueA6TEvgNHSYgD9fh +199: 5EYCAe5jLQhn6ofDSwmFj1v7L3LqUu6nNq2xnsNCkGRQUdiJ +200: 5EYCAe5jLQhn6ofDSwmXU3ThSY22wTCf7VyULp43D6J8kSnP +201: 5EYCAe5jLQhn6ofDSwmoD51HZ2hEQ1JXrAuytkjsfvAs217d +202: 5EYCAe5jLQhn6ofDSwn4x6YsfXNRrZQQaqrVShRi8k3bHTHJ +203: 5EYCAe5jLQhn6ofDSwnLh86Tn23dK7WHKWnzze7YbZvKZ9W4 +204: 5EYCAe5jLQhn6ofDSwncS9e3tWipmfcA4BjWYaoP4Po3ptFu +205: 5EYCAe5jLQhn6ofDSwntBBBe11Q2EDi2nrg26XVDXDfn6UvC +206: 5EYCAe5jLQhn6ofDSwo9vCjE7W5DgmouXXcXeUB3z3YWN5rH +207: 5EYCAe5jLQhn6ofDSwoRfEGpDzkR9KunGCZ3CQrtSsREdeTi +208: 5EYCAe5jLQhn6ofDSwohQFpQLVRcbt1ezsVYkMYiuhHxuG9h +209: 5EYCAe5jLQhn6ofDSwoy9HMzSz6p4S7XjYS4JJEZNXAhB57Z +210: 5EYCAe5jLQhn6ofDSwpEtJuaZUn1WzDQUDNZrEvPqM3RShVi +211: 5EYCAe5jLQhn6ofDSwpWdLTAfyTCyYKHCtK5QBcEJAv9i5ua +212: 5EYCAe5jLQhn6ofDSwpnNMzknU8QS6R9wZFax8J4kznsyjCK +213: 5EYCAe5jLQhn6ofDSwq47PYLtxobteX2gEC6W4yuDpfcFKAe +214: 5EYCAe5jLQhn6ofDSwqKrR5w1TUoMCcuQu8c41fjgeYLXFWn +215: 5EYCAe5jLQhn6ofDSwqbbSdX7x9zokin9a57bxMa9UR4nnKt +216: 5EYCAe5jLQhn6ofDSwqsLUB7ESqCGJpetF1d9u3QcJHo4J89 +217: 5EYCAe5jLQhn6ofDSwr95VihLwWPirvXcux8hqjF58AXKq2x +218: 5EYCAe5jLQhn6ofDSwrQpXGHTSBbBR2QMateFnR5Xx3Fbkd6 +219: 5EYCAe5jLQhn6ofDSwrgZYosZvrndy8H6Fq9oj6uzmuys6og +220: 5EYCAe5jLQhn6ofDSwrxJaMTgRXz6XE9pvmfMfnkTbni8r8h +221: 5EYCAe5jLQhn6ofDSwsE3bu3nvDBZ5L2ZbiAucUavRfSQWNT +222: 5EYCAe5jLQhn6ofDSwsVndSduQtP1dRuJGegTZARPFYAg3e9 +223: 5EYCAe5jLQhn6ofDSwsmXezE1uZaUBXn2wbC1VrFr5QtwtdM +224: 5EYCAe5jLQhn6ofDSwt3GgXp8QEmvjdemcXhZSY6JuHdDNAJ +225: 5EYCAe5jLQhn6ofDSwtK1i5QEtuyPHjXWHUD7PDvmjAMUpH7 +226: 5EYCAe5jLQhn6ofDSwtakjczMPbAqqqQExQifKumEZ35kYLM +227: 5EYCAe5jLQhn6ofDSwtrVmAaTtGNJPwGydMEDGbbhNup2Aem +228: 5EYCAe5jLQhn6ofDSwu8EniAaNwZkx39iJHjmDHSACnYHqVA +229: 5EYCAe5jLQhn6ofDSwuPypFkgscmDW92SyEFK9yGd2fGZP4J +230: 5EYCAe5jLQhn6ofDSwufiqoLoNHxg4EuBeAks6f75rXzqFTr +231: 5EYCAe5jLQhn6ofDSwuwTsLvuryA8cLmvK7GR3LwYgQj6f8r +232: 5EYCAe5jLQhn6ofDSwvDCttX2MeMbASeez3mxz2n1WHTNE7n +233: 5EYCAe5jLQhn6ofDSwvUwvS78rKZ3iYXPezHWvicULABe9U8 +234: 5EYCAe5jLQhn6ofDSwvkgwyhFLzkWGeQ8Kvo4sQSwA2uudWf +235: 5EYCAe5jLQhn6ofDSww2RyXHMqfwxpkGrzsJcp6HPyueBKKS +236: 5EYCAe5jLQhn6ofDSwwJB14sULM9RNr9bfopAkn7ronNSouy +237: 5EYCAe5jLQhn6ofDSwwZv2cTaq2Lsvx2LLkKihTxKdf6iUQ9 +238: 5EYCAe5jLQhn6ofDSwwqf4A3hKhYLV3u51gqGe9nnTXpz5KG +239: 5EYCAe5jLQhn6ofDSwx7Q5hdopNjo39mogdLpaqdFHQZFjX1 +240: 5EYCAe5jLQhn6ofDSwxP97FDvK3wFbFeYMZrNXXTi7HHXSVQ +241: 5EYCAe5jLQhn6ofDSwxet8np2oj8i9MXH2WMvUDJAwA1nq7R +242: 5EYCAe5jLQhn6ofDSwxvdALQ9JQLAhTQ1hSsUQu8dm2k4jo3 +243: 5EYCAe5jLQhn6ofDSwyCNBszFo5XdFZGkNPP2May6auULR3m +244: 5EYCAe5jLQhn6ofDSwyU7DRaNHkj5of9V3KtaJGoZQnCbxRj +245: 5EYCAe5jLQhn6ofDSwyjrEyAUnRvYMm2DiGQ8Exe2Eevsbos +246: 5EYCAe5jLQhn6ofDSwz1bGWkbH77zurtxPCugBeUV4Xf9FEy +247: 5EYCAe5jLQhn6ofDSwzHLJ4LhmnKTTxmh49RE8LJwtQPQuPs +248: 5EYCAe5jLQhn6ofDSwzZ5KbvpGTWv24eRj5vn529QiH7gWns +249: 5EYCAe5jLQhn6ofDSwzppM9Wvm8iNaAXAQ2SL1hysY9qx3ao +250: 5EYCAe5jLQhn6ofDSx16ZNh73Fouq8GPu4xwsxPpLN2aDciL +251: 5EYCAe5jLQhn6ofDSx1NJQEh9kV7HgNGdjuTRu5eoBuJVHP9 +252: 5EYCAe5jLQhn6ofDSx1e3RnHGFAJkEU9NQqxyqmVG1n2kz46 +253: 5EYCAe5jLQhn6ofDSx1unTKsNjqWCna275nUXnTKiqem2beu +254: 5EYCAe5jLQhn6ofDSx2BXUsTVEWhfLftqkiz5j9ABfXVHzQ2 +255: 5EYCAe5jLQhn6ofDSx2TGWR3bjBu7tmmaRfVdfpzeVQDZscZ +256: 5EYCAe5jLQhn6ofDSvqFAHPozxdzP66nvfzRhb4qEUHpQXPv +257: 5EYCAe5jLQhn6ofDSvqWuJwQ7TKBqeCffLvwFXkfhJAYgD3P +258: 5EYCAe5jLQhn6ofDSvqneLUzDwzPJCJYQ1sSoUSWA83GwgQt +259: 5EYCAe5jLQhn6ofDSvr4PN2aLSfakkQR8goxMR8Lcwv1DSbq +260: 5EYCAe5jLQhn6ofDSvrL8PaASwLnDJWHsMkTuMpB5mnjUseM +261: 5EYCAe5jLQhn6ofDSvrbsR7kZS1yfrcAc2gyTJW1YbfTkXuS +262: 5EYCAe5jLQhn6ofDSvrscSfLfvhB8Qi3LhdV1FBr1RYC28tL +263: 5EYCAe5jLQhn6ofDSvs9MUCvnRNNaxov5NZzZBsgUFQvHj55 +264: 5EYCAe5jLQhn6ofDSvsR6VkWtv3a3Wunp3WW78ZWw5HeZQAP +265: 5EYCAe5jLQhn6ofDSvsgqXJ71QimW51fYiT1f5FMPuANqDNX +266: 5EYCAe5jLQhn6ofDSvsxaYqh7uPxxd7YHPPXD1wBrj376tWY +267: 5EYCAe5jLQhn6ofDSvtEKaPHEQ5ARBDR24L2kxd2KYuqNPar +268: 5EYCAe5jLQhn6ofDSvtW4bvsLtkMsjKHkjGYJuJrnNnZe4bY +269: 5EYCAe5jLQhn6ofDSvtmodUTTPRZLHRAVQD3rqzhFCfHuYa9 +270: 5EYCAe5jLQhn6ofDSvu3Yf23Zt6knqX3E59ZQngXi2Y2BNLJ +271: 5EYCAe5jLQhn6ofDSvuKHgZdgNmxFPcuxk64xjNNArQkSnAa +272: 5EYCAe5jLQhn6ofDSvub2i7DnsT9hwinhR2aWg4CdgHUiRdK +273: 5EYCAe5jLQhn6ofDSvurmjeouN8MAVpfS5y64ck36WACzBoE +274: 5EYCAe5jLQhn6ofDSvv8WmCQ1roYd3vYAkubcZRsZL2wFmUp +275: 5EYCAe5jLQhn6ofDSvvQFnjz8MUk5c2QuRr7AW7i29ufXLBs +276: 5EYCAe5jLQhn6ofDSvvfzpHaEr9wYA8He6nciSoYUynPo1wJ +277: 5EYCAe5jLQhn6ofDSvvwjqqAMLq8ziEANmj8GPVNwof84VTF +278: 5EYCAe5jLQhn6ofDSvwDUsNkTqWLTGL37SfdpLBDQdXrLLVP +279: 5EYCAe5jLQhn6ofDSvwVDtvLaLBXupRur7c9NGs3sTQabobG +280: 5EYCAe5jLQhn6ofDSvwkxvTvgprjNNXnanYevDYtLHHJsUWQ +281: 5EYCAe5jLQhn6ofDSvx2hx1WoKXvpvdfKTVAUAEio7A394VB +282: 5EYCAe5jLQhn6ofDSvxJSyZ6upD8HUjY48Rg26vZFw2mQtmG +283: 5EYCAe5jLQhn6ofDSvxaC16h2JtKk2qQnoNBa3cPikuVgH2a +284: 5EYCAe5jLQhn6ofDSvxqw2eH8oZXCawHXUJh7zJEBanDwyyQ +285: 5EYCAe5jLQhn6ofDSvy7g4BsFJEif93AG9FCfvz4eQexDirh +286: 5EYCAe5jLQhn6ofDSvyPR5jTMnuv7h92zpBiDsfu7EXgVLDf +287: 5EYCAe5jLQhn6ofDSvyfA7H3UHb7aFEujV8DmpMja4QQknoB +288: 5EYCAe5jLQhn6ofDSvyvu8pdanGK2oLnUA4jKm3a2tH92NjE +289: 5EYCAe5jLQhn6ofDSvzCeANDhGwWVMSfCq1EshjQVi9sJ6eA +290: 5EYCAe5jLQhn6ofDSvzUPBuoomchwuYXwVwkReRExY2bZe1S +291: 5EYCAe5jLQhn6ofDSvzk8DTPvGHuQTeQgAtFyb75RMuKqVPu +292: 5EYCAe5jLQhn6ofDSw11sEzz2ky6s1kHQqpmXXnutBn46shJ +293: 5EYCAe5jLQhn6ofDSw1HcGYa9FeJKZrA9WmH5UUkM1enNeCq +294: 5EYCAe5jLQhn6ofDSw1ZMJ6AFkKVn7x2tBhndRAaoqXWeDgU +295: 5EYCAe5jLQhn6ofDSw1q6KdkNEzhEg3ucreJBMrRGfQEuqcy +296: 5EYCAe5jLQhn6ofDSw26qMBLUjfthE9nMXaojJYFjVGyBbDv +297: 5EYCAe5jLQhn6ofDSw2NaNivbEM69nFf6CXKHFE6CK9hT1az +298: 5EYCAe5jLQhn6ofDSw2eKQGWhj2HcLMXpsTpqBuvf92Rie2g +299: 5EYCAe5jLQhn6ofDSw2v4Rp6pDhV4tTQZYQLP8bm7xu9zEfU +300: 5EYCAe5jLQhn6ofDSw3BoTMgviNgXSZHJDLqw5HbanmtFw81 +301: 5EYCAe5jLQhn6ofDSw3TYUuH3D3syzfA2tHMV1yS3cecXkzL +302: 5EYCAe5jLQhn6ofDSw3jHWSs9hj5SYm2mZDs2xfGWSXLoPFN +303: 5EYCAe5jLQhn6ofDSw412XzTGCQGu6ruWEANauM6yGQ54y3V +304: 5EYCAe5jLQhn6ofDSw4GmZY3Nh5UMexnEu6t8r2wS6GoLd6G +305: 5EYCAe5jLQhn6ofDSw4YWb5dVBkfpD4eya3Pgnimtv9XcAHv +306: 5EYCAe5jLQhn6ofDSw4pFcdDbgRsGmAXiEyuEjQcMk2Fso4J +307: 5EYCAe5jLQhn6ofDSw55zeAoiB74jKGQSuvQng6SpZtz9Vxy +308: 5EYCAe5jLQhn6ofDSw5MjfiPpfnGBsNHBarvLcnHHPmiQteL +309: 5EYCAe5jLQhn6ofDSw5dUhFywATTeRU9vFoRtZU7kDeSgbg2 +310: 5EYCAe5jLQhn6ofDSw5uDioa3f8f6ya2evjwSW9xD3XAxAx1 +311: 5EYCAe5jLQhn6ofDSw6AxkMAA9orZXfuPbgSzSqnfsPuDqwL +312: 5EYCAe5jLQhn6ofDSw6ShmtkGeV425mn8GcxYPXd8hGdVc6w +313: 5EYCAe5jLQhn6ofDSw6iSoSLP9AFUdserwZU6LDTbX9Mm6rm +314: 5EYCAe5jLQhn6ofDSw6zBpyvVdqSwByXbcVyeGuJ4M262q6Q +315: 5EYCAe5jLQhn6ofDSw7FvrXWc8WePk5QLHSVCDb8XAtpJXc2 +316: 5EYCAe5jLQhn6ofDSw7Xft56idBqrJBH4xNzkAGxyzmYa7Ff +317: 5EYCAe5jLQhn6ofDSw7oQucgq7s3JrH9odKWJ6xoSpeGqdqm +318: 5EYCAe5jLQhn6ofDSw859wAGwcYEmQP2YJG1r3edueX17Sh6 +319: 5EYCAe5jLQhn6ofDSw8Ltxhs47DSDxUuGyCXPzLUNUPjP1sE +320: 5EYCAe5jLQhn6ofDSw8cdzFTAbtdgWan1e92ww2JqJGTedjU +321: 5EYCAe5jLQhn6ofDSw8tP1o3H6Zq94gekK5YVsi9J89Bv64S +322: 5EYCAe5jLQhn6ofDSw9A83LdPbF2bcnXUz243pPykx1vBosi +323: 5EYCAe5jLQhn6ofDSw9Rs4tDW5vE4AtQDexZbm5pDmteTYXr +324: 5EYCAe5jLQhn6ofDSw9hc6RocabRWizGxKu59hmegbmNj178 +325: 5EYCAe5jLQhn6ofDSw9yM7yPj5GcyH69gzqaheTV9Re6zjMq +326: 5EYCAe5jLQhn6ofDSwAF69WyqZwpRqC2Rfn6Fb9KcFWqGBTn +327: 5EYCAe5jLQhn6ofDSwAWqB4Zx4d1tPHuALiboXqA55PZXq5E +328: 5EYCAe5jLQhn6ofDSwAnaCcA4ZJDLwPmu1f7MUWzXuGHoXBP +329: 5EYCAe5jLQhn6ofDSwB4KE9kB3yQoVVedgbcuRCpzj9254yw +330: 5EYCAe5jLQhn6ofDSwBL4FhLHYecG3bXNMY8TMtfTZ1kLpho +331: 5EYCAe5jLQhn6ofDSwBboHEvQ3KoibhQ72Ue1JaVvNtUcXUJ +332: 5EYCAe5jLQhn6ofDSwBsYJnWWY11B9oGqhR9ZFGLPCmCt89h +333: 5EYCAe5jLQhn6ofDSwC9HLL6d2gCdhu9aNMf7BxAr2dw9ppe +334: 5EYCAe5jLQhn6ofDSwCR2MsgjXMQ6G12K3JAf8e1JrWfRNDE +335: 5EYCAe5jLQhn6ofDSwCgmPRGr22bYp6u3iEgD5KqmgPPh3hX +336: 5EYCAe5jLQhn6ofDSwCxWQxrxWho1NCmnPBBm21gEWG7xc6b +337: 5EYCAe5jLQhn6ofDSwDEFSWT51NzTvJeX47hJxhWhL8rE71F +338: 5EYCAe5jLQhn6ofDSwDVzU43BW4BvUQXFj4CruPMAA1aVkSJ +339: 5EYCAe5jLQhn6ofDSwDmjVbdHzjPP2WPzPziQr5BcytJmWaH +340: 5EYCAe5jLQhn6ofDSwE3UX9DQVQaqacGj4wDxnm25om334HL +341: 5EYCAe5jLQhn6ofDSwEKDYgoWz5nJ8i9TjsjWjSrYddmJfht +342: 5EYCAe5jLQhn6ofDSwEaxaEPdUkykgp2CQpF4g8h1TWVaF57 +343: 5EYCAe5jLQhn6ofDSwErhbmyjySBDEutw5kkccpXUHPDqpD4 +344: 5EYCAe5jLQhn6ofDSwF8SdKZrU7Nfo1mfkhGAZWMw7Fx7Wbu +345: 5EYCAe5jLQhn6ofDSwFQBes9xxna8M7eQRdmiWCCPw8gPAnZ +346: 5EYCAe5jLQhn6ofDSwFfvgQk5TTmauDX96aHGSt2rm1QejQE +347: 5EYCAe5jLQhn6ofDSwFwfhxLBx8y3TKPsmWnpPZsKat8vLpQ +348: 5EYCAe5jLQhn6ofDSwGDQjVvJSpAW1RGcSTJNLFhnQksBuxZ +349: 5EYCAe5jLQhn6ofDSwGV9m3WQwVMxZX9M7PovGwYFEdbTqLQ +350: 5EYCAe5jLQhn6ofDSwGktnb6XSAZR7d25nLKUDdNi4WKjHVC +351: 5EYCAe5jLQhn6ofDSwH2dp8gdvqksfitpTGq2AKDAtP416pX +352: 5EYCAe5jLQhn6ofDSwHJNqgGkRWxLDpmZ8DLa713diFnGYZH +353: 5EYCAe5jLQhn6ofDSwHa7sDrrvC9nmveHo9r83gt6Y8WYAUE +354: 5EYCAe5jLQhn6ofDSwHqrtmSyQsMFL2X2U6MfzNiZN1EoxJG +355: 5EYCAe5jLQhn6ofDSwJ7bvK35uYYht8Pm92sDw4Z2Bsy5JYd +356: 5EYCAe5jLQhn6ofDSwJPLwrdCQDkASEGVoyNmskPV1khM3xf +357: 5EYCAe5jLQhn6ofDSwJf5yQDJttwczL9EUutKpSDwqdRcYKN +358: 5EYCAe5jLQhn6ofDSwJvpzwoRPa95YS1y9rPsm84QfW9tMQi +359: 5EYCAe5jLQhn6ofDSwKCa2VPXtFLY6XthpnuRhotsVNtA1nt +360: 5EYCAe5jLQhn6ofDSwKUK42yeNvXzedmSVjQyeVjLKFcRURc +361: 5EYCAe5jLQhn6ofDSwKk45aZksbjTCjeBAfvXbBZo98LhAFX +362: 5EYCAe5jLQhn6ofDSwL1o789sNGvukqWuqcS5XsQFy14xope +363: 5EYCAe5jLQhn6ofDSwLHY8fjyrx8NJwPeWYwdUZEinsoEZUW +364: 5EYCAe5jLQhn6ofDSwLZHADL6MdKps3GPBVTBRF5BckXW7BE +365: 5EYCAe5jLQhn6ofDSwLq2BkvCrJXHR997rRxjMvueSdFmhcW +366: 5EYCAe5jLQhn6ofDSwM6mDJWKLyijyF1rXNUHJck7GVz3PQs +367: 5EYCAe5jLQhn6ofDSwMNWEr6RqevCXLtbCJyqFJaa6NiJyR1 +368: 5EYCAe5jLQhn6ofDSwMeFGPgYLL7f5SmKsFVPBzR2vFSaVaJ +369: 5EYCAe5jLQhn6ofDSwMuzHwGeq1K7dYe4YBzw8gFVk8Ar7jR +370: 5EYCAe5jLQhn6ofDSwNBjKUrmKgWaBeWoD8WV5N5xZzu7st7 +371: 5EYCAe5jLQhn6ofDSwNTUM2SspMi2jkPXt52323vRPsdPTi7 +372: 5EYCAe5jLQhn6ofDSwNjDNa2zK2uVHrGGZ1XaxjktDkMezQq +373: 5EYCAe5jLQhn6ofDSwNzxQ7d6oi6wqx91Dx38uRbM3d5vkb5 +374: 5EYCAe5jLQhn6ofDSwPGhRfDDJPJQQ41jttYgr7RosVpCSnm +375: 5EYCAe5jLQhn6ofDSwPYSTCoKo4Vrx9tUZq4EnoGGhNYTsHL +376: 5EYCAe5jLQhn6ofDSwPpBUkPSHjhKWFmDEmZnjV6jXFGjavn +377: 5EYCAe5jLQhn6ofDSwQ5vWHyYnQtn4Mdwui5LgAwCM811FqV +378: 5EYCAe5jLQhn6ofDSwQMfXqZfH66EcTWgaeatcrmfAzjGwwQ +379: 5EYCAe5jLQhn6ofDSwQdQZP9mmmHhAZPRFb6SZYc7zsTYQ3S +380: 5EYCAe5jLQhn6ofDSwQu9avjtGSV9ifG9vXbzWESapkBpFWq +381: 5EYCAe5jLQhn6ofDSwRAtcUKzm7gcGm8tbU7YSvH3ecv5fQY +382: 5EYCAe5jLQhn6ofDSwRSde1v7Fnt4ps1dGQd6Pc7WUVeMKqG +383: 5EYCAe5jLQhn6ofDSwRiNfZWDkU5XNxtMwM8eLHwyJNNd1x9 +384: 5EYCAe5jLQhn6ofDSwRz7h76LF9Gyw4m6cHeCGynS8F6tUu4 +385: 5EYCAe5jLQhn6ofDSwSFriegSjpUSVAdqHE9kDfctx7qAGqb +386: 5EYCAe5jLQhn6ofDSwSXbkCGZEVfu3GWZxAfJAMTMmzZRjJj +387: 5EYCAe5jLQhn6ofDSwSoLmjrfjAsMbNPJd7Ar73HpbsHhKSv +388: 5EYCAe5jLQhn6ofDSwT55oHSnDr4p9UG3J3gQ3j8HRk1y3JJ +389: 5EYCAe5jLQhn6ofDSwTLppq2tiXGGha8mxzBwzQxkFckEpvZ +390: 5EYCAe5jLQhn6ofDSwTcZrNd1DCTjFg1WdvhVw6oD5VUWKxy +391: 5EYCAe5jLQhn6ofDSwTtJsvD7hsfBomtFJsD3sndfuNCmsJH +392: 5EYCAe5jLQhn6ofDSwUA3uToECYreMskyyoibpUU8jEw3T5N +393: 5EYCAe5jLQhn6ofDSwURnw1PLhE46uydiekE9mAJbZ7fKHH5 +394: 5EYCAe5jLQhn6ofDSwUhXxYyTBuFZU5WTKgjhhr94NzPakX9 +395: 5EYCAe5jLQhn6ofDSwUyGz6ZZgaT22BPBzdFFeXyXCs7rczx +396: 5EYCAe5jLQhn6ofDSwVF21e9gBFeUaHFvfZkobDoz2jr8F9t +397: 5EYCAe5jLQhn6ofDSwVWm3Bjnfvqw8P8fLWGMXueSrcaPq1Z +398: 5EYCAe5jLQhn6ofDSwVnW4jKuAc3PgV1Q1SmuUbUugVJfNVA +399: 5EYCAe5jLQhn6ofDSwW4F6Gv1fHErEat8gPHTRHKNWN2w5vR +400: 5EYCAe5jLQhn6ofDSwWKz7pW89xSJngksMKo1My9qLEmCh5V +401: 5EYCAe5jLQhn6ofDSwWbj9N6EeddmLndc2GJZJezJA7VUKPY +402: 5EYCAe5jLQhn6ofDSwWsUAugM9JqDttWLhCp7FLpkyzDjzxV +403: 5EYCAe5jLQhn6ofDSwX9DCTGTdz2gSzP5N9KfC2fDorx1RVV +404: 5EYCAe5jLQhn6ofDSwXQxDzra8fE916Fp35qD8iVgdjgHGPX +405: 5EYCAe5jLQhn6ofDSwXghFYSgdLRbZC8Yi2Lm5QL9TcQYr74 +406: 5EYCAe5jLQhn6ofDSwXxSH62o81d47J1HNxrK26AcHV8pFSS +407: 5EYCAe5jLQhn6ofDSwYEBJdcucgpWfPt23uMrxn157Ms5xvh +408: 5EYCAe5jLQhn6ofDSwYVvLBD27N1yDVkkiqsQuTqXwEbMX6v +409: 5EYCAe5jLQhn6ofDSwYmfMio8c3DRmbdVPnNxr9fzm7Kd75S +410: 5EYCAe5jLQhn6ofDSwZ3QPGPF6iQtKhWE4itWnqWTaz3tiZd +411: 5EYCAe5jLQhn6ofDSwZK9QoyMbPcLsoNxjfQ4jXLvQrnAT96 +412: 5EYCAe5jLQhn6ofDSwZatSMZU64ooRuFhQbucgDBPEjWS46h +413: 5EYCAe5jLQhn6ofDSwZrdTu9aak1Fz18S5YRAcu1r4cEhoLd +414: 5EYCAe5jLQhn6ofDSwa8NVSjh5RCiY71AkUviZarJtUxyQo6 +415: 5EYCAe5jLQhn6ofDSwaQ7WzKoa6QB6CsuRRSGWGgmiMhEyuu +416: 5EYCAe5jLQhn6ofDSwafrYXuv4mbdeJke6MwpSxXEYERWhTt +417: 5EYCAe5jLQhn6ofDSwawba5W2ZSo6CQdNmJTNPeMhN79n8Xw +418: 5EYCAe5jLQhn6ofDSwbDLbd6947zYkWW7SExvLLCAByt411A +419: 5EYCAe5jLQhn6ofDSwbV5dAgFYoC1JcNr7BUUH22d1rcKXhx +420: 5EYCAe5jLQhn6ofDSwbkpeiGN3UPTriFan7z2Dhs5qjLbAG1 +421: 5EYCAe5jLQhn6ofDSwc2ZgFrUY9avQp8KT4VaAPhYfc4rtSD +422: 5EYCAe5jLQhn6ofDSwcJJhoSb2pnNxv14811875Y1VUo8Udd +423: 5EYCAe5jLQhn6ofDSwca3jM2hXVyqX1snnwWg3mNUKMXQ7Lu +424: 5EYCAe5jLQhn6ofDSwcqnktcp2BBJ57kXTt2DzTCw9EFfjeB +425: 5EYCAe5jLQhn6ofDSwd7XnSCvWrNkdDdG8pXmw93Py6ywNA9 +426: 5EYCAe5jLQhn6ofDSwdPGoyo31XaDBKVzom3KspsrnyiCoBH +427: 5EYCAe5jLQhn6ofDSwdf1qXP9WCmfjRNjUhYspWiKcrSUZJy +428: 5EYCAe5jLQhn6ofDSwdvks4yFzsy8HXFU9e4RmCYnSjAkGv5 +429: 5EYCAe5jLQhn6ofDSweCVtcZNVZAaqd8CpaZyhtPFGbu1r89 +430: 5EYCAe5jLQhn6ofDSweUEvA9UzEN3PizwVX5XeaDi6UdHRW8 +431: 5EYCAe5jLQhn6ofDSwejywhjbUuZVwpsgATb5bG4AvMMYvzn +432: 5EYCAe5jLQhn6ofDSwf1iyFKhyakxVvkQqQ6dXwtdkE5pWEq +433: 5EYCAe5jLQhn6ofDSwfHTznupUFxR42d9WLcBUdj6a6p6RJP +434: 5EYCAe5jLQhn6ofDSwfZD2LVvxw9sc8VtBH7jRKZZPyYN3Qe +435: 5EYCAe5jLQhn6ofDSwfpx3t63TcMLAENcrDdHN1Q2DrGdXnV +436: 5EYCAe5jLQhn6ofDSwg6h5Rg9xHYniLFMXA8qJhEV3izuDDr +437: 5EYCAe5jLQhn6ofDSwgNS6yGGSxkFGS86C6ePFP4wsbjAhNK +438: 5EYCAe5jLQhn6ofDSwgeB8WrNwdwhpXzps39wC4uQhUTSNuL +439: 5EYCAe5jLQhn6ofDSwguvA4SVSK9ANdsZXyfV8kjsXMBi2rM +440: 5EYCAe5jLQhn6ofDSwhBfBc2bvzLcvjkJCvB35SaLMDuyoqD +441: 5EYCAe5jLQhn6ofDSwhTQD9ciRfY5Uqd2srgb28QoB6eFK1Z +442: 5EYCAe5jLQhn6ofDSwhj9EhCpvLjY2wVmYoC8xpFFzyNWvEr +443: 5EYCAe5jLQhn6ofDSwhztGEnwR1vzb3NWDjhguW5ipr6nXHN +444: 5EYCAe5jLQhn6ofDSwiGdHnP3uh8T99FEtgDErBvBeiq45cG +445: 5EYCAe5jLQhn6ofDSwiYNKKyAQNKuhF7yZcinnskeUbZKgnF +446: 5EYCAe5jLQhn6ofDSwip7LsZGu3XNFLziEZELjZb7JUHbZzu +447: 5EYCAe5jLQhn6ofDSwj5rNR9PPiipoSsSuVjtgFRa8M1s5jf +448: 5EYCAe5jLQhn6ofDSwjMbPxjVtPvHMYkBaSFScwG2xDk8eCi +449: 5EYCAe5jLQhn6ofDSwjdLRWKcP57juecvFNkzZd6Vn6UQFZC +450: 5EYCAe5jLQhn6ofDSwju5T3uiskKCTkVevKGYWJvxbyCg4Jv +451: 5EYCAe5jLQhn6ofDSwkApUbVqNRWf1rNPbFn6SzmRRqvwSdW +452: 5EYCAe5jLQhn6ofDSwkSZW95ws6i7ZxF8GCHePgbtFifDEja +453: 5EYCAe5jLQhn6ofDSwkiJXgg4Mmua847rw8oCLNSM5bPUoyW +454: 5EYCAe5jLQhn6ofDSwkz3ZEGArT72g9zbc5JkH4GouU7kW6c +455: 5EYCAe5jLQhn6ofDSwmFnamrHM8JVEFsLH1pJDk7GjLr27EN +456: 5EYCAe5jLQhn6ofDSwmXXcKSPqoVwnMk4wxKrARwjZDaHjGj +457: 5EYCAe5jLQhn6ofDSwmoGds2WLUhQLTcoctqQ77nCP6JZSt5 +458: 5EYCAe5jLQhn6ofDSwn51fQccq9trtZVYHqLx3ocfCy2pusw +459: 5EYCAe5jLQhn6ofDSwnLkgxCjKq6KSfNGxmrVzVT82qm6X4V +460: 5EYCAe5jLQhn6ofDSwncViVnqpWHmzmF1diN3wBHariVNDY4 +461: 5EYCAe5jLQhn6ofDSwntEk3NxKBVEYs7kJesbss83gbDdh5y +462: 5EYCAe5jLQhn6ofDSwo9ymay4orgh6xzUybP9pYxWWTwuZdk +463: 5EYCAe5jLQhn6ofDSwoRio8ZBJXt9f4sDeXthmEnyLLgAtRS +464: 5EYCAe5jLQhn6ofDSwohTpg9HoD5cDAjxKUQFhvdSADQSd7q +465: 5EYCAe5jLQhn6ofDSwoyCrDjQHtH4mGcgzQuoecTtz68iBNB +466: 5EYCAe5jLQhn6ofDSwpEwsmKWnZUXKNVRfMRMbJJMoxrz5ZA +467: 5EYCAe5jLQhn6ofDSwpWguJudHEfysUNALHvuXz8pdqbFU2C +468: 5EYCAe5jLQhn6ofDSwpnRvrVjmusSRaEu1ESTUfyHTiKX3yD +469: 5EYCAe5jLQhn6ofDSwq4AxQ5rGb4tyg7dgAx1RMokHb3nmAN +470: 5EYCAe5jLQhn6ofDSwqKuywfxmGGMXmzNM7TZN3eD7Tn4RDB +471: 5EYCAe5jLQhn6ofDSwqbf1VG5FwTp5ss723y7JjUfwLWL6mj +472: 5EYCAe5jLQhn6ofDSwqsQ32rBkcfGdyjqgzUfFRK8mDEbpVV +473: 5EYCAe5jLQhn6ofDSwr994aSJFHrjC5caMvzDC79bb5xsU5M +474: 5EYCAe5jLQhn6ofDSwrQt682Qjy4BkBVK2sVm8nz4Qxh8rdZ +475: 5EYCAe5jLQhn6ofDSwrgd7fcXEeFeJHN3hp1K5UpXEqRQQvd +476: 5EYCAe5jLQhn6ofDSwrxN9DCdjKT6rPEnNkWs2Aez4i9g7hy +477: 5EYCAe5jLQhn6ofDSwsE7AknkDzeZQV7X3h2QxrVStaswkrq +478: 5EYCAe5jLQhn6ofDSwsVrCJNrifr1xazFidXxuYKuiTcDVjH +479: 5EYCAe5jLQhn6ofDSwsmbDqxyDM3UWgrzPa3WrEANYLLUw1X +480: 5EYCAe5jLQhn6ofDSwt3LFPZ5i2Ew4njj4WZ4nuzqND4kbZN +481: 5EYCAe5jLQhn6ofDSwtK5Gw9CChSPctcTjT4cjbqJC5o2PJV +482: 5EYCAe5jLQhn6ofDSwtapJUjJhNdrAzVCQPaAgHfm1xXHtKK +483: 5EYCAe5jLQhn6ofDSwtrZL2KRC3qJj6Mw5L5icyWDqqFZadE +484: 5EYCAe5jLQhn6ofDSwu8JMZuXgj2mHCEfkGbGZfLgfhyqAhn +485: 5EYCAe5jLQhn6ofDSwuQ3P7VeBQEDqJ7QRD6pWMB9Vai6mth +486: 5EYCAe5jLQhn6ofDSwufnQf5kg5RgPPz969cNT31cKTSNSCS +487: 5EYCAe5jLQhn6ofDSwuwXSCfsAkd8wVrsm67vPir59LAe42A +488: 5EYCAe5jLQhn6ofDSwvDGTkFyfRpbVbjcS2dULQgXyCtubBX +489: 5EYCAe5jLQhn6ofDSwvV1VHr6A7243hcM6y92H6Wzo5dBP9F +490: 5EYCAe5jLQhn6ofDSwvkkWqSCenDWboV5mueaDnMTcxMT382 +491: 5EYCAe5jLQhn6ofDSww2VYP2K9TQy9uMpSrA8AUBvSq5iXaj +492: 5EYCAe5jLQhn6ofDSwwJEZvcRe8cRi1EZ7nfg7A2PGhoz9d6 +493: 5EYCAe5jLQhn6ofDSwwZybUCY8ootG77HnjBE3qrr6aYFvzB +494: 5EYCAe5jLQhn6ofDSwwqid1nedV1LpCz2TfgmzXhJvTGXUBY +495: 5EYCAe5jLQhn6ofDSwx7TeZNm8ACoNJrm8cCKwDXmkKznwLK +496: 5EYCAe5jLQhn6ofDSwxPCg6xscqQFvQjVoYhssuNEaCj4cV1 +497: 5EYCAe5jLQhn6ofDSwxewheYz7WbiUWcEUVDRpbChQ5TLKHR +498: 5EYCAe5jLQhn6ofDSwxvgjC96cBoB2cUy9RiymH3ADxBc4Lb +499: 5EYCAe5jLQhn6ofDSwyCRkjjD6rzdaiMhpNEXhxsd3pusccP +500: 5EYCAe5jLQhn6ofDSwyUAnHKKbYC68pESVJk5eei5she97qy +501: 5EYCAe5jLQhn6ofDSwyjuopuS6DPYgv7BAFFdbLYYhaNQuf9 +502: 5EYCAe5jLQhn6ofDSwz1eqNVYatb1F1yuqBmBY2P1XT6gLkd +503: 5EYCAe5jLQhn6ofDSwzHPrv5f5ZnTo7reW8GjUiDUMKpwzz6 +504: 5EYCAe5jLQhn6ofDSwzZ8tTfmaEyvMDjPB4nHRQ3wBCZDsZs +505: 5EYCAe5jLQhn6ofDSwzpsv1Ft4vBNuKc7r1HqN5tQ15HVCrz +506: 5EYCAe5jLQhn6ofDSx16cwYqzZbNqTRUrWwoPJmirpx1kybJ +507: 5EYCAe5jLQhn6ofDSx1NMy6S74GaJ1XMbBtJwFTZKepk2ghP +508: 5EYCAe5jLQhn6ofDSx1e6ze2DYwmkZdEKrppVC9PnUhUJ51w +509: 5EYCAe5jLQhn6ofDSx1ur2BcL3cyD7j74XmL38qEFJaCZfuR +510: 5EYCAe5jLQhn6ofDSx2Bb3jCSYJAffpyoChqb5X4i8SvqbpQ +511: 5EYCAe5jLQhn6ofDSx2TL5GnZ2yN8DvrXseM92CuAxKf7ELV +512: 5EYCAe5jLQhn6ofDSvqFDrFYxGRTPRFst7yHCwSjkwDFwear +513: 5EYCAe5jLQhn6ofDSvqWxso94m6eqyMkcnunkt8aDm5zDaY6 +514: 5EYCAe5jLQhn6ofDSvqnhuLjBFmrJXTdMTrJJppQgaxiV33E +515: 5EYCAe5jLQhn6ofDSvr4SvtKHkT3m5ZW68normWF9QqSkkTc +516: 5EYCAe5jLQhn6ofDSvrLBxRuQF8FDdfNpojKQiC5cEiB2Sk4 +517: 5EYCAe5jLQhn6ofDSvrbvyyVWjoSgBmFZUfpxesv54auJ2W6 +518: 5EYCAe5jLQhn6ofDSvrsg1X5dEUe8js8J9cLWbZkXtTdZh9r +519: 5EYCAe5jLQhn6ofDSvs9R34fjj9qbHy12pYr4YFaziLMqJTN +520: 5EYCAe5jLQhn6ofDSvsRA4cFrDq33r4smVVMcUwRTYD66mFA +521: 5EYCAe5jLQhn6ofDSvsgu69qxiWEWQAkWARsARdFvN5pNJkh +522: 5EYCAe5jLQhn6ofDSvsxe7hS5DBRxxGdEqNNiNK6PBxYdy7w +523: 5EYCAe5jLQhn6ofDSvtEP9F2BhrdRWNVyWJtGJzvr1qGuotj +524: 5EYCAe5jLQhn6ofDSvtW8AncJCXpt4UNiBFPpFgmJqi1BLVs +525: 5EYCAe5jLQhn6ofDSvtmsCLCQhD2LcaFSrBuNCNbmfajSpjE +526: 5EYCAe5jLQhn6ofDSvu3cDsnXBtDoAg8BX8Qv94SEVTTiiQ6 +527: 5EYCAe5jLQhn6ofDSvuKMFRNdgZRFimzvC4vU5kGhKLBzAn6 +528: 5EYCAe5jLQhn6ofDSvub6GxxkBEciGsses1S22S7A9CvFx62 +529: 5EYCAe5jLQhn6ofDSvurqJWYrfupApykPXwwZy7wcy5eXHgT +530: 5EYCAe5jLQhn6ofDSvv8aL48yAb1dP5d8CtT7uon5nxNoAHh +531: 5EYCAe5jLQhn6ofDSvvQKMbj5fGD5wBVrspxfrVcYcq74cff +532: 5EYCAe5jLQhn6ofDSvvg4P9KC9wQYVHNbYmUDoBT1ShqLTbC +533: 5EYCAe5jLQhn6ofDSvvwoQguJecc13PFLDhymjsHUGaZc2LL +534: 5EYCAe5jLQhn6ofDSvwDYSEVR9HoTbV84teVKgZ7w6THsXG4 +535: 5EYCAe5jLQhn6ofDSvwVHTn5Xdxzv9azoZazsdExPvL29Fyn +536: 5EYCAe5jLQhn6ofDSvwm2VKfe8eCNhgsYEXWRZvnrkCkQm3z +537: 5EYCAe5jLQhn6ofDSvx2mWsFkdKPqFnkGuU1yWcdKa5UgYgt +538: 5EYCAe5jLQhn6ofDSvxJWYQqs7zbHotd1aQXXTJTnPxCx4qw +539: 5EYCAe5jLQhn6ofDSvxaFZxRycfnkMzVkFM35PzJFDpwDdau +540: 5EYCAe5jLQhn6ofDSvxqzbW267LzCv6NUvHYdLg8i3hfVTFu +541: 5EYCAe5jLQhn6ofDSvy7jd3cCc2BfUCFDbE4BHMyAsaPm1sJ +542: 5EYCAe5jLQhn6ofDSvyPUebCK6hP82J7xGAZjE3odhT82USX +543: 5EYCAe5jLQhn6ofDSvyfDg8nRbNaaaPzgw75HAje6XKrJNeZ +544: 5EYCAe5jLQhn6ofDSvyvxhgNY63n38VsRc3aq7RUZMCaZzZ9 +545: 5EYCAe5jLQhn6ofDSvzChjDxeaiyVgbkAGz6P47K2B5JqJmC +546: 5EYCAe5jLQhn6ofDSvzUSkmYm5QAxEhctwvbvzo9Uzx375yn +547: 5EYCAe5jLQhn6ofDSvzkBnK8sa5NQnoVdcs7UwUywppmNkyL +548: 5EYCAe5jLQhn6ofDSw11voriz4kZsLuNNHod2tApQehVeFiH +549: 5EYCAe5jLQhn6ofDSw1HfqQK6ZRmKu1F6xk8apresUaDuoLZ +550: 5EYCAe5jLQhn6ofDSw1ZQrwuD46xnT77qdge8mYVLJSxBd2D +551: 5EYCAe5jLQhn6ofDSw1q9tVVKYnAF1CzaJd9giEKo8KgTFV2 +552: 5EYCAe5jLQhn6ofDSw26tv35S3TMhZJsJyZfEevAFxCQireQ +553: 5EYCAe5jLQhn6ofDSw2NdwafYY8ZA7Qk3eWAnbbzin58zS6X +554: 5EYCAe5jLQhn6ofDSw2eNy8Ff2okcfWcnKSgLYHqBbwsG4y5 +555: 5EYCAe5jLQhn6ofDSw2v7zfqmXUx5DcVWzPBtUyfeRpbXoLx +556: 5EYCAe5jLQhn6ofDSw3Bs2DRt2A9XmiNFfKhSRfW7FhKoSHv +557: 5EYCAe5jLQhn6ofDSw3Tc3m1zWqLzKpEzLGCzNMLa5a457dP +558: 5EYCAe5jLQhn6ofDSw3jM5Jc71WYSsv7j1CiYK3B2uSnLbvm +559: 5EYCAe5jLQhn6ofDSw4166rCDWBjuS1zTg9E6Fj1VjKWcGKu +560: 5EYCAe5jLQhn6ofDSw4Gq8PnKzrwMz7sCM5jeCQqxZCEsiEp +561: 5EYCAe5jLQhn6ofDSw4Ya9wNSVY8pYDjw22FC96gRP4y9dUg +562: 5EYCAe5jLQhn6ofDSw4pKBUxYzDLH6Kcfgxkk5nWtCwhR9aG +563: 5EYCAe5jLQhn6ofDSw564D2YfUtXjeRVQMuGJ2UMM2pRghUV +564: 5EYCAe5jLQhn6ofDSw5MoEa8myZjCCXN92qmqyABorh9xK4t +565: 5EYCAe5jLQhn6ofDSw5dYG7itUEvekdEshnHPur2GgZtE7i9 +566: 5EYCAe5jLQhn6ofDSw5uHHfJzxv87Jj7cNinwrXrjWScVjpQ +567: 5EYCAe5jLQhn6ofDSw6B2KCu7TbKZrpzM3fJVoDhCLKLmP5E +568: 5EYCAe5jLQhn6ofDSw6SmLkVDxGX2Qvs5ibp3juXfAC52pBg +569: 5EYCAe5jLQhn6ofDSw6iWNJ5LSwiUy2jpPYKbgbN7z4oJPDJ +570: 5EYCAe5jLQhn6ofDSw6zFPqfSwcuwX8cZ4Uq9dHCaowXZxCq +571: 5EYCAe5jLQhn6ofDSw7FzRPFZSJ7Q5EVHjRLhZy33dpFqdiY +572: 5EYCAe5jLQhn6ofDSw7XjSvqfvyJrdLN2QMrFWesWTgz7Wb2 +573: 5EYCAe5jLQhn6ofDSw7oUUURnReWKBSEm5JMoTLhyHZiNqXY +574: 5EYCAe5jLQhn6ofDSw85DW21tvKhmjY7VkEsMQ2YS7SSeWt4 +575: 5EYCAe5jLQhn6ofDSw8LxXZc1QzuEHdzERBNuLiNtwKAv6Pe +576: 5EYCAe5jLQhn6ofDSw8chZ7C7ug6gqjry67tTHQDMmBuC1de +577: 5EYCAe5jLQhn6ofDSw8tSaenEQMJ9Pqjhm4Q1E63pb4dTM7S +578: 5EYCAe5jLQhn6ofDSw9ABcCNLu2VbwwcSRzuZAmtHQwMj1Cb +579: 5EYCAe5jLQhn6ofDSw9RvdjxTPhh4W3VB6wR77TikEp5ze9D +580: 5EYCAe5jLQhn6ofDSw9hffHYZtNtX49Mumsvf49ZD4gpGUGb +581: 5EYCAe5jLQhn6ofDSw9yQgq8gP45ycFEeSpSCzqPftZYY8c6 +582: 5EYCAe5jLQhn6ofDSwAF9iNinsjHSAM7P7kwkwXE8iSGofY4 +583: 5EYCAe5jLQhn6ofDSwAWtjvJuNQUtiSz7nhTJtD4bYK15Mwa +584: 5EYCAe5jLQhn6ofDSwAndmTu1s5gMGYrrTdxrptu4NBjLm3i +585: 5EYCAe5jLQhn6ofDSwB4No1V8Mksopejb8aUQmajXC4TcdTi +586: 5EYCAe5jLQhn6ofDSwBL7pZ5ErS5GNkcKoWyxiGZz1wBt1hh +587: 5EYCAe5jLQhn6ofDSwBbrr6fMM7GivrV4UTVWexQSqov9udn +588: 5EYCAe5jLQhn6ofDSwBsbseFTqnUBUxMo9Q14beEufgeRF5g +589: 5EYCAe5jLQhn6ofDSwC9LuBqaLTfe34EXpLWcYL5NVZNguqD +590: 5EYCAe5jLQhn6ofDSwCR5vjRgq8s6bA7GVH2AV1uqKS6xivf +591: 5EYCAe5jLQhn6ofDSwCgpxH1oKp4Z9Fz1ADXiRhkJ9JqEH3f +592: 5EYCAe5jLQhn6ofDSwCxZypbupVG1hMrjqA3GNPakyBZW2hz +593: 5EYCAe5jLQhn6ofDSwDEK1NC2KATUFTjUW6YpK5RDo4Hmbw9 +594: 5EYCAe5jLQhn6ofDSwDW42un8oqevoZcDB34NFmFgcw23A4Q +595: 5EYCAe5jLQhn6ofDSwDmo4TNFJWrPMfUwqyZvCT69SokJrjA +596: 5EYCAe5jLQhn6ofDSwE3Y5zxMoC3qumMgWv5U98vcGgUaM83 +597: 5EYCAe5jLQhn6ofDSwEKH7YYUHsFJTsERBrb25pm56ZCr2CG +598: 5EYCAe5jLQhn6ofDSwEb2968anYSm1y79ro6a2WbXvRw7i4R +599: 5EYCAe5jLQhn6ofDSwErmAdihHDeDa4ytXjc7yCRzkJfPCoW +600: 5EYCAe5jLQhn6ofDSwF8WCBJomtqg8ArdCg7futGTaBPev5r +601: 5EYCAe5jLQhn6ofDSwFQFDitvGa38gGjMscdDra6vQ47vagE +602: 5EYCAe5jLQhn6ofDSwFfzFGV2mFEbENc6YZ8moFwPDvrCCj6 +603: 5EYCAe5jLQhn6ofDSwFwjGp59FvS3nUUqDVeKjwmr3oaTkqS +604: 5EYCAe5jLQhn6ofDSwGDUJMfFkbdWLaMZtS9sgdcJsgJjJF4 +605: 5EYCAe5jLQhn6ofDSwGVDKuFNFGpxtgEJZNfRdKSmhZ2zvJd +606: 5EYCAe5jLQhn6ofDSwGkxMSqUjx2RSn73EKAya1HEXRmGjcm +607: 5EYCAe5jLQhn6ofDSwH2hNzRbEdDszsymuFgXWh7hMJVY8Fk +608: 5EYCAe5jLQhn6ofDSwHJSQY1hjJRLYyrWaCC5TNxABBDonmo +609: 5EYCAe5jLQhn6ofDSwHaBS5bpDyco75jFF8hdQ4nd13x5VQN +610: 5EYCAe5jLQhn6ofDSwHqvTdBviepFfBbyv5DBLkd5pvgM2q6 +611: 5EYCAe5jLQhn6ofDSwJ7fVAn3DL1iDHUib1ijHSTYeoQcmDP +612: 5EYCAe5jLQhn6ofDSwJPQWiN9i1DAmPMTFxEHE8J1Ug8tTWG +613: 5EYCAe5jLQhn6ofDSwJf9YFxGCgQdKVEBvtjqAp8UJYs9xDb +614: 5EYCAe5jLQhn6ofDSwJvtZoYNhMc5sb6vbqFP7Vxw8RbReGD +615: 5EYCAe5jLQhn6ofDSwKCdbM8VC2oYRgyfGmkw4BoPxJKhPnV +616: 5EYCAe5jLQhn6ofDSwKUNctibghzzynrPwiGUzsdrnB3xs32 +617: 5EYCAe5jLQhn6ofDSwKk7eSJiBPCTXtj8cen2wZUKc3nESFM +618: 5EYCAe5jLQhn6ofDSwL1rfytpg4Pv5zbsHbHatFJnRvWW1je +619: 5EYCAe5jLQhn6ofDSwLHbhXUwAjbNe6UbxXo8pw9FFoEms63 +620: 5EYCAe5jLQhn6ofDSwLZLj553fQnqCCMLdUJgmcyi5fy3Gdi +621: 5EYCAe5jLQhn6ofDSwLq5kcfAA5zHkJE5JQpEiJpAuYhJzsm +622: 5EYCAe5jLQhn6ofDSwM6pnAFGemBkJQ6oyMKnezedjRRaevZ +623: 5EYCAe5jLQhn6ofDSwMNZohqP9SPCrVyYeHqLbgV6ZJ9rGoC +624: 5EYCAe5jLQhn6ofDSwMeJqFRVe7afQbrHKELtYNKZPAt7wnT +625: 5EYCAe5jLQhn6ofDSwMv3ro1c8nn7xhj1zArSV4A2D3cPfMu +626: 5EYCAe5jLQhn6ofDSwNBntLbidTyaWobkf7MzRjzV2vLfF6h +627: 5EYCAe5jLQhn6ofDSwNTXutBq89B34uUVL3sYNRpwro4vqqw +628: 5EYCAe5jLQhn6ofDSwNjGwRmwcpNVd1MDzzP6K7fQgfoCQtG +629: 5EYCAe5jLQhn6ofDSwP11xyN47VZxB7DxfvteFoVsWYXUAVF +630: 5EYCAe5jLQhn6ofDSwPGkzWxAcAmQjD6hLsQCCVLLLRFjZ9g +631: 5EYCAe5jLQhn6ofDSwPYW24YH6qxsHJyS1ouk9BAoAHz1CWW +632: 5EYCAe5jLQhn6ofDSwPpF3c8PbXAKqQrAgkRJ5s1FzAiGwZa +633: 5EYCAe5jLQhn6ofDSwQ5z59iW6CMnPWiuMgvr2Yqip3SYVED +634: 5EYCAe5jLQhn6ofDSwQMj6hJcasZEwcbe2dSPyEgBdvApCdf +635: 5EYCAe5jLQhn6ofDSwQdU8Etj5YkhViUNhZwwuvWeTnu5ev2 +636: 5EYCAe5jLQhn6ofDSwQuD9nUqaDxA3pM7NWTVrcM7HfdMWqo +637: 5EYCAe5jLQhn6ofDSwRAxBL4x4u9cbvDr3Sy3oJBa7YMd9mJ +638: 5EYCAe5jLQhn6ofDSwRShCsf4ZaM5A26aiPUbjz22wR5tne4 +639: 5EYCAe5jLQhn6ofDSwRiSERFB4FYXi7yKPKz9gfrVmHpALrK +640: 5EYCAe5jLQhn6ofDSwRzBFxqHYvjzGDr44GVhdMgxbAYS1GD +641: 5EYCAe5jLQhn6ofDSwSFvHWRQ3bwSpKinjD1Fa3XRR3GhTgF +642: 5EYCAe5jLQhn6ofDSwSXfK41WYH8uNRbXQ9WoWjMtEuzyHBD +643: 5EYCAe5jLQhn6ofDSwSoQLbbd2xLMvXUG562MTRCM4njEg2N +644: 5EYCAe5jLQhn6ofDSwT59N9BjXdXpUdLzk2XuQ72otfTWSny +645: 5EYCAe5jLQhn6ofDSwTLtPgmr2JjH2jDjQy3TLnsGiYBn5Bi +646: 5EYCAe5jLQhn6ofDSwTcdREMxWyvjaq6U5uZ1HUhjYQv3fCb +647: 5EYCAe5jLQhn6ofDSwTtNSmx51f8C8vyCkr4ZEAYCNHeKPyY +648: 5EYCAe5jLQhn6ofDSwUA7UKYBWLKeh2qwRna7ArNfCANb6AU +649: 5EYCAe5jLQhn6ofDSwURrVs8J11X7F8ig6j5f7YD8236rSyk +650: 5EYCAe5jLQhn6ofDSwUhbXQiQVgiZoEbQmfbD4E3aquq87RN +651: 5EYCAe5jLQhn6ofDSwUyLYxJWzMv2MLU9Sc6kzut3fnZPti5 +652: 5EYCAe5jLQhn6ofDSwVF5aVtdV37UuSLt7YcJwbiWVfHfaaG +653: 5EYCAe5jLQhn6ofDSwVWpc3UjyiJwTYDcnV7rtHYyKY1w8Yy +654: 5EYCAe5jLQhn6ofDSwVnZdb4rUPWQ1e6MTRdQpyPS9QkCqGi +655: 5EYCAe5jLQhn6ofDSwW4Jf8exy4hrZjy68N8xmfDtyHUUStT +656: 5EYCAe5jLQhn6ofDSwWL3ggF5TjuK7qqpoJeWiM4MoACjq89 +657: 5EYCAe5jLQhn6ofDSwWbniDqBxR6mfwiZUFA4f2tpd2w1WS2 +658: 5EYCAe5jLQhn6ofDSwWsXjmRJT6JEE3bJ9BfcbijHSufH4xW +659: 5EYCAe5jLQhn6ofDSwX9GmK1QwmVgn9U2p8BAYQZkGnPYjux +660: 5EYCAe5jLQhn6ofDSwXR1nrbXSSh9LFLmV4giV6QD6f7pJFn +661: 5EYCAe5jLQhn6ofDSwXgkpQBdw7tbtMDWA1CGRnEfvXr5vs6 +662: 5EYCAe5jLQhn6ofDSwXxVqwmkRo64ST6EpwhpNU58kQaMqxk +663: 5EYCAe5jLQhn6ofDSwYEEsVMrvUHWzYxyVtDNK9ubaHJdUrm +664: 5EYCAe5jLQhn6ofDSwYVyu2wyR9UyYeqiApivFqk4QA2tzVu +665: 5EYCAe5jLQhn6ofDSwYmivaY5upgS6kiSqmEUCXaXE2mAc6w +666: 5EYCAe5jLQhn6ofDSwZ3Tx88CQVsterbBWhk29DQz3uVS7i8 +667: 5EYCAe5jLQhn6ofDSwZKCyfiJuB5MCxTvBeFa5uFSsnDhgGK +668: 5EYCAe5jLQhn6ofDSwZax1DJRPrGom4Leram82b5uhewyNoP +669: 5EYCAe5jLQhn6ofDSwZrh2ktXtXUGKADPXXGfyGvNXXgEyZ3 +670: 5EYCAe5jLQhn6ofDSwa8S4JUePCfisG68CTnDuxkqMQQWXvt +671: 5EYCAe5jLQhn6ofDSwaQB5r4ksssBRMxrsQHmrebJBH8nAcv +672: 5EYCAe5jLQhn6ofDSwafv7PesNZ4dyTqbYLoKoLRm19s3qJz +673: 5EYCAe5jLQhn6ofDSwawf8wEysEG6XZiLDHJsk2GDq2bKTmw +674: 5EYCAe5jLQhn6ofDSwbDQAUq6MuTZ5fb4tDpRgi6geuKbLxK +675: 5EYCAe5jLQhn6ofDSwbV9C2RCraf1dmToZAKydPw9Un3rxbC +676: 5EYCAe5jLQhn6ofDSwbktDa1KMFrUBsLYE6qXa5mcJen8ckd +677: 5EYCAe5jLQhn6ofDSwc2dF7bRqw3vjyDGu3M5Wmc58XWQBuK +678: 5EYCAe5jLQhn6ofDSwcJNGfBYLcFPJ561ZyrdTTSXxQEfmEu +679: 5EYCAe5jLQhn6ofDSwca7JCmeqHSqrAxkEvNBQ9GznGxwBdp +680: 5EYCAe5jLQhn6ofDSwcqrKkMmKxeJQGqUursjLq7Tc9hCorw +681: 5EYCAe5jLQhn6ofDSwd7bMHwspdqkxNiDaoPHHWwvS2RUdXJ +682: 5EYCAe5jLQhn6ofDSwdPLNqXzKK3DWUaxFjtqECnPFu9kC7N +683: 5EYCAe5jLQhn6ofDSwdf5QP86ozEg4aTgvgQPAtcr5mt1rjf +684: 5EYCAe5jLQhn6ofDSwdvpRviDJfS8cgLRbcuw7aTJuecHTjg +685: 5EYCAe5jLQhn6ofDSweCZTUJKoLdbAnDAGZRV4GHmjXLZCAK +686: 5EYCAe5jLQhn6ofDSweUJV1tSJ1q3it5twVw2zx8EZQ4pj72 +687: 5EYCAe5jLQhn6ofDSwek3WZUYnh2WGyxdcSSawdxhPGo6JxS +688: 5EYCAe5jLQhn6ofDSwf1nY74fHNDxq5qNHNx8tKoAD9XMwtS +689: 5EYCAe5jLQhn6ofDSwfHXZeemn3RRPBi6xKTgq1dd32FdSM8 +690: 5EYCAe5jLQhn6ofDSwfZGbCEtGicswHaqdFyEmhU5rtyu5Z6 +691: 5EYCAe5jLQhn6ofDSwfq1cjpzmPpLVPTaJCUniPJYgmiAiMS +692: 5EYCAe5jLQhn6ofDSwg6keHR7G51o3VLJy8zLf591WeSSbhJ +693: 5EYCAe5jLQhn6ofDSwgNVfq1DkkDFbbD3e5VtbkyULXAhwZq +694: 5EYCAe5jLQhn6ofDSwgeEhNbLFRQi9h5nK21SYSowAPtycbw +695: 5EYCAe5jLQhn6ofDSwguyivBSk6cAhnxWyxWzV8ePzGdFWrE +696: 5EYCAe5jLQhn6ofDSwhBikTmZEmodFtqFeu2YRpUrp9MX6zu +697: 5EYCAe5jLQhn6ofDSwhTTn1MfjT15ozhzKqY6NWKKe25nf5e +698: 5EYCAe5jLQhn6ofDSwhjCoYwnE8CYN6aizn3eKC9nTtp4KzB +699: 5EYCAe5jLQhn6ofDSwhzwq6XtioPzvCTTfiZCFszFHmYL1bG +700: 5EYCAe5jLQhn6ofDSwiGgre81DUbTUJLCLf4kCZpi7eGbazB +701: 5EYCAe5jLQhn6ofDSwiYRtBi7i9nv2QCw1baJ9FfAwWzrzNK +702: 5EYCAe5jLQhn6ofDSwipAujJECpzNaW5fgY5r5wVdmPj8dgE +703: 5EYCAe5jLQhn6ofDSwj5uwGtLhWBq8bxQMUbQ2dL6bGTQRjz +704: 5EYCAe5jLQhn6ofDSwjMexpUTCBPHghq92R6wyKAZR9Bg2zd +705: 5EYCAe5jLQhn6ofDSwjdPzN4ZgrakEohshMcVv112F1uwSaB +706: 5EYCAe5jLQhn6ofDSwju91uegBXnCnuacNJ83rgqV4teDK4S +707: 5EYCAe5jLQhn6ofDSwkAt3TEngCyfM1TM3EdboNfwtmNUvX2 +708: 5EYCAe5jLQhn6ofDSwkSd4zpuAtB7u7L5iB99k4WQie6kRBz +709: 5EYCAe5jLQhn6ofDSwkiN6YR1fZNaTDCpP7ehgkLsYWq238p +710: 5EYCAe5jLQhn6ofDSwkz78618AEa31K5Z44AFdSBLNPZHotQ +711: 5EYCAe5jLQhn6ofDSwmFr9dbEeumVZQxHizfoa81oCGHZWCA +712: 5EYCAe5jLQhn6ofDSwmXbBBBM9axx7Wq2PwBMWorG291q82P +713: 5EYCAe5jLQhn6ofDSwmoLCimTeGAQfchm4sguTVgir1k6n9U +714: 5EYCAe5jLQhn6ofDSwn55EGMa8wMsDiaVjpCTQBXBftUNKJq +715: 5EYCAe5jLQhn6ofDSwnLpFowgdcZKmpTEQki1LsMeVmCdpCj +716: 5EYCAe5jLQhn6ofDSwncZHMXo8HknKvKy5hDZHZC7KdvuXrB +717: 5EYCAe5jLQhn6ofDSwntJJu7ucxxEt2Chkdj7EF2a9WfB5ec +718: 5EYCAe5jLQhn6ofDSwoA3LSi27e9hS85SRaEfAvs2yPPSqde +719: 5EYCAe5jLQhn6ofDSwoRnMzJ8cKM9zDxB6WkD7chVoG7iPFH +720: 5EYCAe5jLQhn6ofDSwohXPXtF6zYcYKpumTFm4JXxd8qyujN +721: 5EYCAe5jLQhn6ofDSwoyGR5UMbfk56RheSPmJzzNRT1aFek4 +722: 5EYCAe5jLQhn6ofDSwpF1Sd4U6LwXeXaP7LGrwgCtGtJXFVq +723: 5EYCAe5jLQhn6ofDSwpWkUAeab28zCdT7nGnQtN3M6m2nvKc +724: 5EYCAe5jLQhn6ofDSwpnVViEh5hLSkjKrTDHxq3sovdm4Nw6 +725: 5EYCAe5jLQhn6ofDSwq4EXFpoaNXuJqCb89oWmjiGkWVLAto +726: 5EYCAe5jLQhn6ofDSwqKyYoQv53jMrw5Ko6K4iRYjaPDbs68 +727: 5EYCAe5jLQhn6ofDSwqbiaM12ZivpR2x4U2pcf7PCQFwsHxu +728: 5EYCAe5jLQhn6ofDSwqsTbtb94Q8Gy8po8yLAboDfE8g8qfu +729: 5EYCAe5jLQhn6ofDSwr9CdSBFZ5KjXEhXouqiYV4841QQmeD +730: 5EYCAe5jLQhn6ofDSwrQweymN3kXC5LaGUrMGVAtast8g9P8 +731: 5EYCAe5jLQhn6ofDSwrgggXMUYRiedST19nrpRrj3hkrwr4P +732: 5EYCAe5jLQhn6ofDSwrxRi4wb36v7BYKjpjNNNYZWXdbDbT2 +733: 5EYCAe5jLQhn6ofDSwsEAjcXhXn7ZjeCUVfsvKEPyMWKUy64 +734: 5EYCAe5jLQhn6ofDSwsVumA7p2TK2Hk5DAcPUFvESBP3kgYh +735: 5EYCAe5jLQhn6ofDSwsmenhhvX8WUqqwwqYu2Cc4u1Fn2KSS +736: 5EYCAe5jLQhn6ofDSwt3PpFJ31ohwPwpgWVQa9HuMq8WHtcH +737: 5EYCAe5jLQhn6ofDSwtK8qnt9WUuPx3hRBRv85yjpf1EZadQ +738: 5EYCAe5jLQhn6ofDSwtassLUG1A6rW9a9rNRg2faHUsxqMFi +739: 5EYCAe5jLQhn6ofDSwtrctt4NVqJK4FStXJwDyMQkJkh6s1F +740: 5EYCAe5jLQhn6ofDSwu8MvReUzWVmcMKdCFSmv3FD8dRNY1G +741: 5EYCAe5jLQhn6ofDSwuQ6wyEbVBhEATCMsBxKrj5fxW9dz3E +742: 5EYCAe5jLQhn6ofDSwufqyWphyrtgiZ56Y8TsoQv8nNsuu3k +743: 5EYCAe5jLQhn6ofDSwuwb14QpUY69GewqD4yRk6kbcFcBLhB +744: 5EYCAe5jLQhn6ofDSwvDL2bzvyDHbpkpZt1Uygnb4S8LT5bh +745: 5EYCAe5jLQhn6ofDSwvV549b3TtV4NrhJYwzXdURXG14ikbp +746: 5EYCAe5jLQhn6ofDSwvkp5hB9xZgWvxa3DtW5aAFz5snz8Jm +747: 5EYCAe5jLQhn6ofDSww2Z7EmGTEsyV4Smtq1dWr6SukXFvLY +748: 5EYCAe5jLQhn6ofDSwwJJ8nMNwv5S3AKWZmXBTXvujdFXSdM +749: 5EYCAe5jLQhn6ofDSwwa3AKwVSbGtbGCFEi2jQDmNZVyo8bB +750: 5EYCAe5jLQhn6ofDSwwqnBsXbwGUM9N4yueYHLubqPNi4em6 +751: 5EYCAe5jLQhn6ofDSwx7XDR7iRwfohTwiab3qHbSJDFSLPWB +752: 5EYCAe5jLQhn6ofDSwxPGExhpvcsGFZpTFXZPEHGm38Ac8CZ +753: 5EYCAe5jLQhn6ofDSwxf1GWHwRJ4iofhBvU4wAy7Drztsd4d +754: 5EYCAe5jLQhn6ofDSwxvkJ3t3uyGBMmZvbQaV7ewggsd9LGE +755: 5EYCAe5jLQhn6ofDSwyCVKbUAQeTdusSfGM634Ln9WkMQkR1 +756: 5EYCAe5jLQhn6ofDSwyUEM94GuKf6TyKPwHbb12ccLd5gYNT +757: 5EYCAe5jLQhn6ofDSwyjyNgePPzrZ25C8cE78wiT5AVowzRJ +758: 5EYCAe5jLQhn6ofDSwz1iQEEVtg41aB4sHAcgtQHXzNYDfiq +759: 5EYCAe5jLQhn6ofDSwzHTRmpcPMFU8Gwbx78Eq67zpFGVKAV +760: 5EYCAe5jLQhn6ofDSwzZCTKQit2SvgNpLd3dnmmxTe7zm2FX +761: 5EYCAe5jLQhn6ofDSwzpwUrzqNhePEUh5Hz9LiTnvTzj2o2S +762: 5EYCAe5jLQhn6ofDSx16gWQawsNqqnaZoxvetf9dPHsTJGVj +763: 5EYCAe5jLQhn6ofDSx1NRXxB4N43JLgSYdsASbqTr7kBZkMV +764: 5EYCAe5jLQhn6ofDSx1eAZVmArjEktnKHJofzYXJJwcuqZNN +765: 5EYCAe5jLQhn6ofDSx1uub3MHMQSDStC1ykBYVD8mmVe72B9 +766: 5EYCAe5jLQhn6ofDSx2BecawPr5dfzz4kegh6RtyEbNNNvFe +767: 5EYCAe5jLQhn6ofDSx2TPe8XWLkq8Z5wVKdCeNaohRF6eYof +768: 5EYCAe5jLQhn6ofDSvqFHR7HuaCvPkQxqZx8iHpeHQ8hVBKK +769: 5EYCAe5jLQhn6ofDSvqX2Set24t7rJWqaEteGEWUkE1RkrkR +770: 5EYCAe5jLQhn6ofDSvqnmUCU8ZZKJrciJuq9pBCKD3tA2LLb +771: 5EYCAe5jLQhn6ofDSvr4WVk4F4EWmQib3amfN7t9fsktJ77C +772: 5EYCAe5jLQhn6ofDSvrLFXHeMYuiDxpTnFiAv4Zz8hdcZbNr +773: 5EYCAe5jLQhn6ofDSvrbzYqEU3augWvLWvegU1FpbXWLqFRa +774: 5EYCAe5jLQhn6ofDSvrsjaNpaYG7952DFbbC1wwf4MP56tjt +775: 5EYCAe5jLQhn6ofDSvs9UbvQh2wJbd85zGXhZtdVXBFoNVtD +776: 5EYCAe5jLQhn6ofDSvsRDdTzoXcW4BDxiwUD7qKKz18Xe12H +777: 5EYCAe5jLQhn6ofDSvsgxf1av2HhWjKqTcQifn1ASq1FucFi +778: 5EYCAe5jLQhn6ofDSvsxhgZB2WxtyHRiCHMEDigzueszBXBe +779: 5EYCAe5jLQhn6ofDSvtESi6m91e6RqXavxHjmfNqNUkiT7mZ +780: 5EYCAe5jLQhn6ofDSvtWBjeMFWKHtPdTfdEFKc4fqJdSidEs +781: 5EYCAe5jLQhn6ofDSvtmvmBwMzzVLwjLQJAksYkWJ8WAzBPh +782: 5EYCAe5jLQhn6ofDSvu3fnjXUVfgoVqD8y7GRVSLkxNuFnUr +783: 5EYCAe5jLQhn6ofDSvuKQpH7azLtG3w5se3myS8BDnFdXVDQ +784: 5EYCAe5jLQhn6ofDSvub9qphhV25ic2xcJzHXNp1gc8MoHRr +785: 5EYCAe5jLQhn6ofDSvurtsNHoyhHBA8qLyvo5KVr9S164gKm +786: 5EYCAe5jLQhn6ofDSvv8dtusvUNUdiEi5esJdGBgcFspLGuj +787: 5EYCAe5jLQhn6ofDSvvQNvTU2y3g6GLapKopBCsX55kYc6uX +788: 5EYCAe5jLQhn6ofDSvvg7x149TisYpSTYzkKj9ZMXudGsZQv +789: 5EYCAe5jLQhn6ofDSvvwryYeFxQ51NYLHfgqH6FBzjW19P2E +790: 5EYCAe5jLQhn6ofDSvwDc16ENT5GTveD2LdLq2w2TZNjQxh2 +791: 5EYCAe5jLQhn6ofDSvwVM2dpUwkTvUk5m1ZrNycrvPFTgNTx +792: 5EYCAe5jLQhn6ofDSvwm64BQbSRfP2qxVgWMvvJhPD8BxAmh +793: 5EYCAe5jLQhn6ofDSvx2q5izhw6rqawqEMSsUrzXr2zvDq9o +794: 5EYCAe5jLQhn6ofDSvxJa7GapRn4J93hy2PP2ogNJrseVQgH +795: 5EYCAe5jLQhn6ofDSvxaK8pAvvTFkh9ahhKtakNCmgkNm9iv +796: 5EYCAe5jLQhn6ofDSvxr4AMm3R8TDFFTSNGQ8h43EWd72aAb +797: 5EYCAe5jLQhn6ofDSvy7oBuM9uoefoMLB3CugdjshLVqJHyr +798: 5EYCAe5jLQhn6ofDSvyPYDSwGQUr8MTCui9REaRiAANZa2Nj +799: 5EYCAe5jLQhn6ofDSvyfHEzXNuA3auZ5eP5vnX7YczFHqVq1 +800: 5EYCAe5jLQhn6ofDSvyw2GY7VPqF3TexP42SLToP5p826zpV +801: 5EYCAe5jLQhn6ofDSvzCmJ5hbtWSW1kq7ixwtQVDYdzkNx8G +802: 5EYCAe5jLQhn6ofDSvzUWKdHiPBdxZrhrPuTSMB41TsUeMmn +803: 5EYCAe5jLQhn6ofDSvzkFMAspsrqR7xab4qxzHrtUHkCuxdQ +804: 5EYCAe5jLQhn6ofDSw11zNiTwNY2sg4TKjnUYEYiw7cwBaW5 +805: 5EYCAe5jLQhn6ofDSw1HjQG43sDELEAL4Qiz6BEZPwVfTFoS +806: 5EYCAe5jLQhn6ofDSw1ZURoeAMtRnnGCo5fVe7vPrmNPiyE5 +807: 5EYCAe5jLQhn6ofDSw1qDTMEGrZdFLN5Xkc1C4cEKbF7zTVD +808: 5EYCAe5jLQhn6ofDSw26xUtpPMEphtTxGRYWk1J4nR7rGFA2 +809: 5EYCAe5jLQhn6ofDSw2NhWSQVqv2ASZq16V2HwyuFEzaXgBa +810: 5EYCAe5jLQhn6ofDSw2eSXyzcLbDczfhjmRXqtfji4sJoHqH +811: 5EYCAe5jLQhn6ofDSw2vBZXaiqGR5YmaUSN3PqMaAtk34tDS +812: 5EYCAe5jLQhn6ofDSw3Bvb5AqKwcY6sTD7JYwn3QdicmLfNX +813: 5EYCAe5jLQhn6ofDSw3TfcckwpcozeyKwnF4VijF6YVVcNh2 +814: 5EYCAe5jLQhn6ofDSw3jQeAM4KJ1TD5CgTBa3fR5ZNNDspJU +815: 5EYCAe5jLQhn6ofDSw419fhwAoyCumB5R885bc6v2CEx9X8a +816: 5EYCAe5jLQhn6ofDSw4GthFXHJeQNKGx9o4b9YnkV27gRHoX +817: 5EYCAe5jLQhn6ofDSw4Ydio7PoKbpsNptU16hVUawqzQgipw +818: 5EYCAe5jLQhn6ofDSw4pNkLhWHzoHRUhd8wcFSARQfs8xMi8 +819: 5EYCAe5jLQhn6ofDSw567mtHcnfzjyaaMot7oNrFsVjsE6KH +820: 5EYCAe5jLQhn6ofDSw5MroRsjHMCCXgT6UpdMKY6LKcbViXv +821: 5EYCAe5jLQhn6ofDSw5dbpyTqn2Pf5nKq9m8uGDvo9VKmA4Z +822: 5EYCAe5jLQhn6ofDSw5uLrX3xGhb7dtCZpheTCumFyN42vWo +823: 5EYCAe5jLQhn6ofDSw6B5t4e4mNnaBz5JVeA19bbioEnJXJY +824: 5EYCAe5jLQhn6ofDSw6SpucEBG3z2k5x3AafZ6HSBd7Wa7SR +825: 5EYCAe5jLQhn6ofDSw6iZw9pHkjBVJBpmqXB72yGeSzEqniD +826: 5EYCAe5jLQhn6ofDSw6zJxhQQFQNwrHhWWTgeyf77Gry7Sno +827: 5EYCAe5jLQhn6ofDSw7G3zEzWk5aQQPaFBQCCvLwa6jhP465 +828: 5EYCAe5jLQhn6ofDSw7Xo1nadEkmrxVSyrLhks2n2vcRefhC +829: 5EYCAe5jLQhn6ofDSw7oY3LAjjRyKWbKiXHDJoicVkV9vBPY +830: 5EYCAe5jLQhn6ofDSw85H4skrE7An4hCTCDirkQSxaMtBsPh +831: 5EYCAe5jLQhn6ofDSw8M26RLxinNEco5BsAEQh6HRQEcTSy1 +832: 5EYCAe5jLQhn6ofDSw8cm7xw5DTZhAtwvY6jxdn7tE7Lj4CV +833: 5EYCAe5jLQhn6ofDSw8tW9WXBi8m9izpfD3FWaTxM3z4ztxu +834: 5EYCAe5jLQhn6ofDSw9AFB47JCoxcH6hPsym4X9nosroGaQ6 +835: 5EYCAe5jLQhn6ofDSw9RzCbhQhVA4qCa8YvGcTqdGhjXY5dN +836: 5EYCAe5jLQhn6ofDSw9hjE9HXCAMXPJSsDrnAQXTjXcFomyC +837: 5EYCAe5jLQhn6ofDSw9yUFgsdgqYywQKbtoHiMDJCMUz5KLc +838: 5EYCAe5jLQhn6ofDSwAFDHETkBWkSVWCLZjoGHu8fBMiLqyN +839: 5EYCAe5jLQhn6ofDSwAWxJn3rgBwu3c55EgJpEay81EScYDv +840: 5EYCAe5jLQhn6ofDSwAnhLKdyAs9MbhwoucpNBGoaq7AtBND +841: 5EYCAe5jLQhn6ofDSwB4SMsE5fYLp9opYaZKv7xe3eyu9pB6 +842: 5EYCAe5jLQhn6ofDSwBLBPQpCADYGhuhHFVqU4eUWUrdRM7v +843: 5EYCAe5jLQhn6ofDSwBbvQxQJetjjG1a1vSM21LJyJjMhBbF +844: 5EYCAe5jLQhn6ofDSwBsfSVzR9ZwBp7SkbNrZx29S8c5xc5e +845: 5EYCAe5jLQhn6ofDSwC9QU3aXeF8eNDKVGKN7thytxUpEQ99 +846: 5EYCAe5jLQhn6ofDSwCR9VbAe8vL6vKCDwFsfqPpMnMYVzgn +847: 5EYCAe5jLQhn6ofDSwCgtX8kkdbXZUR4xcCPDn5epcEGmiXF +848: 5EYCAe5jLQhn6ofDSwCxdYgLs8Gj22WwhH8tmimVHS713CwC +849: 5EYCAe5jLQhn6ofDSwDENaDvycwvUacpRx5QKfTKkFyjJjj9 +850: 5EYCAe5jLQhn6ofDSwDW7bmX67d7w8ihAd1usc9AD5rTacmN +851: 5EYCAe5jLQhn6ofDSwDmrdK7CcJKPgpZuHxRRYpzfujBr41L +852: 5EYCAe5jLQhn6ofDSwE3berhK6yWrEvSdxtvyVWq8jbv7aHQ +853: 5EYCAe5jLQhn6ofDSwEKLgQHRbeiJo2KNdqSXSCfbZUePR9i +854: 5EYCAe5jLQhn6ofDSwEb5hwsY6KumM8C7Jmx5NtW4PMNeshT +855: 5EYCAe5jLQhn6ofDSwErpjVTeb17DuE4qyiTdKaLXDE6vTCK +856: 5EYCAe5jLQhn6ofDSwF8Zm33m5gJgTKwaeeyBGGAz36qCDCc +857: 5EYCAe5jLQhn6ofDSwFQJnadsaMW91RpKKbUjCx1SryZTqHp +858: 5EYCAe5jLQhn6ofDSwFg3p8Dz52hbZXh3zXzH9dqugrHjWf4 +859: 5EYCAe5jLQhn6ofDSwFwnqfp6Zhu47dZnfUVq6KgNWj219Be +860: 5EYCAe5jLQhn6ofDSwGDXsDQD4P6WfjSXLR1P31WqLbkGqnY +861: 5EYCAe5jLQhn6ofDSwGVGtkzKZ4HyDqKG1MWvyhMJAUUYEpB +862: 5EYCAe5jLQhn6ofDSwGm1vJaS3jVRmwBzgJ2UvPBkzMCoxJh +863: 5EYCAe5jLQhn6ofDSwH2kwrAYYQgtL34jMEY2s52DpDw5VNy +864: 5EYCAe5jLQhn6ofDSwHJVyPkf35tLt8wU2B3aokrge6fMCZn +865: 5EYCAe5jLQhn6ofDSwHaEzwLmXm5oSEpCh7Z8kSh9TyPctVW +866: 5EYCAe5jLQhn6ofDSwHqz2Uvt2SHFzLgwN44gh8XcHr7tZat +867: 5EYCAe5jLQhn6ofDSwJ7j42WzX7UiYSZg2zaEdpN57irABgV +868: 5EYCAe5jLQhn6ofDSwJPU5a771ngB6YSQhw5naWCXwbaReyd +869: 5EYCAe5jLQhn6ofDSwJfD77hDWTsdeeK9NsbLXC2zmUJhHzL +870: 5EYCAe5jLQhn6ofDSwJvx8fHL1956CkBt3p6tTssTbM2xpSz +871: 5EYCAe5jLQhn6ofDSwKChACsSVpGYkr4cikcSQZhvRDmEWT2 +872: 5EYCAe5jLQhn6ofDSwKUSBkTYzVU1JwwMPh7zMFYPF6VWKwG +873: 5EYCAe5jLQhn6ofDSwKkBDJ3fVAfTs3p64ddYHwNr4yDmodk +874: 5EYCAe5jLQhn6ofDSwL1vEqdmyqrvR9gpja96EdDJtqx3NjK +875: 5EYCAe5jLQhn6ofDSwLHfGPDtUX4NyFZZQWeeBK3miigK7b2 +876: 5EYCAe5jLQhn6ofDSwLZQHvozyCFqXMSJ5TAC7ztEYbQaawr +877: 5EYCAe5jLQhn6ofDSwLq9KUQ7TsTJ5TK2kPfk4gihNU8rKKM +878: 5EYCAe5jLQhn6ofDSwM6tM1zDxYekdZBmRLBJ1NZACLs7za8 +879: 5EYCAe5jLQhn6ofDSwMNdNZaLTDrDBf4W6Ggqx4Pd2DbPm1Z +880: 5EYCAe5jLQhn6ofDSwMeNQ7ASwu3fjkwEmDCPtkE5r6KfCqS +881: 5EYCAe5jLQhn6ofDSwMv7RekZSaF8HroyS9hwqS4Yfy3vs1i +882: 5EYCAe5jLQhn6ofDSwNBrTCLfwFSaqxgi76DVn7u1VqnCKdZ +883: 5EYCAe5jLQhn6ofDSwNTbUjvnRve3Q4ZSn2j3iojUKiWTxuR +884: 5EYCAe5jLQhn6ofDSwNjLWHWtvbqVxASBSyEbfVZw9bEjuGF +885: 5EYCAe5jLQhn6ofDSwP15Xq71RH2xWGJv7uk9cBQPyTy1KK2 +886: 5EYCAe5jLQhn6ofDSwPGpZNh7uxER4NBenrFhYsEroLhGv4U +887: 5EYCAe5jLQhn6ofDSwPYZavHEQdRscU4PTnmFVZ5KdDRYbDs +888: 5EYCAe5jLQhn6ofDSwPpJcTsLuJdLAZw88jGoSEunT69pCcM +889: 5EYCAe5jLQhn6ofDSwQ63e1TTPypniforofnMNvkFGxt5i9U +890: 5EYCAe5jLQhn6ofDSwQMnfZ3Ztf2FGmgbUcHuKcai6qcMaYL +891: 5EYCAe5jLQhn6ofDSwQdXh6dgPLDhpsZL9YoTGJRAviLdEmZ +892: 5EYCAe5jLQhn6ofDSwQuGieDnt1RANyS4pVK1CzFdkb4tgYN +893: 5EYCAe5jLQhn6ofDSwRB1kBouNgccw5JoVRpZ9g66aToALaq +894: 5EYCAe5jLQhn6ofDSwRSkmjQ1sMp5VBBYANL76MvZQLXS4Vu +895: 5EYCAe5jLQhn6ofDSwRiVoGz8N31Y3H4GqJqf33m2EDFhj5S +896: 5EYCAe5jLQhn6ofDSwRzEppaEriCzbNw1WFMCyjbV45yy81m +897: 5EYCAe5jLQhn6ofDSwSFyrNAMMPQT9UokBBrkvRRwsxiErwp +898: 5EYCAe5jLQhn6ofDSwSXisukTr4buhagUr8NJs7GQhqSWRPe +899: 5EYCAe5jLQhn6ofDSwSoTuTLaLjoNFgZDX4sroo6sXiAmzRT +900: 5EYCAe5jLQhn6ofDSwT5CvzvgqQzponRxC1PQkUwLMau3ePs +901: 5EYCAe5jLQhn6ofDSwTLwxYWoL6CHMtJgrwtxhAmoBTdKXyR +902: 5EYCAe5jLQhn6ofDSwTcgz66upmPjuzBRXtQWdrcG1LMazy6 +903: 5EYCAe5jLQhn6ofDSwTtS1dh2KSbCU64ACpv4aYSiqD5rdpF +904: 5EYCAe5jLQhn6ofDSwUAB3BH8p7nf2BvtsmRcXEHBf5p874L +905: 5EYCAe5jLQhn6ofDSwURv4isFJnz7aHodYhwATv7eUxYPt9r +906: 5EYCAe5jLQhn6ofDSwUhf6GTMoUBa8PgNDeSiQbx7JqGfSEd +907: 5EYCAe5jLQhn6ofDSwUyQ7p3UJ9P2gVZ6taxGMHna8hzw5fY +908: 5EYCAe5jLQhn6ofDSwVF99MdanpaVEbRqZXTpHyd2xajCotp +909: 5EYCAe5jLQhn6ofDSwVWtAuDhHVmwnhJaETyNEfTVnTTUJ62 +910: 5EYCAe5jLQhn6ofDSwVndCSoonAyQLoBJuQUvBMHxcLBjrJi +911: 5EYCAe5jLQhn6ofDSwW4NDzPvGrArtu43aLzU838RSCv1ehN +912: 5EYCAe5jLQhn6ofDSwWL7FXz2mXNKSzvnFHW24ixtG5eHAkV +913: 5EYCAe5jLQhn6ofDSwWbrH5a9GCZn16oWvE1a1QoM5xNYqL8 +914: 5EYCAe5jLQhn6ofDSwWsbJdAFksmEZCgFbAX7x6douq6pSNe +915: 5EYCAe5jLQhn6ofDSwX9LLAkNFYxh7JYzG72ftnUGjhq6ByT +916: 5EYCAe5jLQhn6ofDSwXR5MiLUkEA9fQRiw3YDqUJjZaZMp43 +917: 5EYCAe5jLQhn6ofDSwXgpPFvbEuMcDWJTbz3mnA9CPTHdVsC +918: 5EYCAe5jLQhn6ofDSwXxZQoWhjaZ4mcBCGvZKiqyfDL1u7hx +919: 5EYCAe5jLQhn6ofDSwYEJSM6pEFkXKi3vws4sfXp83CkAivU +920: 5EYCAe5jLQhn6ofDSwYW3TtgvivwysovfcoaRcDeas5USMYT +921: 5EYCAe5jLQhn6ofDSwYmnVSH3Dc9SRuoQHk5yYuV3gxChyyF +922: 5EYCAe5jLQhn6ofDSwZ3XWys9iHLtz1g8xgbXVbKWWpvydok +923: 5EYCAe5jLQhn6ofDSwZKGYXTGCxYMY7Ysdd75SH9yLhfFKGv +924: 5EYCAe5jLQhn6ofDSwZb1a53Nhdjp6DRcJZcdNxzSAaPWrbz +925: 5EYCAe5jLQhn6ofDSwZrkbcdVCJwGeKJLyW8BKeptzT7nFc5 +926: 5EYCAe5jLQhn6ofDSwa8VdADbgz8jCRB5eSdjGLfMpKr3te4 +927: 5EYCAe5jLQhn6ofDSwaQEehoiBfLBkX3pKP9HD2VpeCaKhrx +928: 5EYCAe5jLQhn6ofDSwafygFPpgLXeJcvYzKeq9iLHU5JbD23 +929: 5EYCAe5jLQhn6ofDSwawihnywB1j6rioHfGAP6QAkHx2rjpq +930: 5EYCAe5jLQhn6ofDSwbDTjLa3fgvZQpg2LCfw361D7pm8UxM +931: 5EYCAe5jLQhn6ofDSwbVCktAAAN81xvYm19BUymqfwhVQ2QH +932: 5EYCAe5jLQhn6ofDSwbkwnRkGf3KUX2RVg5h2vTg8maDfkCV +933: 5EYCAe5jLQhn6ofDSwc2goyLP9iWw58JEM2Cas9WbbSwwF7u +934: 5EYCAe5jLQhn6ofDSwcJRqWvVePiPdEAy1xi8oqM4RKgCyrk +935: 5EYCAe5jLQhn6ofDSwcaAs4Wc94urBL3hguDgkXBXFCQUa1R +936: 5EYCAe5jLQhn6ofDSwcqutc6idk7JjRvSMqjEhD1z558kNSY +937: 5EYCAe5jLQhn6ofDSwd7ev9gq8RJmHXoB2nEndtrStws24Jk +938: 5EYCAe5jLQhn6ofDSwdPPwhGwd6WDqdfuhikLaaguipbHhTc +939: 5EYCAe5jLQhn6ofDSwdf8yEs47mhgPjYeNfFtXGXNYhKZGZk +940: 5EYCAe5jLQhn6ofDSwdvsznTAcSu8wqRP3bmSTxMqNa3puJ5 +941: 5EYCAe5jLQhn6ofDSweCd2L3H786bVwJ7iYGzQeCJCSn6H8L +942: 5EYCAe5jLQhn6ofDSweUN3sdPboJ443ArPUnYML2m2KWNBKc +943: 5EYCAe5jLQhn6ofDSwek75RDW6UVWc93b4RJ6J1sDrCEdfCm +944: 5EYCAe5jLQhn6ofDSwf1r6xocb9gyAEvKjMoeEhhgg4xuJbG +945: 5EYCAe5jLQhn6ofDSwfHb8WPj5ptRiLo4QJKCBPY9VwhAwVR +946: 5EYCAe5jLQhn6ofDSwfZLA3yqaW5tGSfo5Epk85NcKpRSg1o +947: 5EYCAe5jLQhn6ofDSwfq5BbZx5BHLpYYXkBLJ4mD59h9iEEa +948: 5EYCAe5jLQhn6ofDSwg6pD9A4ZrUoNeRGR7qr1T3XyZsyiw8 +949: 5EYCAe5jLQhn6ofDSwgNZEgkB4XgFvkJ164MPx8szoScFak2 +950: 5EYCAe5jLQhn6ofDSwgeJGELHZCsiUrAjkzrwtpiTdKLXB4X +951: 5EYCAe5jLQhn6ofDSwgv3HmvQ3t5B2x3URwNVqWYvTC4naER +952: 5EYCAe5jLQhn6ofDSwhBnKKWWYZGdb3vD6st3nCPPH4o4Mat +953: 5EYCAe5jLQhn6ofDSwhTXLs6d3EU699nwmpPbitDr6wXL32D +954: 5EYCAe5jLQhn6ofDSwhjGNQgjXufYhFfgSku9fa4JvpFbgt4 +955: 5EYCAe5jLQhn6ofDSwi11PxGr2as1FMYR7hQhcFtmkgysGkc +956: 5EYCAe5jLQhn6ofDSwiGkRVrxXG4ToTR9ndvFYwjEaZi8ngw +957: 5EYCAe5jLQhn6ofDSwiYVT3T51wFvMZHtTaRoVdZhQSSQP4g +958: 5EYCAe5jLQhn6ofDSwipEUb3BWcTNufAd8WwMSKQAEKAg1NA +959: 5EYCAe5jLQhn6ofDSwj5yW8dJ1HeqTm3MoTSuP1Ed4BtwabB +960: 5EYCAe5jLQhn6ofDSwjMiXgDQVxrJ1rv6UPxTKh55t4dDCHY +961: 5EYCAe5jLQhn6ofDSwjdTZDoWze3kZxnq9LU1GNuYhwMUvSK +962: 5EYCAe5jLQhn6ofDSwjuCamPdVKFD84fZpGyZD4k1Xp5kfJk +963: 5EYCAe5jLQhn6ofDSwkAwcJyjyzSfgAYJVDV79kaUMgp2EPZ +964: 5EYCAe5jLQhn6ofDSwkSgdrZrUfe8EGR3A9zf6SQwBZYHmgP +965: 5EYCAe5jLQhn6ofDSwkiRfQ9xyLqanNHmq6WD38FQ1SGZTTQ +966: 5EYCAe5jLQhn6ofDSwkzAgwk5U233LUAWW31kyp5rqJzqBXR +967: 5EYCAe5jLQhn6ofDSwmFuiVLBxhEVta3FAyXJvVvKfBj6fFs +968: 5EYCAe5jLQhn6ofDSwmXek2vJTNRxSfuyqv2rsBknV4TNUaZ +969: 5EYCAe5jLQhn6ofDSwmoPmaWQx3dQzmniWrYQosbFJwBdpHQ +970: 5EYCAe5jLQhn6ofDSwn58o86XSipsYsfTBo3xkZRi8ouujZb +971: 5EYCAe5jLQhn6ofDSwnLspfgdwQ2L6yYBrjZWhFGAxgeBF8E +972: 5EYCAe5jLQhn6ofDSwnccrDGkS5Dnf5QvXg54dw6dnZNSred +973: 5EYCAe5jLQhn6ofDSwntMskrrvkRFDBHfCcacacw6cS6iMUZ +974: 5EYCAe5jLQhn6ofDSwoA6uJSyRRchmHAPsZ6AXJmZSJpzCSq +975: 5EYCAe5jLQhn6ofDSwoRqvr35v6pAKP38YVbiTzc2GBZFfwh +976: 5EYCAe5jLQhn6ofDSwohaxPdCQn1csUusDS7GQgSV64HXCQ1 +977: 5EYCAe5jLQhn6ofDSwoyKywDJuTD5RanbtNcpMNGwuw1npdK +978: 5EYCAe5jLQhn6ofDSwpF51UoRQ8QXygfLZK8NJ47Qjok4jZT +979: 5EYCAe5jLQhn6ofDSwpWp32PXtobzXnY5EFdvEjwsZgULCga +980: 5EYCAe5jLQhn6ofDSwpnZ4ZyePUoT5tQouC9UBRnLPZCbmdk +981: 5EYCAe5jLQhn6ofDSwq4J67Zkt9zudzHYa8f287coDRvsRGc +982: 5EYCAe5jLQhn6ofDSwqL37f9sNqCNC6AHF5Aa4oTG3Jf8xpw +983: 5EYCAe5jLQhn6ofDSwqbn9CjysWPpkC31v1g81VHisBPQptU +984: 5EYCAe5jLQhn6ofDSwqsXAkL6NBbHJHukaxBfxB8Bh47gAbt +985: 5EYCAe5jLQhn6ofDSwr9GCHvCrrnjrPnVFthDtrxeWvqx2sP +986: 5EYCAe5jLQhn6ofDSwrR1DqWKMXzCQVfDvqCmqYo7LoaDX1B +987: 5EYCAe5jLQhn6ofDSwrgkFP6RrDBexbXxbmiKnEdaAgJVGJY +988: 5EYCAe5jLQhn6ofDSwrxVGvgYLtP7WhQhGiDsivU2zZ2kvZ6 +989: 5EYCAe5jLQhn6ofDSwsEEJUGeqZaa4oHRwejRfcJVpRm2TYT +990: 5EYCAe5jLQhn6ofDSwsVyL1rmLEn2cuAAcbEycJ8xeJVJAsG +991: 5EYCAe5jLQhn6ofDSwsmiMZSspuyVB12uHXkXYyyRUBDZmPU +992: 5EYCAe5jLQhn6ofDSwt3TP72zKbAwj6udxUG5VfotJ3wqCyJ +993: 5EYCAe5jLQhn6ofDSwtKCQed6pGNQHCnNdQmdSMeM7vg76HA +994: 5EYCAe5jLQhn6ofDSwtawSCDDJwZrqJf7JMHBP3UowoQNX1a +995: 5EYCAe5jLQhn6ofDSwtrgTjoKocmKPQXqyHnjKjKGmg8eN7M +996: 5EYCAe5jLQhn6ofDSwu8RVHPSJHxmwWQaeEJHGR9jbYruht5 +997: 5EYCAe5jLQhn6ofDSwuQAWpyYnyAEVcHKKAoqD6zCRRbBXTx +998: 5EYCAe5jLQhn6ofDSwufuYNZfHeMh3iA3z7KP9npfFJKSy6v +999: 5EYCAe5jLQhn6ofDSwuweZv9mnKZ9bp2nf3pw6Uf85B3ia6x +1000: 5EYCAe5jLQhn6ofDSwvDPbTjtGzkc9uuXKzLV3AVau3mzD3F +1001: 5EYCAe5jLQhn6ofDSwvV8d1Kzmfx4i1nFzvr2yrL3ivWG1Wg +1002: 5EYCAe5jLQhn6ofDSwvkseYv7GM9XG7ezfsMavYAWYoEXUrg +1003: 5EYCAe5jLQhn6ofDSww2cg6WDm2LypDXjLos8sDzyNfxoEqD +1004: 5EYCAe5jLQhn6ofDSwwJMhe6LFhYSNKQU1kNgouqSCYh4mRE +1005: 5EYCAe5jLQhn6ofDSwwa6jBgSkNjtvRHCggtEkbfu2RRLWcA +1006: 5EYCAe5jLQhn6ofDSwwqqkjGZF3wMUX9wMdPnhHWMrJ9bwiU +1007: 5EYCAe5jLQhn6ofDSwx7anGrfjj8p2d2g2ZuLdyLpgAssiC4 +1008: 5EYCAe5jLQhn6ofDSwxPKopSnEQLGaiuQhWQtafBHW3c9VSf +1009: 5EYCAe5jLQhn6ofDSwxf4qN2tj5Xj8pn9NSvSXM1kKvLQtJU +1010: 5EYCAe5jLQhn6ofDSwxvorud1DkjBgvet3PRzU2rD9o4gWNR +1011: 5EYCAe5jLQhn6ofDSwyCYtTD7iRveF2XciKwYQigfyfnxDDr +1012: 5EYCAe5jLQhn6ofDSwyUHuzoED786o8QMPGT6MQX8oYXDznu +1013: 5EYCAe5jLQhn6ofDSwyk2wYPLhnKZMEH64CxeJ6MbdRFVYiN +1014: 5EYCAe5jLQhn6ofDSwz1my5yTCTX1uL9pj9UCEnC4THymGqt +1015: 5EYCAe5jLQhn6ofDSwzHWzdZZh8iUTS2ZQ5ykBU2XHAi2c5d +1016: 5EYCAe5jLQhn6ofDSwzZG2B9gBouw1XuJ52VJ89rz73SJLQt +1017: 5EYCAe5jLQhn6ofDSwzq13ijngV7PZdn2jxzr4qhSvvAZqXq +1018: 5EYCAe5jLQhn6ofDSx16k5GKuBAJr7jemQuWQ1XXukntqcmK +1019: 5EYCAe5jLQhn6ofDSx1NV6ov1fqWJfqXW5r1wxDNNafd76hY +1020: 5EYCAe5jLQhn6ofDSx1eE8MW8AWhmDwQEknXVtuCqQYMNwWU +1021: 5EYCAe5jLQhn6ofDSx1uy9u6EfBuDn3GyRj33qb3JER5eVd4 +1022: 5EYCAe5jLQhn6ofDSx2BiBSgM9s6gL99i6fYbnGsm4HovEUC +1023: 5EYCAe5jLQhn6ofDSx2TTCzGTeYJ8tF2Smc49ixiDtAYBtUX +1024: 5EYCAe5jLQhn6ofDSvqFLyy2rszPQ5a3o1vzDeCYos492Xug \ No newline at end of file From 4fcb78405291780fa83ee00837a6a1e6b5000303 Mon Sep 17 00:00:00 2001 From: Greg Zaitsev Date: Wed, 25 Mar 2026 14:50:58 -0400 Subject: [PATCH 4/4] Organize TAO critical operations --- contract-tests/src/subtensor.ts | 137 +++++++++++++++ pallets/subtensor/src/coinbase/block_step.rs | 2 +- pallets/subtensor/src/coinbase/mod.rs | 1 + pallets/subtensor/src/coinbase/tao.rs | 159 ++++++++++++++++++ .../migrate_subnet_tao_to_subnet_balance.rs | 4 + pallets/subtensor/src/staking/add_stake.rs | 2 - pallets/subtensor/src/staking/helpers.rs | 93 ---------- pallets/subtensor/src/staking/move_stake.rs | 4 +- 8 files changed, 304 insertions(+), 98 deletions(-) create mode 100644 pallets/subtensor/src/coinbase/tao.rs create mode 100644 pallets/subtensor/src/migrations/migrate_subnet_tao_to_subnet_balance.rs diff --git a/contract-tests/src/subtensor.ts b/contract-tests/src/subtensor.ts index f5829c76aa..af9b88be80 100644 --- a/contract-tests/src/subtensor.ts +++ b/contract-tests/src/subtensor.ts @@ -419,4 +419,141 @@ export async function setNetworkLastLockCost(api: TypedApi, defau const valueOnChain = await api.query.SubtensorModule.NetworkLastLockCost.getValue() assert.equal(defaultNetworkLastLockCost, valueOnChain) +} + +export function getSubnetAccountId(netuid: number): string { + // Hardcode to speed up tests + const NETUID_TO_ACCOUNT_ID: Record = { + 0: "5EYCAe5jLQhn6ofDSvqF6iY53erXNkwhyE1aCEgvi1NNs91F", + 1: "5EYCAe5jLQhn6ofDSvqWqk5fA9XiqK3ahtx5kBNmAqF78mqL", + 2: "5EYCAe5jLQhn6ofDSvqnamdFGeCvHs9TSZtbJ84bdf7qQRc6", + 3: "5EYCAe5jLQhn6ofDSvr4KoAqP8t7kRFLBEq6r4kS6UzZgCb5", + 4: "5EYCAe5jLQhn6ofDSvrL4piRVdZKCyMCuumcQ1SGZJsHwmeE", + 5: "5EYCAe5jLQhn6ofDSvrborG1c8EWfXT5eai7wx8728k2DHK7", + 6: "5EYCAe5jLQhn6ofDSvrsYsobicui85YxPFedVtowUxckUuF8", + 7: "5EYCAe5jLQhn6ofDSvs9HuMBq7auadeq7vb93qVmwnVUkg5A", + 8: "5EYCAe5jLQhn6ofDSvsR2vtmwcG73BkhrbXebnBcQcND2Bdh", + 9: "5EYCAe5jLQhn6ofDSvsgmxSN46wJVjrabGUA9isSsSEwHnFy", + 10: "5EYCAe5jLQhn6ofDSvsxWyyxAbcVxHxTKwQfhfZHLG7fZUJG", + 11: "5EYCAe5jLQhn6ofDSvtEG1XYH6HhQr4L4cMBFcF7o5zPpyA3", + 12: "5EYCAe5jLQhn6ofDSvtW1358PaxtsQACoHHgoYvxFus86kJK", + 13: "5EYCAe5jLQhn6ofDSvtmk4ciW5e6KxG5XxECMVcnijjrN8rz", + 14: "5EYCAe5jLQhn6ofDSvu3V6AJcaKHnWMxGdAhuSJdBZcadwDn", + 15: "5EYCAe5jLQhn6ofDSvuKE7htj4zVF4Tq1J7DTNzTePVJucfX", + 16: "5EYCAe5jLQhn6ofDSvuay9FUqZfghcZhjy3j1KgJ7DN3BDc2", + 17: "5EYCAe5jLQhn6ofDSvuriAo4x4LtAAfaUdzEZGN8a3EmSncG", + 18: "5EYCAe5jLQhn6ofDSvv8TCLf4Z25cimTDJvk7D3y2s7ViZEm", + 19: "5EYCAe5jLQhn6ofDSvvQCDtFB3hH5GsKwysFf9joVgzDytnb", + 20: "5EYCAe5jLQhn6ofDSvvfwFRqHYNUXpyCgeomD6RdxWrxFpQR", + 21: "5EYCAe5jLQhn6ofDSvvwgGyRQ33fzP55RKkGm37URLjgXG7M", + 22: "5EYCAe5jLQhn6ofDSvwDRJX1WXisSwAx9zgnJyoJtAcQo59Y", + 23: "5EYCAe5jLQhn6ofDSvwVAL4bd2Q4uVGptfdHrvV9LzV94VBb", + 24: "5EYCAe5jLQhn6ofDSvwkuMcBjX5GN3NhdLZoQsAyopMsL7A7", + 25: "5EYCAe5jLQhn6ofDSvx2eP9mr1kTpbUaN1WJxorpGeEbbfgG", + 26: "5EYCAe5jLQhn6ofDSvxJPQhMxWRfH9aT6gSpWkYejU7KsbGp", + 27: "5EYCAe5jLQhn6ofDSvxa8SEx516rjhgKqMPL4hEVCHz49DPw", + 28: "5EYCAe5jLQhn6ofDSvxqsTnYBVn4CFnCa2KqcdvKf7rnQo7f", + 29: "5EYCAe5jLQhn6ofDSvy7cVL8HzTFeot5JhGMAacA7wjWgPix", + 30: "5EYCAe5jLQhn6ofDSvyPMWsiQV8T7Myx3NCriXHzamcEwyqa", + 31: "5EYCAe5jLQhn6ofDSvyf6YRJWyoeZv5pn39NGTyq3bUyDc8k", + 32: "5EYCAe5jLQhn6ofDSvyvqZxtdUUr2UBhWi5spQffWRMhV5hU", + 33: "5EYCAe5jLQhn6ofDSvzCabWUjyA3V2HaFP2PNMMVyFERkxPm", + 34: "5EYCAe5jLQhn6ofDSvzUKd44rTqEwaPSz3xtvJ3LS57A2Td3", + 35: "5EYCAe5jLQhn6ofDSvzk4ebexxWSQ8VKiiuQUEjAttytJ8Nx", + 36: "5EYCAe5jLQhn6ofDSw11og9F5TBdrgbCTPqv2BR1MircZp68", + 37: "5EYCAe5jLQhn6ofDSw1HYhgqBwrqKEh5C4nRa86qpYjLqCQd", + 38: "5EYCAe5jLQhn6ofDSw1ZHjERJSY2mnnwvjiw84ngHNc56t9n", + 39: "5EYCAe5jLQhn6ofDSw1q2kn1QwDEELtpfQfSg1UWkCUoNVFh", + 40: "5EYCAe5jLQhn6ofDSw26mnKbXRtRgtzhQ5bxDxAMD2MXeL6A", + 41: "5EYCAe5jLQhn6ofDSw2NWosBdvZd9T6a8kYTmtrBfrEFusmX", + 42: "5EYCAe5jLQhn6ofDSw2eFqQmkREpc1CSsRUyKqY28g6zBbxD", + 43: "5EYCAe5jLQhn6ofDSw2uzrxMruv24ZJKc6RUsnDrbVyiT4uZ", + 44: "5EYCAe5jLQhn6ofDSw3BjtVwyQbDX7QCLmMzRiuh4KrSienC", + 45: "5EYCAe5jLQhn6ofDSw3TUv3Y5uGQyfW55SJVyfbXX9jAzHBc", + 46: "5EYCAe5jLQhn6ofDSw3jDwb8CPwcSDbwp7F1XcHMyybuFsV6", + 47: "5EYCAe5jLQhn6ofDSw3zxy8iJtcotmhpYnBX5YyCSoUdXS9C", + 48: "5EYCAe5jLQhn6ofDSw4GhzgJRPJ1MKohHT82dVf2udMMo854", + 49: "5EYCAe5jLQhn6ofDSw4YT2DtXsyCosua284YBSLsNTE64sjn", + 50: "5EYCAe5jLQhn6ofDSw4pC3mUeNeQGS1Sko13jP2hqH6pLJc1", + 51: "5EYCAe5jLQhn6ofDSw55w5K4ksKbiz7KVTwZHKiYJ6yYc62p", + 52: "5EYCAe5jLQhn6ofDSw5Mg6resMzoBYDCE8t4qGQNkvrGsYLi", + 53: "5EYCAe5jLQhn6ofDSw5dR8QEyrfze6K4xopaPD6DDkj19BcH", + 54: "5EYCAe5jLQhn6ofDSw5uA9wq6MMC6eQwhUm5w9n3gabjR243", + 55: "5EYCAe5jLQhn6ofDSw6AuBVRCr2PZCWpS9hbV6Tt9QUTghER", + 56: "5EYCAe5jLQhn6ofDSw6SeD31KLhb1kchApe7339icEMBxKr7", + 57: "5EYCAe5jLQhn6ofDSw6iPEabRqNnUJiZuVacayqZ54DvDhGB", + 58: "5EYCAe5jLQhn6ofDSw6z8G8BYL3yvrpSeAX88vXPXt6eVRoY", + 59: "5EYCAe5jLQhn6ofDSw7FsHfmepjBPQvKNqTdgsDDzhyNky3e", + 60: "5EYCAe5jLQhn6ofDSw7XcKDMmKQNqy2C7WQ9Eou4TXr72f1B", + 61: "5EYCAe5jLQhn6ofDSw7oMLkwsp5aJX84rBLenkatvMiqJC2W", + 62: "5EYCAe5jLQhn6ofDSw856NJXzJkmm5DwarHALhGjPBbZa5wW", + 63: "5EYCAe5jLQhn6ofDSw8LqPr86oRyDdKpKXDftdxZr1UHqWzq", + 64: "5EYCAe5jLQhn6ofDSw8caRPiDJ7AgBRh4CABSaeQJqM278gq", + 65: "5EYCAe5jLQhn6ofDSw8tKSwJKnnN8jXZns6gzXLEmfDkNtXu", + 66: "5EYCAe5jLQhn6ofDSw9A4UUtSHTZbHdSXY3CYU25EV6UeLH2", + 67: "5EYCAe5jLQhn6ofDSw9RoW2UYn8m3qjKGCyi6QhuhJyCv9nu", + 68: "5EYCAe5jLQhn6ofDSw9hYXa4fGoxWPqBzsvDeMPkA8qwBecQ", + 69: "5EYCAe5jLQhn6ofDSw9yHZ7emmV9xww4jYrjCJ5acxifTH7b", + 70: "5EYCAe5jLQhn6ofDSwAF2afEtGAMRW2wUDoEkEmR5nbPiuFf", + 71: "5EYCAe5jLQhn6ofDSwAWmcCpzkqYt48pCtjkJBTFYcU7ziWG", + 72: "5EYCAe5jLQhn6ofDSwAnWdkR7FWkLcEgwZgFr8961SLrGPJp", + 73: "5EYCAe5jLQhn6ofDSwB4FfJ1DkBwoALZgEcmQ4pvUGDaXxGw", + 74: "5EYCAe5jLQhn6ofDSwBKzgqbLEs9FiSSQuZGx1Wkw66JoNQY", + 75: "5EYCAe5jLQhn6ofDSwBbjiPBSjYLiGYK9aVnVxCbPuy357eQ", + 76: "5EYCAe5jLQhn6ofDSwBsUjvmZEDYApeBtFSJ3ttRrjqmLmRP", + 77: "5EYCAe5jLQhn6ofDSwC9DmUMfitjdNk4cvNobqaGKZiVcSd4", + 78: "5EYCAe5jLQhn6ofDSwCQxo1wnDZw5vqwMbKK9nG6nPbDsr3v", + 79: "5EYCAe5jLQhn6ofDSwCghpZXtiF8YUwp6GFphiwwFDTx9ZXw", + 80: "5EYCAe5jLQhn6ofDSwCxSr781CvL133gpwCLFfdmi3LgRGUs", + 81: "5EYCAe5jLQhn6ofDSwDEBsei7hbXTb9ZZc8qocKcAsDQgmDH", + 82: "5EYCAe5jLQhn6ofDSwDVvuCJECGiv9FSJH5MMZ1Sdh68xe6G", + 83: "5EYCAe5jLQhn6ofDSwDmfvjtLgwvNhMK2x1ruVhH6WxsE2Rh", + 84: "5EYCAe5jLQhn6ofDSwE3QxHUTBd7qFTBmcxNTSP7ZLqbVqHX", + 85: "5EYCAe5jLQhn6ofDSwEK9yq4ZgJKHoZ4WHtt1P4x2AiKmP2V", + 86: "5EYCAe5jLQhn6ofDSwEau1NegAyWkMewExqPZKknUzb42r36", + 87: "5EYCAe5jLQhn6ofDSwEre2vEnfeiCukoydmu7GScwpTnJa5d", + 88: "5EYCAe5jLQhn6ofDSwF8P4TpuAKufTrgiJiQfD8TQeLWaGop", + 89: "5EYCAe5jLQhn6ofDSwFQ861R1f1781xZSyevD9pHsUDEqiBR", + 90: "5EYCAe5jLQhn6ofDSwFfs7Z189gJaa4SBebRm6W8LJ5y7dfH", + 91: "5EYCAe5jLQhn6ofDSwFwc96bEeMW38AJvKXwK3Bxo7xhP3yn", + 92: "5EYCAe5jLQhn6ofDSwGDMAeBM92hVgGBezUSrysoFwqReqrS", + 93: "5EYCAe5jLQhn6ofDSwGV6CBmTdhtxEN4PfQxQvZdimi9vW9r", + 94: "5EYCAe5jLQhn6ofDSwGkqDjMa8P6QnTw8LMTxsFUBbatC8C5", + 95: "5EYCAe5jLQhn6ofDSwH2aFGwgd4HsLZos1HyWowJeRTcTVsg", + 96: "5EYCAe5jLQhn6ofDSwHJKGpXo7jVKtfgbgEV4kd97FLLjBeJ", + 97: "5EYCAe5jLQhn6ofDSwHa4JN7ucQgnSmZLMAzchJya5D4zq8v", + 98: "5EYCAe5jLQhn6ofDSwHqoKui275tEzsS527WAdzp2u5oGNSd", + 99: "5EYCAe5jLQhn6ofDSwJ7YMTJ8bm5hYyJoh41iageVixXYH59", + 100: "5EYCAe5jLQhn6ofDSwJPHNztF6SHA75BYMzXGXNUxYqFoj9g", + 101: "5EYCAe5jLQhn6ofDSwJf2QYUMb7UcfB4H2w2pU4KRNhz5GP5", + 102: "5EYCAe5jLQhn6ofDSwJvmS64U5ng5DGw1hsYNQk9tCaiLvoS", + 103: "5EYCAe5jLQhn6ofDSwKCWTdeaaTsXmNokNp3vMRzM2TScknA", + 104: "5EYCAe5jLQhn6ofDSwKUFVBEh594zKUgV3kZUJ7porLAtE76", + 105: "5EYCAe5jLQhn6ofDSwKjzWipoZpGSsaZDih52EofGgCu9mbP", + 106: "5EYCAe5jLQhn6ofDSwL1jYGQv4VTuRgRxPdaaBVVjW5dRU9u", + 107: "5EYCAe5jLQhn6ofDSwLHUZp12ZAfMynJh4a688BLCKxMhEMq", + 108: "5EYCAe5jLQhn6ofDSwLZDbMb93qrpXtBRjWbg4sAf9q5xtB8", + 109: "5EYCAe5jLQhn6ofDSwLpxcuBFYX4H5z4AQT7E1Z17yhpELLK", + 110: "5EYCAe5jLQhn6ofDSwM6heSmN3CFje5vu5PcmxEqaoaYW1KP", + 111: "5EYCAe5jLQhn6ofDSwMNSfzMUXsTCCBodkL8Ktvg3dTGmYbX", + 112: "5EYCAe5jLQhn6ofDSwMeBhXwb2YeekHgNRGdsqcWWTL13NLP", + 113: "5EYCAe5jLQhn6ofDSwMuvj5XhXDr7JPZ76D9RnJLyHCjK2Zy", + 114: "5EYCAe5jLQhn6ofDSwNBfkd7p1u3ZrVRqm9eyizBS75TaPgK", + 115: "5EYCAe5jLQhn6ofDSwNTQnAhvWaF2QbJaS6AXfg1tvxBrDUN", + 116: "5EYCAe5jLQhn6ofDSwNj9oiJ31FSUxhBK72g5cMrMkpv7iJx", + 117: "5EYCAe5jLQhn6ofDSwNztqFt9VvdwWo43myBdZ3gpahePQpf", + 118: "5EYCAe5jLQhn6ofDSwPGdroUFzbqQ4tvnSuhBVjXHQaNet2o", + 119: "5EYCAe5jLQhn6ofDSwPYNtM4NVH2rczoX7rCjSRMkET6vioH", + 120: "5EYCAe5jLQhn6ofDSwPp7uteUyxEKB6gFnniHP7CD4KqCQDN", + 121: "5EYCAe5jLQhn6ofDSwQ5rwSEbUdRmjCYzTjDqKo2ftCZTubr", + 122: "5EYCAe5jLQhn6ofDSwQMbxyphyJdEHJRj8fjPGUs8i5HjcA3", + 123: "5EYCAe5jLQhn6ofDSwQdLzXQpTypgqQJTocEwDAhbXx21Awy", + 124: "5EYCAe5jLQhn6ofDSwQu624zvxf29PWBCUYkV9rY4MpkGu1f", + 125: "5EYCAe5jLQhn6ofDSwRAq3cb3TLDbwc3w9VG36YNXBhUYKDi", + 126: "5EYCAe5jLQhn6ofDSwRSa5AB9x1R4VhvfpRmb3ECz1aCp2ze", + 127: "5EYCAe5jLQhn6ofDSwRiK6hmGSgcX3ooQVNH8yv3SqSw5mpH", + 128: "5EYCAe5jLQhn6ofDSwRz48FMNwMoybug9AJngvbsufKfME2t", + } + + return NETUID_TO_ACCOUNT_ID[netuid]; } \ No newline at end of file diff --git a/pallets/subtensor/src/coinbase/block_step.rs b/pallets/subtensor/src/coinbase/block_step.rs index a7d41b7f3a..00bdad8193 100644 --- a/pallets/subtensor/src/coinbase/block_step.rs +++ b/pallets/subtensor/src/coinbase/block_step.rs @@ -293,7 +293,7 @@ impl Pallet { pub fn root_proportion(netuid: NetUid) -> U96F32 { let alpha_issuance = U96F32::from_num(Self::get_alpha_issuance(netuid)); - let root_tao: U96F32 = U96F32::from_num(SubnetTAO::::get(NetUid::ROOT)); + let root_tao: U96F32 = U96F32::from_num(Self::get_subnet_tao(NetUid::ROOT)); let tao_weight: U96F32 = root_tao.saturating_mul(Self::get_tao_weight()); let root_proportion: U96F32 = tao_weight diff --git a/pallets/subtensor/src/coinbase/mod.rs b/pallets/subtensor/src/coinbase/mod.rs index 8d06228593..834ef83dd3 100644 --- a/pallets/subtensor/src/coinbase/mod.rs +++ b/pallets/subtensor/src/coinbase/mod.rs @@ -5,3 +5,4 @@ pub mod reveal_commits; pub mod root; pub mod run_coinbase; pub mod subnet_emissions; +pub mod tao; \ No newline at end of file diff --git a/pallets/subtensor/src/coinbase/tao.rs b/pallets/subtensor/src/coinbase/tao.rs new file mode 100644 index 0000000000..fff17a0bf3 --- /dev/null +++ b/pallets/subtensor/src/coinbase/tao.rs @@ -0,0 +1,159 @@ +/// This file contains all critical operations with TAO and Alpha: +/// +/// - Minting, burning, recycling, and transferring +/// - Reading colkey TAO balances +/// - Access to subnet TAO reserves +/// + +use frame_support::traits::{ + Imbalance, fungible::Mutate, + tokens::{ + Fortitude, Precision, Preservation, + fungible::{Balanced as _, Inspect as _}, + }, +}; +use subtensor_runtime_common::{NetUid, TaoBalance}; + +use super::*; + +pub type BalanceOf = + <::Currency as fungible::Inspect<::AccountId>>::Balance; + +impl Pallet { + + pub fn get_subnet_tao(netuid: NetUid) -> TaoBalance { + let maybe_subnet_account = Self::get_subnet_account_id(netuid); + if let Some(subnet_account) = maybe_subnet_account { + Self::get_coldkey_balance(&subnet_account) + } else { + 0.into() + } + } + + pub fn transfer_tao( + origin_coldkey: &T::AccountId, + destination_coldkey: &T::AccountId, + amount: BalanceOf, + ) -> DispatchResult { + ::Currency::transfer(origin_coldkey, destination_coldkey, amount, Preservation::Expendable)?; + Ok(()) + } + + pub fn burn_tao( + coldkey: &T::AccountId, + amount: BalanceOf, + ) -> DispatchResult { + let _ = ::Currency::withdraw( + coldkey, + amount, + Precision::Exact, + Preservation::Expendable, + Fortitude::Force, + ) + .map_err(|_| Error::::BalanceWithdrawalError)? + .peek(); + + Ok(()) + } + + pub fn can_remove_balance_from_coldkey_account( + coldkey: &T::AccountId, + amount: BalanceOf, + ) -> bool { + let current_balance = Self::get_coldkey_balance(coldkey); + if amount > current_balance { + return false; + } + + // This bit is currently untested. @todo + + ::Currency::can_withdraw(coldkey, amount) + .into_result(false) + .is_ok() + } + + pub fn get_coldkey_balance( + coldkey: &T::AccountId, + ) -> BalanceOf + { + ::Currency::reducible_balance( + coldkey, + Preservation::Expendable, + Fortitude::Polite, + ) + } + + pub fn kill_coldkey_account( + coldkey: &T::AccountId, + amount: BalanceOf, + ) -> Result { + if amount.is_zero() { + return Ok(0.into()); + } + + let credit = ::Currency::withdraw( + coldkey, + amount, + Precision::Exact, + Preservation::Expendable, + Fortitude::Force, + ) + .map_err(|_| Error::::BalanceWithdrawalError)? + .peek(); + + if credit.is_zero() { + return Err(Error::::ZeroBalanceAfterWithdrawn.into()); + } + + Ok(credit) + } + + pub fn add_balance_to_coldkey_account( + coldkey: &T::AccountId, + amount: BalanceOf, + ) { + // infallible + let _ = ::Currency::deposit(coldkey, amount, Precision::BestEffort); + } + + #[must_use = "Balance must be used to preserve total issuance of token"] + pub fn remove_balance_from_coldkey_account( + coldkey: &T::AccountId, + amount: BalanceOf, + ) -> Result { + if amount.is_zero() { + return Ok(TaoBalance::ZERO); + } + + let credit = ::Currency::withdraw( + coldkey, + amount, + Precision::BestEffort, + Preservation::Preserve, + Fortitude::Polite, + ) + .map_err(|_| Error::::BalanceWithdrawalError)? + .peek(); + + if credit.is_zero() { + return Err(Error::::ZeroBalanceAfterWithdrawn.into()); + } + + Ok(credit.into()) + } + + // pub fn drain_tao_imbalance_into_subnet_reserve(imbalance: NegativeImbalance, netuid: NetUid) { + // } + + // pub fn mint_tao_for_subnet_reserve(tao: TaoBalance, netuid: NetUid) -> DispatchResult { + // let maybe_subnet_account = SubtensorModule::get_subnet_account_id(netuid); + // if let Some(subnet_account) = maybe_subnet_account { + // let _ = ::Currency::deposit(subnet_account, tao, Precision::BestEffort); + // Ok(()) + // } else { + // Err(Error::::SubnetNotExists) + // } + // } + + +} \ No newline at end of file diff --git a/pallets/subtensor/src/migrations/migrate_subnet_tao_to_subnet_balance.rs b/pallets/subtensor/src/migrations/migrate_subnet_tao_to_subnet_balance.rs new file mode 100644 index 0000000000..8d7d465aa3 --- /dev/null +++ b/pallets/subtensor/src/migrations/migrate_subnet_tao_to_subnet_balance.rs @@ -0,0 +1,4 @@ + // /// --- MAP ( netuid ) --> tao_in_subnet | Returns the amount of TAO in the subnet. + // #[pallet::storage] + // pub type SubnetTAO = + // StorageMap<_, Identity, NetUid, TaoBalance, ValueQuery, DefaultZeroTao>; \ No newline at end of file diff --git a/pallets/subtensor/src/staking/add_stake.rs b/pallets/subtensor/src/staking/add_stake.rs index 17dd9e2eb9..00068c3aa9 100644 --- a/pallets/subtensor/src/staking/add_stake.rs +++ b/pallets/subtensor/src/staking/add_stake.rs @@ -49,8 +49,6 @@ impl Pallet { "do_add_stake( origin:{coldkey:?} hotkey:{hotkey:?}, netuid:{netuid:?}, stake_to_be_added:{stake_to_be_added:?} )" ); - Self::ensure_subtoken_enabled(netuid)?; - // 2. Validate user input Self::validate_add_stake( &coldkey, diff --git a/pallets/subtensor/src/staking/helpers.rs b/pallets/subtensor/src/staking/helpers.rs index 42344faed5..62641c7e07 100644 --- a/pallets/subtensor/src/staking/helpers.rs +++ b/pallets/subtensor/src/staking/helpers.rs @@ -1,11 +1,4 @@ use alloc::collections::BTreeMap; -use frame_support::traits::{ - Imbalance, - tokens::{ - Fortitude, Precision, Preservation, - fungible::{Balanced as _, Inspect as _}, - }, -}; use safe_math::*; use share_pool::SafeFloat; use substrate_fixed::types::U96F32; @@ -288,92 +281,6 @@ impl Pallet { } } - pub fn add_balance_to_coldkey_account( - coldkey: &T::AccountId, - amount: <::Currency as fungible::Inspect<::AccountId>>::Balance, - ) { - // infallible - let _ = ::Currency::deposit(coldkey, amount, Precision::BestEffort); - } - - pub fn can_remove_balance_from_coldkey_account( - coldkey: &T::AccountId, - amount: <::Currency as fungible::Inspect<::AccountId>>::Balance, - ) -> bool { - let current_balance = Self::get_coldkey_balance(coldkey); - if amount > current_balance { - return false; - } - - // This bit is currently untested. @todo - - ::Currency::can_withdraw(coldkey, amount) - .into_result(false) - .is_ok() - } - - pub fn get_coldkey_balance( - coldkey: &T::AccountId, - ) -> <::Currency as fungible::Inspect<::AccountId>>::Balance - { - ::Currency::reducible_balance( - coldkey, - Preservation::Expendable, - Fortitude::Polite, - ) - } - - #[must_use = "Balance must be used to preserve total issuance of token"] - pub fn remove_balance_from_coldkey_account( - coldkey: &T::AccountId, - amount: <::Currency as fungible::Inspect<::AccountId>>::Balance, - ) -> Result { - if amount.is_zero() { - return Ok(TaoBalance::ZERO); - } - - let credit = ::Currency::withdraw( - coldkey, - amount, - Precision::BestEffort, - Preservation::Preserve, - Fortitude::Polite, - ) - .map_err(|_| Error::::BalanceWithdrawalError)? - .peek(); - - if credit.is_zero() { - return Err(Error::::ZeroBalanceAfterWithdrawn.into()); - } - - Ok(credit.into()) - } - - pub fn kill_coldkey_account( - coldkey: &T::AccountId, - amount: <::Currency as fungible::Inspect<::AccountId>>::Balance, - ) -> Result { - if amount.is_zero() { - return Ok(0.into()); - } - - let credit = ::Currency::withdraw( - coldkey, - amount, - Precision::Exact, - Preservation::Expendable, - Fortitude::Force, - ) - .map_err(|_| Error::::BalanceWithdrawalError)? - .peek(); - - if credit.is_zero() { - return Err(Error::::ZeroBalanceAfterWithdrawn.into()); - } - - Ok(credit) - } - pub fn is_user_liquidity_enabled(netuid: NetUid) -> bool { T::SwapInterface::is_user_liquidity_enabled(netuid) } diff --git a/pallets/subtensor/src/staking/move_stake.rs b/pallets/subtensor/src/staking/move_stake.rs index 2cc08b4f02..ed13caf3f8 100644 --- a/pallets/subtensor/src/staking/move_stake.rs +++ b/pallets/subtensor/src/staking/move_stake.rs @@ -471,9 +471,9 @@ impl Pallet { } // Corner case: SubnetTAO for any of two subnets is zero - let subnet_tao_1 = SubnetTAO::::get(origin_netuid) + let subnet_tao_1 = Self::get_subnet_tao(origin_netuid) .saturating_add(SubnetTaoProvided::::get(origin_netuid)); - let subnet_tao_2 = SubnetTAO::::get(destination_netuid) + let subnet_tao_2 = Self::get_subnet_tao(destination_netuid) .saturating_add(SubnetTaoProvided::::get(destination_netuid)); if subnet_tao_1.is_zero() || subnet_tao_2.is_zero() { return Err(Error::::ZeroMaxStakeAmount.into());