feat: add staking score pallet to relay chain and fix referral default
Relay Chain: - Add pezpallet-staking-score to runtime - Implement RelayStakingInfoProvider to read from pallet_staking - StakingScore pallet index = 92 People Chain: - Add DefaultReferrer type to identity-kyc pallet Config - Change DefaultReferrer from Alice to founder address - Make referrer parameter optional in apply_for_citizenship - Fallback to DefaultReferrer when no valid referrer provided
This commit is contained in:
@@ -131,6 +131,9 @@ pub mod pezpallet {
|
||||
|
||||
type WeightInfo: WeightInfo;
|
||||
|
||||
/// Default referrer account (founder) - used when no valid referrer is provided
|
||||
type DefaultReferrer: Get<Self::AccountId>;
|
||||
|
||||
/// Hook called when citizenship is approved - used by referral pezpallet
|
||||
type OnKycApproved: crate::types::OnKycApproved<Self::AccountId>;
|
||||
|
||||
@@ -284,49 +287,59 @@ pub mod pezpallet {
|
||||
///
|
||||
/// # Arguments
|
||||
/// - `identity_hash`: H256 hash of identity documents (calculated off-chain)
|
||||
/// - `referrer`: Account of existing citizen who will vouch for you
|
||||
/// - `referrer`: Optional account of existing citizen who will vouch for you.
|
||||
/// If None or invalid, DefaultReferrer (founder) is used.
|
||||
///
|
||||
/// # Workflow
|
||||
/// 1. Applicant submits hash + referrer
|
||||
/// 2. Deposit is reserved (spam prevention)
|
||||
/// 3. Status becomes PendingReferral
|
||||
/// 4. Referrer must call approve_referral
|
||||
/// 1. Applicant submits hash + optional referrer
|
||||
/// 2. If referrer is None/invalid, DefaultReferrer is used
|
||||
/// 3. Deposit is reserved (spam prevention)
|
||||
/// 4. Status becomes PendingReferral
|
||||
/// 5. Referrer must call approve_referral
|
||||
#[pezpallet::call_index(0)]
|
||||
#[pezpallet::weight(T::WeightInfo::apply_for_citizenship())]
|
||||
pub fn apply_for_citizenship(
|
||||
origin: OriginFor<T>,
|
||||
identity_hash: H256,
|
||||
referrer: T::AccountId,
|
||||
referrer: Option<T::AccountId>,
|
||||
) -> DispatchResult {
|
||||
let applicant = ensure_signed(origin)?;
|
||||
|
||||
// Cannot refer yourself
|
||||
ensure!(applicant != referrer, Error::<T>::SelfReferral);
|
||||
|
||||
// Must not have existing application
|
||||
ensure!(
|
||||
KycStatuses::<T>::get(&applicant) == KycLevel::NotStarted,
|
||||
Error::<T>::ApplicationAlreadyExists
|
||||
);
|
||||
|
||||
// Referrer must be an approved citizen
|
||||
// Determine the actual referrer:
|
||||
// 1. Use provided referrer if valid (approved citizen and not self)
|
||||
// 2. Fall back to DefaultReferrer otherwise
|
||||
let actual_referrer = referrer
|
||||
.filter(|r| *r != applicant) // Not self-referral
|
||||
.filter(|r| KycStatuses::<T>::get(r) == KycLevel::Approved) // Must be citizen
|
||||
.unwrap_or_else(|| T::DefaultReferrer::get());
|
||||
|
||||
// Verify the actual referrer is valid (including DefaultReferrer)
|
||||
ensure!(
|
||||
KycStatuses::<T>::get(&referrer) == KycLevel::Approved,
|
||||
KycStatuses::<T>::get(&actual_referrer) == KycLevel::Approved,
|
||||
Error::<T>::ReferrerNotCitizen
|
||||
);
|
||||
|
||||
// Cannot refer yourself (even with DefaultReferrer)
|
||||
ensure!(applicant != actual_referrer, Error::<T>::SelfReferral);
|
||||
|
||||
// Reserve deposit (spam prevention, returned on approval)
|
||||
let deposit = T::KycApplicationDeposit::get();
|
||||
T::Currency::reserve(&applicant, deposit)?;
|
||||
|
||||
// Store application (only hash, no personal data)
|
||||
let application = CitizenshipApplication { identity_hash, referrer: referrer.clone() };
|
||||
let application = CitizenshipApplication { identity_hash, referrer: actual_referrer.clone() };
|
||||
Applications::<T>::insert(&applicant, application);
|
||||
|
||||
// Update status
|
||||
KycStatuses::<T>::insert(&applicant, KycLevel::PendingReferral);
|
||||
|
||||
Self::deposit_event(Event::CitizenshipApplied { applicant, referrer, identity_hash });
|
||||
Self::deposit_event(Event::CitizenshipApplied { applicant, referrer: actual_referrer, identity_hash });
|
||||
Ok(())
|
||||
}
|
||||
|
||||
|
||||
@@ -298,6 +298,7 @@ impl pezpallet_identity_kyc::Config for Runtime {
|
||||
type KycApplicationDeposit = KycApplicationDeposit;
|
||||
type MaxStringLength = MaxStringLength;
|
||||
type MaxCidLength = MaxCidLength;
|
||||
type DefaultReferrer = DefaultReferrer;
|
||||
}
|
||||
|
||||
// =============================================================================
|
||||
@@ -371,8 +372,14 @@ impl pezpallet_perwerde::Config for Runtime {
|
||||
// =============================================================================
|
||||
|
||||
parameter_types! {
|
||||
/// Default referrer account (genesis/system account)
|
||||
pub DefaultReferrer: AccountId = pezsp_keyring::Sr25519Keyring::Alice.to_account_id();
|
||||
/// Default referrer account - Founder address
|
||||
/// SS58: 5CyuFfbF95rzBxru7c9yEsX4XmQXUxpLUcbj9RLg9K1cGiiF
|
||||
pub DefaultReferrer: AccountId = AccountId::from([
|
||||
0x28, 0x92, 0x5e, 0xd8, 0xb4, 0xc0, 0xc9, 0x54,
|
||||
0x02, 0xb3, 0x15, 0x63, 0x25, 0x1f, 0xd3, 0x18,
|
||||
0x41, 0x43, 0x51, 0x11, 0x4b, 0x1c, 0x77, 0x97,
|
||||
0xee, 0x78, 0x86, 0x66, 0xd2, 0x7d, 0x63, 0x05,
|
||||
]);
|
||||
/// Penalty per revocation (trust score reduction)
|
||||
pub const PenaltyPerRevocation: u32 = 10;
|
||||
}
|
||||
|
||||
@@ -112,6 +112,7 @@ pezkuwi-teyrchain-primitives = { workspace = true }
|
||||
|
||||
# Custom Pezkuwi Pallets
|
||||
pezpallet-validator-pool = { workspace = true }
|
||||
pezpallet-staking-score = { workspace = true }
|
||||
|
||||
xcm = { workspace = true }
|
||||
xcm-builder = { workspace = true }
|
||||
@@ -192,6 +193,7 @@ std = [
|
||||
"pezpallet-treasury/std",
|
||||
"pezpallet-utility/std",
|
||||
"pezpallet-validator-pool/std",
|
||||
"pezpallet-staking-score/std",
|
||||
"pezpallet-vesting/std",
|
||||
"pezpallet-whitelist/std",
|
||||
"pezpallet-xcm-benchmarks?/std",
|
||||
@@ -284,6 +286,7 @@ runtime-benchmarks = [
|
||||
"pezpallet-treasury/runtime-benchmarks",
|
||||
"pezpallet-utility/runtime-benchmarks",
|
||||
"pezpallet-validator-pool/runtime-benchmarks",
|
||||
"pezpallet-staking-score/runtime-benchmarks",
|
||||
"pezpallet-vesting/runtime-benchmarks",
|
||||
"pezpallet-whitelist/runtime-benchmarks",
|
||||
"pezpallet-xcm-benchmarks/runtime-benchmarks",
|
||||
@@ -366,6 +369,7 @@ try-runtime = [
|
||||
"pezpallet-treasury/try-runtime",
|
||||
"pezpallet-utility/try-runtime",
|
||||
"pezpallet-validator-pool/try-runtime",
|
||||
"pezpallet-staking-score/try-runtime",
|
||||
"pezpallet-vesting/try-runtime",
|
||||
"pezpallet-whitelist/try-runtime",
|
||||
"pezpallet-xcm-benchmarks?/try-runtime",
|
||||
|
||||
@@ -565,6 +565,42 @@ impl pezpallet_staking::Config for Runtime {
|
||||
type BenchmarkingConfig = PezkuwiStakingBenchmarkingConfig;
|
||||
}
|
||||
|
||||
// =====================================================
|
||||
// STAKING SCORE CONFIGURATION
|
||||
// =====================================================
|
||||
|
||||
/// Relay Chain StakingInfoProvider - reads directly from pezpallet_staking
|
||||
/// This is the REAL implementation that accesses actual staking data
|
||||
pub struct RelayStakingInfoProvider;
|
||||
|
||||
impl pezpallet_staking_score::StakingInfoProvider<AccountId, Balance>
|
||||
for RelayStakingInfoProvider
|
||||
{
|
||||
fn get_staking_details(
|
||||
who: &AccountId,
|
||||
) -> Option<pezpallet_staking_score::StakingDetails<Balance>> {
|
||||
// Get staking ledger from pezpallet_staking
|
||||
let ledger = pezpallet_staking::Ledger::<Runtime>::get(who)?;
|
||||
|
||||
// Get nominations if any
|
||||
let nominations_count = pezpallet_staking::Nominators::<Runtime>::get(who)
|
||||
.map(|n| n.targets.len() as u32)
|
||||
.unwrap_or(0);
|
||||
|
||||
Some(pezpallet_staking_score::StakingDetails {
|
||||
staked_amount: ledger.active,
|
||||
nominations_count,
|
||||
unlocking_chunks_count: ledger.unlocking.len() as u32,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl pezpallet_staking_score::Config for Runtime {
|
||||
type Balance = Balance;
|
||||
type StakingInfo = RelayStakingInfoProvider;
|
||||
type WeightInfo = pezpallet_staking_score::weights::BizinikiwiWeight<Runtime>;
|
||||
}
|
||||
|
||||
// =====================================================
|
||||
// FAST UNSTAKE CONFIGURATION
|
||||
// =====================================================
|
||||
@@ -1570,6 +1606,9 @@ construct_runtime! {
|
||||
// TNPoS Validator Pool - Shadow Mode (runs parallel to NPoS)
|
||||
ValidatorPool: pezpallet_validator_pool = 91,
|
||||
|
||||
// Staking Score - Time-weighted staking reputation score
|
||||
StakingScore: pezpallet_staking_score = 92,
|
||||
|
||||
// Root testing pezpallet.
|
||||
RootTesting: pezpallet_root_testing = 249,
|
||||
|
||||
@@ -1821,6 +1860,7 @@ mod benches {
|
||||
[pezpallet_whitelist, Whitelist]
|
||||
// Pezkuwichain Custom Pallets
|
||||
[pezpallet_validator_pool, ValidatorPool]
|
||||
[pezpallet_staking_score, StakingScore]
|
||||
// XCM
|
||||
[pezpallet_xcm, PalletXcmExtrinsicsBenchmark::<Runtime>]
|
||||
[pezpallet_xcm_benchmarks::fungible, pezpallet_xcm_benchmarks::fungible::Pezpallet::<Runtime>]
|
||||
|
||||
Reference in New Issue
Block a user