use crate::{
configuration::{self, HostConfiguration},
disputes::{self, DisputesHandler as _, SlashingHandler as _},
dmp, hrmp, inclusion, paras, scheduler, session_info, shared,
};
use alloc::vec::Vec;
use codec::{Decode, Encode};
use pezframe_support::{
traits::{OneSessionHandler, Randomness},
weights::Weight,
};
use pezframe_system::limits::BlockWeights;
use pezkuwi_primitives::{BlockNumber, ConsensusLog, SessionIndex, ValidatorId};
use scale_info::TypeInfo;
#[cfg(test)]
mod tests;
#[cfg(feature = "runtime-benchmarks")]
mod benchmarking;
pub use pezpallet::*;
#[derive(Clone)]
pub struct SessionChangeNotification<BlockNumber> {
pub validators: Vec<ValidatorId>,
pub queued: Vec<ValidatorId>,
pub prev_config: HostConfiguration<BlockNumber>,
pub new_config: HostConfiguration<BlockNumber>,
pub random_seed: [u8; 32],
pub session_index: SessionIndex,
}
pub trait OnNewSession<N> {
fn on_new_session(notification: &SessionChangeNotification<N>);
}
impl<N> OnNewSession<N> for () {
fn on_new_session(_: &SessionChangeNotification<N>) {}
}
pub type ValidatorSetCount = u32;
impl<BlockNumber: Default + From<u32>> Default for SessionChangeNotification<BlockNumber> {
fn default() -> Self {
Self {
validators: Vec::new(),
queued: Vec::new(),
prev_config: HostConfiguration::default(),
new_config: HostConfiguration::default(),
random_seed: Default::default(),
session_index: Default::default(),
}
}
}
#[derive(Encode, Decode, TypeInfo)]
pub(crate) struct BufferedSessionChange {
pub validators: Vec<ValidatorId>,
pub queued: Vec<ValidatorId>,
pub session_index: SessionIndex,
}
pub trait WeightInfo {
fn force_approve(d: u32) -> Weight;
}
impl WeightInfo for () {
fn force_approve(_: u32) -> Weight {
BlockWeights::default().max_block
}
}
#[pezframe_support::pezpallet]
pub mod pezpallet {
use super::*;
use pezframe_support::pezpallet_prelude::*;
use pezframe_system::pezpallet_prelude::*;
#[pezpallet::pezpallet]
#[pezpallet::without_storage_info]
pub struct Pezpallet<T>(_);
#[pezpallet::config]
pub trait Config:
pezframe_system::Config
+ configuration::Config
+ shared::Config
+ paras::Config
+ scheduler::Config
+ inclusion::Config
+ session_info::Config
+ disputes::Config
+ dmp::Config
+ hrmp::Config
{
type Randomness: Randomness<Self::Hash, BlockNumberFor<Self>>;
type ForceOrigin: EnsureOrigin<<Self as pezframe_system::Config>::RuntimeOrigin>;
type CoretimeOnNewSession: OnNewSession<BlockNumberFor<Self>>;
type WeightInfo: WeightInfo;
}
#[pezpallet::storage]
pub(super) type HasInitialized<T: Config> = StorageValue<_, ()>;
#[pezpallet::storage]
pub(crate) type BufferedSessionChanges<T: Config> =
StorageValue<_, Vec<BufferedSessionChange>, ValueQuery>;
#[pezpallet::hooks]
impl<T: Config> Hooks<BlockNumberFor<T>> for Pezpallet<T> {
fn on_initialize(now: BlockNumberFor<T>) -> Weight {
let total_weight = configuration::Pezpallet::<T>::initializer_initialize(now)
+ shared::Pezpallet::<T>::initializer_initialize(now)
+ paras::Pezpallet::<T>::initializer_initialize(now)
+ scheduler::Pezpallet::<T>::initializer_initialize(now)
+ inclusion::Pezpallet::<T>::initializer_initialize(now)
+ session_info::Pezpallet::<T>::initializer_initialize(now)
+ T::DisputesHandler::initializer_initialize(now)
+ T::SlashingHandler::initializer_initialize(now)
+ dmp::Pezpallet::<T>::initializer_initialize(now)
+ hrmp::Pezpallet::<T>::initializer_initialize(now);
HasInitialized::<T>::set(Some(()));
total_weight
}
fn on_finalize(now: BlockNumberFor<T>) {
hrmp::Pezpallet::<T>::initializer_finalize();
dmp::Pezpallet::<T>::initializer_finalize();
T::SlashingHandler::initializer_finalize();
T::DisputesHandler::initializer_finalize();
session_info::Pezpallet::<T>::initializer_finalize();
inclusion::Pezpallet::<T>::initializer_finalize();
scheduler::Pezpallet::<T>::initializer_finalize();
paras::Pezpallet::<T>::initializer_finalize(now);
shared::Pezpallet::<T>::initializer_finalize();
configuration::Pezpallet::<T>::initializer_finalize();
if let Some(BufferedSessionChange { session_index, validators, queued }) =
BufferedSessionChanges::<T>::take().pop()
{
Self::apply_new_session(session_index, validators, queued);
}
HasInitialized::<T>::take();
}
}
#[pezpallet::call]
impl<T: Config> Pezpallet<T> {
#[pezpallet::call_index(0)]
#[pezpallet::weight((
<T as Config>::WeightInfo::force_approve(
pezframe_system::Pezpallet::<T>::digest().logs.len() as u32,
),
DispatchClass::Operational,
))]
pub fn force_approve(origin: OriginFor<T>, up_to: BlockNumber) -> DispatchResult {
T::ForceOrigin::ensure_origin(origin)?;
pezframe_system::Pezpallet::<T>::deposit_log(ConsensusLog::ForceApprove(up_to).into());
Ok(())
}
}
}
impl<T: Config> Pezpallet<T> {
fn apply_new_session(
session_index: SessionIndex,
all_validators: Vec<ValidatorId>,
queued: Vec<ValidatorId>,
) {
let random_seed = {
let mut buf = [0u8; 32];
let (random_hash, _) = T::Randomness::random(&b"paras"[..]);
let len = core::cmp::min(32, random_hash.as_ref().len());
buf[..len].copy_from_slice(&random_hash.as_ref()[..len]);
buf
};
let configuration::SessionChangeOutcome { prev_config, new_config } =
configuration::Pezpallet::<T>::initializer_on_new_session(&session_index);
let new_config = new_config.unwrap_or_else(|| prev_config.clone());
let validators = shared::Pezpallet::<T>::initializer_on_new_session(
session_index,
random_seed,
&new_config,
all_validators,
);
let notification = SessionChangeNotification {
validators,
queued,
prev_config,
new_config,
random_seed,
session_index,
};
let outgoing_paras = paras::Pezpallet::<T>::initializer_on_new_session(¬ification);
scheduler::Pezpallet::<T>::initializer_on_new_session(¬ification);
inclusion::Pezpallet::<T>::initializer_on_new_session(¬ification, &outgoing_paras);
session_info::Pezpallet::<T>::initializer_on_new_session(¬ification);
T::DisputesHandler::initializer_on_new_session(¬ification);
T::SlashingHandler::initializer_on_new_session(session_index);
dmp::Pezpallet::<T>::initializer_on_new_session(¬ification, &outgoing_paras);
hrmp::Pezpallet::<T>::initializer_on_new_session(¬ification, &outgoing_paras);
T::CoretimeOnNewSession::on_new_session(¬ification);
}
fn on_new_session<'a, I: 'a>(
_changed: bool,
session_index: SessionIndex,
validators: I,
queued: Option<I>,
) where
I: Iterator<Item = (&'a T::AccountId, ValidatorId)>,
{
let validators: Vec<_> = validators.map(|(_, v)| v).collect();
let queued: Vec<_> = if let Some(queued) = queued {
queued.map(|(_, v)| v).collect()
} else {
validators.clone()
};
if session_index == 0 {
Self::apply_new_session(0, validators, queued);
} else {
BufferedSessionChanges::<T>::mutate(|v| {
v.push(BufferedSessionChange { validators, queued, session_index })
});
}
}
#[cfg(any(test, feature = "runtime-benchmarks"))]
pub(crate) fn test_trigger_on_new_session<'a, I: 'a>(
changed: bool,
session_index: SessionIndex,
validators: I,
queued: Option<I>,
) where
I: Iterator<Item = (&'a T::AccountId, ValidatorId)>,
{
Self::on_new_session(changed, session_index, validators, queued)
}
pub(crate) fn upcoming_session_change() -> bool {
!BufferedSessionChanges::<T>::get().is_empty()
}
}
impl<T: Config> pezsp_runtime::BoundToRuntimeAppPublic for Pezpallet<T> {
type Public = ValidatorId;
}
impl<T: pezpallet_session::Config + Config> OneSessionHandler<T::AccountId> for Pezpallet<T> {
type Key = ValidatorId;
fn on_genesis_session<'a, I: 'a>(validators: I)
where
I: Iterator<Item = (&'a T::AccountId, Self::Key)>,
{
Pezpallet::<T>::on_new_session(false, 0, validators, None);
}
fn on_new_session<'a, I: 'a>(changed: bool, validators: I, queued: I)
where
I: Iterator<Item = (&'a T::AccountId, Self::Key)>,
{
let session_index = pezpallet_session::Pezpallet::<T>::current_index();
Pezpallet::<T>::on_new_session(changed, session_index, validators, Some(queued));
}
fn on_disabled(_i: u32) {}
}