Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: introduce pallet-parameters to Westend to parameterize inflation #4938

Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
81da185
feat: setup pallet-parameters in westend runtime
marcuspang Jul 3, 2024
e84a472
feat: introduce dynamic params for staking reward params
marcuspang Jul 3, 2024
e8d2ec5
feat: add new era relay payout function
marcuspang Jul 4, 2024
77a28cc
feat: add new era payout fn with dynamic params
marcuspang Jul 4, 2024
b010d0b
fix: total stakable value + remove NIS
marcuspang Jul 5, 2024
30caa8c
chore: remove unused weight params
marcuspang Jul 5, 2024
08b9287
build: remove unused reward staking curve pallet
marcuspang Jul 5, 2024
282aa21
chore: remove unused era_payout function
marcuspang Jul 5, 2024
ff9e782
chore: leave deprecated function in crate
marcuspang Jul 5, 2024
90f031d
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 8, 2024
ae01d45
test: add unit test to ensure payout logic is same using legacy aucti…
marcuspang Jul 10, 2024
aa563f5
fix: add allow deprecated macro for tests
marcuspang Jul 10, 2024
ae0a72d
fix: missing default trait for runtimeparameters
marcuspang Jul 10, 2024
451f622
docs: add comments for era payout params
marcuspang Jul 10, 2024
849604c
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 10, 2024
e866cd9
doc: update stakable amount description
marcuspang Jul 11, 2024
833d209
refactor: move deprecated fn to test module
marcuspang Jul 11, 2024
2ea077f
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 17, 2024
8843d73
works
kianenigma Jul 17, 2024
704e13c
add weight file
kianenigma Jul 17, 2024
4884a48
fmt toml
kianenigma Jul 17, 2024
c7ee56c
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 17, 2024
ede367a
remove api
kianenigma Jul 17, 2024
fdeaacd
Merge branch 'feat/make-westend-inflation-parameterizable' of github.…
kianenigma Jul 17, 2024
c866a69
update lock
kianenigma Jul 17, 2024
7ce4449
fix
kianenigma Jul 17, 2024
e8037de
remove more unused stuff
kianenigma Jul 17, 2024
8d62d46
add prodoc
kianenigma Jul 18, 2024
810b2b0
Merge branch 'master' of github.com:paritytech/polkadot-sdk into feat…
kianenigma Jul 18, 2024
d8760c7
add release related stuff
kianenigma Jul 18, 2024
026dcb6
revert
kianenigma Jul 18, 2024
2e128ac
".git/.scripts/commands/fmt/fmt.sh"
Jul 18, 2024
8a6d11e
remove unused
kianenigma Jul 18, 2024
6ab2d43
Merge branch 'master' of github.com:paritytech/polkadot-sdk into feat…
kianenigma Jul 18, 2024
e993b59
Merge branch 'feat/make-westend-inflation-parameterizable' of github.…
kianenigma Jul 18, 2024
e520f23
Empty-Commit
kianenigma Jul 18, 2024
28c1c2d
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 18, 2024
98d7d91
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 19, 2024
4f6316a
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 19, 2024
b972461
Merge branch 'master' into feat/make-westend-inflation-parameterizable
kianenigma Jul 22, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

49 changes: 48 additions & 1 deletion polkadot/runtime/common/src/impls.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ use frame_support::traits::{
};
use pallet_treasury::TreasuryAccountId;
use polkadot_primitives::Balance;
use sp_runtime::{traits::TryConvert, Perquintill, RuntimeDebug};
use sp_runtime::{traits::{TryConvert, Saturating}, Perquintill, RuntimeDebug};
use xcm::VersionedLocation;

/// Logic for the author to get a portion of fees.
Expand Down Expand Up @@ -67,6 +67,53 @@ where
}
}

pub struct EraPayoutParams {
pub total_staked: Balance,
pub total_stakable: Balance,
pub ideal_stake: Perquintill,
pub max_annual_inflation: Perquintill,
pub min_annual_inflation: Perquintill,
pub falloff: Perquintill,
pub period_fraction: Perquintill,
pub legacy_auction_proportion: Option<Perquintill>,
}

pub fn relay_era_payout(params: EraPayoutParams) -> (Balance, Balance) {
let EraPayoutParams {
total_staked,
total_stakable,
ideal_stake,
max_annual_inflation,
min_annual_inflation,
falloff,
period_fraction,
legacy_auction_proportion,
} = params;

let delta_annual_inflation = max_annual_inflation.saturating_sub(min_annual_inflation);

let ideal_stake = ideal_stake.saturating_sub(legacy_auction_proportion.unwrap_or_default());

let stake = Perquintill::from_rational(total_staked, total_stakable);
let adjustment = pallet_staking_reward_fn::compute_inflation(stake, ideal_stake, falloff);
let staking_inflation =
min_annual_inflation.saturating_add(delta_annual_inflation * adjustment);

let max_payout = period_fraction * max_annual_inflation * total_stakable;
let staking_payout = (period_fraction * staking_inflation) * total_stakable;
let rest = max_payout.saturating_sub(staking_payout);

let other_issuance = total_stakable.saturating_sub(total_staked);
if total_staked > other_issuance {
let _cap_rest = Perquintill::from_rational(other_issuance, total_staked) * staking_payout;
// We don't do anything with this, but if we wanted to, we could introduce a cap on the
// treasury amount with: `rest = rest.min(cap_rest);`
}
(staking_payout, rest)
}


#[deprecated = "use relay_era_payout instead which supports dynamic parameters"]
marcuspang marked this conversation as resolved.
Show resolved Hide resolved
pub fn era_payout(
total_staked: Balance,
total_stakable: Balance,
Expand Down
9 changes: 8 additions & 1 deletion polkadot/runtime/westend/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ pallet-multisig = { workspace = true }
pallet-nomination-pools = { workspace = true }
pallet-conviction-voting = { workspace = true }
pallet-offences = { workspace = true }
pallet-parameters = { workspace = true }
pallet-preimage = { workspace = true }
pallet-proxy = { workspace = true }
pallet-recovery = { workspace = true }
Expand Down Expand Up @@ -174,6 +175,7 @@ std = [
"pallet-nomination-pools/std",
"pallet-offences-benchmarking?/std",
"pallet-offences/std",
"pallet-parameters/std",
"pallet-preimage/std",
"pallet-proxy/std",
"pallet-recovery/std",
Expand Down Expand Up @@ -261,6 +263,7 @@ runtime-benchmarks = [
"pallet-nomination-pools/runtime-benchmarks",
"pallet-offences-benchmarking/runtime-benchmarks",
"pallet-offences/runtime-benchmarks",
"pallet-parameters/runtime-benchmarks",
"pallet-preimage/runtime-benchmarks",
"pallet-proxy/runtime-benchmarks",
"pallet-recovery/runtime-benchmarks",
Expand Down Expand Up @@ -319,6 +322,7 @@ try-runtime = [
"pallet-multisig/try-runtime",
"pallet-nomination-pools/try-runtime",
"pallet-offences/try-runtime",
"pallet-parameters/try-runtime",
"pallet-preimage/try-runtime",
"pallet-proxy/try-runtime",
"pallet-recovery/try-runtime",
Expand Down Expand Up @@ -348,7 +352,10 @@ metadata-hash = ["substrate-wasm-builder/metadata-hash"]
# Set timing constants (e.g. session period) to faster versions to speed up testing.
fast-runtime = []

runtime-metrics = ["polkadot-runtime-parachains/runtime-metrics", "sp-io/with-tracing"]
runtime-metrics = [
"polkadot-runtime-parachains/runtime-metrics",
"sp-io/with-tracing",
]

# A feature that should be enabled when the runtime should be built for on-chain
# deployment. This will disable stuff that shouldn't be part of the on-chain wasm
Expand Down
176 changes: 159 additions & 17 deletions polkadot/runtime/westend/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,19 +24,22 @@ use codec::{Decode, Encode, MaxEncodedLen};
use frame_election_provider_support::{bounds::ElectionBoundsBuilder, onchain, SequentialPhragmen};
use frame_support::{
derive_impl,
dynamic_params::{dynamic_pallet_params, dynamic_params},
genesis_builder_helper::{build_state, get_preset},
parameter_types,
traits::{
fungible::HoldConsideration, tokens::UnityOrOuterConversion, ConstU32, Contains, EitherOf,
EitherOfDiverse, EverythingBut, FromContains, InstanceFilter, KeyOwnerProofSystem,
LinearStoragePrice, ProcessMessage, ProcessMessageError, VariantCountOf, WithdrawReasons,
EitherOfDiverse, EnsureOrigin, EnsureOriginWithArg, EverythingBut, FromContains,
InstanceFilter, KeyOwnerProofSystem, LinearStoragePrice, ProcessMessage,
ProcessMessageError, VariantCountOf, WithdrawReasons,
},
weights::{ConstantMultiplier, WeightMeter, WeightToFee as _},
PalletId,
};
use frame_system::{EnsureRoot, EnsureSigned};
use pallet_grandpa::{fg_primitives, AuthorityId as GrandpaId};
use pallet_identity::legacy::IdentityInfo;
use pallet_nis::WithMaximumOf;
use pallet_session::historical as session_historical;
use pallet_transaction_payment::{FeeDetails, FungibleAdapter, RuntimeDispatchInfo};
use polkadot_primitives::{
Expand All @@ -45,15 +48,15 @@ use polkadot_primitives::{
GroupRotationInfo, Hash, Id as ParaId, InboundDownwardMessage, InboundHrmpMessage, Moment,
NodeFeatures, Nonce, OccupiedCoreAssumption, PersistedValidationData, PvfCheckStatement,
ScrapedOnChainVotes, SessionInfo, Signature, ValidationCode, ValidationCodeHash, ValidatorId,
ValidatorIndex, ValidatorSignature, PARACHAIN_KEY_TYPE_ID,
ValidatorIndex, ValidatorSignature, LOWEST_PUBLIC_ID, PARACHAIN_KEY_TYPE_ID,
};
use polkadot_runtime_common::{
assigned_slots, auctions, crowdloan,
elections::OnChainAccuracy,
identity_migrator, impl_runtime_weights,
impls::{
ContainsParts, LocatableAssetConverter, ToAuthor, VersionedLocatableAsset,
VersionedLocationConverter,
relay_era_payout, ContainsParts, EraPayoutParams, LocatableAssetConverter, ToAuthor,
VersionedLocatableAsset, VersionedLocationConverter,
},
paras_registrar, paras_sudo_wrapper, prod_or_fast, slots,
traits::{Leaser, OnSwap},
Expand Down Expand Up @@ -241,6 +244,74 @@ parameter_types! {
pub const PreimageHoldReason: RuntimeHoldReason = RuntimeHoldReason::Preimage(pallet_preimage::HoldReason::Preimage);
}

/// Dynamic params that can be adjusted at runtime.
#[dynamic_params(RuntimeParameters, pallet_parameters::Parameters::<Runtime>)]
pub mod dynamic_params {
use super::*;

#[dynamic_pallet_params]
#[codec(index = 0)]
pub mod nis {
use super::*;

#[codec(index = 0)]
pub static Target: Perquintill = Perquintill::zero();

#[codec(index = 1)]
pub static MinBid: Balance = 100 * UNITS;
}

#[dynamic_pallet_params]
#[codec(index = 1)]
pub mod inflation {
// TODO: should these values be updated to match the new inflation pallet?
#[codec(index = 0)]
pub static MinInflation: Perquintill = Perquintill::from_rational(25u64, 1000u64);
marcuspang marked this conversation as resolved.
Show resolved Hide resolved

#[codec(index = 1)]
pub static MaxInflation: Perquintill = Perquintill::from_rational(100u64, 1000u64);

#[codec(index = 2)]
pub static IdealStake: Perquintill = Perquintill::from_rational(500u64, 1000u64);

#[codec(index = 3)]
pub static Falloff: Perquintill = Perquintill::from_rational(50u64, 1000u64);

// TODO: what should be the default here?
#[codec(index = 4)]
pub static UseAuctionSlots: bool = false;
}
}

impl pallet_parameters::Config for Runtime {
type RuntimeEvent = RuntimeEvent;
type RuntimeParameters = RuntimeParameters;
type AdminOrigin = DynamicParameterOrigin;
// TODO: add benchmarking and update weight info
type WeightInfo = ();
}

/// Defines what origin can modify which dynamic parameters.
pub struct DynamicParameterOrigin;
impl EnsureOriginWithArg<RuntimeOrigin, RuntimeParametersKey> for DynamicParameterOrigin {
type Success = ();

fn try_origin(
origin: RuntimeOrigin,
key: &RuntimeParametersKey,
) -> Result<Self::Success, RuntimeOrigin> {
use crate::{dynamic_params::*, governance::*, RuntimeParametersKey::*};
// TODO: update to correct origin

match key {
Nis(nis::ParametersKey::MinBid(_)) => StakingAdmin::ensure_origin(origin.clone()),
Nis(nis::ParametersKey::Target(_)) => GeneralAdmin::ensure_origin(origin.clone()),
Inflation(_) => frame_system::ensure_root(origin.clone()),
kianenigma marked this conversation as resolved.
Show resolved Hide resolved
}
.map_err(|_| origin)
}
}

impl pallet_preimage::Config for Runtime {
type WeightInfo = weights::pallet_preimage::WeightInfo<Runtime>;
type RuntimeEvent = RuntimeEvent;
Expand Down Expand Up @@ -596,15 +667,79 @@ impl pallet_bags_list::Config<VoterBagsListInstance> for Runtime {
type Score = sp_npos_elections::VoteWeight;
}

pallet_staking_reward_curve::build! {
const REWARD_CURVE: PiecewiseLinear<'static> = curve!(
min_inflation: 0_025_000,
max_inflation: 0_100_000,
ideal_stake: 0_500_000,
falloff: 0_050_000,
max_piece_count: 40,
test_precision: 0_005_000,
);
// parameter_types! {
// pub const NisBasePeriod: BlockNumber = 30 * DAYS;
// pub MinReceipt: Perquintill = Perquintill::from_rational(1u64, 10_000_000u64);
// pub const IntakePeriod: BlockNumber = 5 * MINUTES;
// pub MaxIntakeWeight: Weight = MAXIMUM_BLOCK_WEIGHT / 10;
// pub const ThawThrottle: (Perquintill, BlockNumber) = (Perquintill::from_percent(25), 5);
// pub const NisPalletId: PalletId = PalletId(*b"py/nis ");
// }

// impl pallet_nis::Config for Runtime {
// type WeightInfo = weights::pallet_nis::WeightInfo<Runtime>;
// type RuntimeEvent = RuntimeEvent;
// type Currency = Balances;
// type CurrencyBalance = Balance;
// type FundOrigin = frame_system::EnsureSigned<AccountId>;
// type Counterpart = NisCounterpartBalances;
// type CounterpartAmount = WithMaximumOf<ConstU128<21_000_000_000_000_000_000u128>>;
// type Deficit = (); // Mint
// type IgnoredIssuance = ();
// type Target = dynamic_params::nis::Target;
// type PalletId = NisPalletId;
// type QueueCount = ConstU32<300>;
// type MaxQueueLen = ConstU32<1000>;
// type FifoQueueLen = ConstU32<250>;
// type BasePeriod = NisBasePeriod;
// type MinBid = dynamic_params::nis::MinBid;
// type MinReceipt = MinReceipt;
// type IntakePeriod = IntakePeriod;
// type MaxIntakeWeight = MaxIntakeWeight;
// type ThawThrottle = ThawThrottle;
// type RuntimeHoldReason = RuntimeHoldReason;
// #[cfg(feature = "runtime-benchmarks")]
// type BenchmarkSetup = ();
// }

pub struct EraPayout;
impl pallet_staking::EraPayout<Balance> for EraPayout {
fn era_payout(
total_staked: Balance,
_total_issuance: Balance,
era_duration_millis: u64,
) -> (Balance, Balance) {
let auctioned_slots: u64 = 0;
// TODO: figure out if using parachains here makes sense
// // all para-ids that are currently active.
// let auctioned_slots = Paras::parachains()
// .into_iter()
// // all active para-ids that do not belong to a system chain is the number
// // of parachains that we should take into account for inflation.
// .filter(|i| *i >= LOWEST_PUBLIC_ID)
// .count() as u64;
const MAX_ANNUAL_INFLATION: Perquintill = Perquintill::from_percent(10);
const MILLISECONDS_PER_YEAR: u64 = 1000 * 3600 * 24 * 36525 / 100;

let use_auctioned_slots = dynamic_params::inflation::UseAuctionSlots::get();
let params = EraPayoutParams {
total_staked,
// TODO: update this to use NIS issuance
total_stakable: 1,
marcuspang marked this conversation as resolved.
Show resolved Hide resolved
// total_stakable: Nis::issuance().other,
ideal_stake: dynamic_params::inflation::IdealStake::get(),
max_annual_inflation: dynamic_params::inflation::MaxInflation::get(),
min_annual_inflation: dynamic_params::inflation::MinInflation::get(),
falloff: dynamic_params::inflation::Falloff::get(),
period_fraction: Perquintill::from_rational(era_duration_millis, MILLISECONDS_PER_YEAR),
legacy_auction_proportion: if use_auctioned_slots {
Some(Perquintill::from_rational(auctioned_slots.min(60), 200))
} else {
None
},
};
relay_era_payout(params)
}
}

parameter_types! {
Expand All @@ -614,7 +749,6 @@ parameter_types! {
pub const BondingDuration: sp_staking::EraIndex = 2;
// 1 era in which slashes can be cancelled (6 hours).
pub const SlashDeferDuration: sp_staking::EraIndex = 1;
pub const RewardCurve: &'static PiecewiseLinear<'static> = &REWARD_CURVE;
pub const MaxExposurePageSize: u32 = 64;
// Note: this is not really correct as Max Nominators is (MaxExposurePageSize * page_count) but
// this is an unbounded number. We just set it to a reasonably high value, 1 full page
Expand All @@ -638,7 +772,7 @@ impl pallet_staking::Config for Runtime {
type SlashDeferDuration = SlashDeferDuration;
type AdminOrigin = EitherOf<EnsureRoot<AccountId>, StakingAdmin>;
type SessionInterface = Self;
type EraPayout = pallet_staking::ConvertCurve<RewardCurve>;
type EraPayout = EraPayout;
type MaxExposurePageSize = MaxExposurePageSize;
type NextNewSession = Session;
type ElectionProvider = ElectionProviderMultiPhase;
Expand Down Expand Up @@ -1008,7 +1142,8 @@ impl InstanceFilter<RuntimeCall> for ProxyType {
matches!(
c,
RuntimeCall::Staking(..) |
RuntimeCall::Session(..) | RuntimeCall::Utility(..) |
RuntimeCall::Session(..) |
RuntimeCall::Utility(..) |
RuntimeCall::FastUnstake(..) |
RuntimeCall::VoterList(..) |
RuntimeCall::NominationPools(..)
Expand Down Expand Up @@ -1475,6 +1610,13 @@ mod runtime {
pub type Offences = pallet_offences;
#[runtime::pallet_index(27)]
pub type Historical = session_historical;
#[runtime::pallet_index(70)]
pub type Parameters = pallet_parameters;
// NIS pallet.
// #[runtime::pallet_index(71)]
// pub type Nis = pallet_nis;
// #[runtime::pallet_index(72)]
// pub type NisCounterpartBalances = pallet_balances::Instance2;

#[runtime::pallet_index(8)]
pub type Session = pallet_session;
Expand Down