use std::fmt::Display;
use anyhow::Context;
use anyhow::Result;
use penumbra_sdk_auction::params::AuctionParameters;
use penumbra_sdk_community_pool::params::CommunityPoolParameters;
use penumbra_sdk_dex::DexParameters;
use penumbra_sdk_distributions::params::DistributionsParameters;
use penumbra_sdk_fee::FeeParameters;
use penumbra_sdk_funding::params::FundingParameters;
use penumbra_sdk_governance::change::ParameterChange;
use penumbra_sdk_governance::{params::GovernanceParameters, tally::Ratio};
use penumbra_sdk_ibc::params::IBCParameters;
use penumbra_sdk_sct::params::SctParameters;
use penumbra_sdk_shielded_pool::params::ShieldedPoolParameters;
use penumbra_sdk_stake::params::StakeParameters;
use super::AppParameters;
pub trait ParameterChangeExt {
fn apply_changes(&self, app_parameters: AppParameters) -> Result<AppParameters, anyhow::Error>;
}
impl ParameterChangeExt for ParameterChange {
fn apply_changes(&self, app_parameters: AppParameters) -> Result<AppParameters, anyhow::Error> {
for _change in &self.changes {
}
let app_parameters_raw = serde_json::value::to_value(app_parameters.clone())
.context("could not encode app parameters to json value")?;
let new_app_parameters_raw = self
.apply_changes_raw(app_parameters_raw)
.context("error applying parameter changes")?;
let new_app_parameters = serde_json::value::from_value(new_app_parameters_raw)
.context("error parsing changed app parameters")?;
app_parameters
.check_valid_update(&new_app_parameters)
.context("parameter change was invalid")?;
Ok(new_app_parameters)
}
}
#[deny(unused)] impl AppParameters {
pub fn check_valid_update(&self, new: &AppParameters) -> Result<()> {
new.check_valid()?;
let AppParameters {
chain_id,
auction_params: AuctionParameters {},
community_pool_params:
CommunityPoolParameters {
community_pool_spend_proposals_enabled: _,
},
distributions_params:
DistributionsParameters {
staking_issuance_per_block: _,
},
fee_params:
FeeParameters {
fixed_gas_prices: _,
fixed_alt_gas_prices: _,
},
funding_params: FundingParameters {},
governance_params:
GovernanceParameters {
proposal_voting_blocks: _,
proposal_deposit_amount: _,
proposal_valid_quorum,
proposal_pass_threshold,
proposal_slash_threshold,
},
ibc_params:
IBCParameters {
ibc_enabled: _,
inbound_ics20_transfers_enabled: _,
outbound_ics20_transfers_enabled: _,
},
sct_params: SctParameters { epoch_duration },
shielded_pool_params: ShieldedPoolParameters { fmd_meta_params: _ },
stake_params:
StakeParameters {
active_validator_limit,
slashing_penalty_misbehavior: _,
slashing_penalty_downtime: _,
signed_blocks_window_len,
missed_blocks_maximum: _,
min_validator_stake: _,
unbonding_delay: _,
},
dex_params:
DexParameters {
is_enabled: _,
fixed_candidates: _,
max_hops: _,
max_positions_per_pair: _,
max_execution_budget: _,
},
} = self;
check_invariant([(chain_id, &new.chain_id, "chain ID")])?;
check_invariant([
(
epoch_duration,
&new.sct_params.epoch_duration,
"epoch duration",
),
(
active_validator_limit,
&new.stake_params.active_validator_limit,
"active validator limit",
),
(
signed_blocks_window_len,
&new.stake_params.signed_blocks_window_len,
"signed blocks window length",
),
])?;
check_invariant([
(
proposal_valid_quorum,
&new.governance_params.proposal_valid_quorum,
"proposal valid quorum",
),
(
proposal_pass_threshold,
&new.governance_params.proposal_pass_threshold,
"proposal pass threshold",
),
(
proposal_slash_threshold,
&new.governance_params.proposal_slash_threshold,
"proposal slash threshold",
),
])?;
Ok(())
}
pub fn check_valid(&self) -> Result<()> {
let AppParameters {
chain_id,
auction_params: AuctionParameters {},
community_pool_params:
CommunityPoolParameters {
community_pool_spend_proposals_enabled: _,
},
distributions_params:
DistributionsParameters {
staking_issuance_per_block: _,
},
fee_params:
FeeParameters {
fixed_gas_prices: _,
fixed_alt_gas_prices: _,
},
funding_params: FundingParameters {},
governance_params:
GovernanceParameters {
proposal_voting_blocks,
proposal_deposit_amount,
proposal_valid_quorum,
proposal_pass_threshold,
proposal_slash_threshold,
},
ibc_params:
IBCParameters {
ibc_enabled,
inbound_ics20_transfers_enabled,
outbound_ics20_transfers_enabled,
},
sct_params: SctParameters { epoch_duration },
shielded_pool_params: ShieldedPoolParameters { fmd_meta_params: _ },
stake_params:
StakeParameters {
active_validator_limit,
slashing_penalty_misbehavior,
slashing_penalty_downtime,
signed_blocks_window_len,
missed_blocks_maximum,
min_validator_stake,
unbonding_delay,
},
dex_params:
DexParameters {
is_enabled: _,
fixed_candidates: _,
max_hops: _,
max_positions_per_pair: _,
max_execution_budget: _,
},
} = self;
check_all([
(!chain_id.is_empty(), "chain ID must be a non-empty string"),
(
*epoch_duration >= 1,
"epoch duration must be at least one block",
),
(
*unbonding_delay >= epoch_duration * 2 + 1,
"unbonding must take at least two epochs",
),
(
*active_validator_limit > 3,
"active validator limit must be at least 4",
),
(
*slashing_penalty_misbehavior >= 1,
"slashing penalty (misbehavior) must be at least 1 basis point",
),
(
*slashing_penalty_misbehavior <= 100_000_000,
"slashing penalty (misbehavior) must be at most 10,000 basis points^2",
),
(
*slashing_penalty_downtime >= 1,
"slashing penalty (downtime) must be at least 1 basis point",
),
(
*slashing_penalty_downtime <= 100_000_000,
"slashing penalty (downtime) must be at most 10,000 basis points^2",
),
(
*signed_blocks_window_len >= 2,
"signed blocks window length must be at least 2",
),
(
*missed_blocks_maximum >= 1,
"missed blocks maximum must be at least 1",
),
(
(!*inbound_ics20_transfers_enabled && !*outbound_ics20_transfers_enabled)
|| *ibc_enabled,
"IBC must be enabled if either inbound or outbound ICS20 transfers are enabled",
),
(
*proposal_voting_blocks >= 1,
"proposal voting blocks must be at least 1",
),
(
*proposal_deposit_amount >= 1u64.into(),
"proposal deposit amount must be at least 1",
),
(
*proposal_valid_quorum > Ratio::new(0, 1),
"proposal valid quorum must be greater than 0",
),
(
*proposal_pass_threshold >= Ratio::new(1, 2),
"proposal pass threshold must be greater than or equal to 1/2",
),
(
*proposal_slash_threshold > Ratio::new(1, 2),
"proposal slash threshold must be greater than 1/2",
),
(
*min_validator_stake >= 1_000_000u128.into(),
"the minimum validator stake must be at least 1penumbra",
),
])
}
}
fn check_all<'a>(checks: impl IntoIterator<Item = (bool, impl Display + 'a)>) -> Result<()> {
let failed_because = checks
.into_iter()
.filter_map(|(ok, description)| {
if !ok {
Some(description.to_string())
} else {
None
}
})
.collect::<Vec<_>>();
if !failed_because.is_empty() {
anyhow::bail!("invalid chain parameters: {}", failed_because.join(", "));
}
Ok(())
}
fn check_invariant<'a, T: Eq + 'a>(
sides: impl IntoIterator<Item = (&'a T, &'a T, impl Display + 'a)>,
) -> Result<()> {
check_all(
sides
.into_iter()
.map(|(old, new, name)| ((*old == *new), format!("{name} can't be changed"))),
)
}