use super::*;
use frame_support::{
migrations::VersionedMigration,
pallet_prelude::ValueQuery,
storage_alias,
traits::{GetStorageVersion, OnRuntimeUpgrade, UncheckedOnRuntimeUpgrade},
Twox64Concat,
};
#[cfg(feature = "try-runtime")]
use frame_support::ensure;
#[cfg(feature = "try-runtime")]
use sp_runtime::TryRuntimeError;
#[derive(Encode, Decode, Clone, Copy, PartialEq, Eq, RuntimeDebug, TypeInfo, MaxEncodedLen)]
enum ObsoleteReleases {
V5_0_0, V6_0_0, V7_0_0, V8_0_0, V9_0_0, V10_0_0, V11_0_0, V12_0_0, }
impl Default for ObsoleteReleases {
fn default() -> Self {
ObsoleteReleases::V12_0_0
}
}
#[storage_alias]
type StorageVersion<T: Config> = StorageValue<Pallet<T>, ObsoleteReleases, ValueQuery>;
pub mod v17 {
use super::*;
#[derive(Encode, Decode, TypeInfo, MaxEncodedLen)]
struct OldUnappliedSlash<T: Config> {
validator: T::AccountId,
own: BalanceOf<T>,
others: Vec<(T::AccountId, BalanceOf<T>)>,
reporters: Vec<T::AccountId>,
payout: BalanceOf<T>,
}
#[frame_support::storage_alias]
pub type OldUnappliedSlashes<T: Config> =
StorageMap<Pallet<T>, Twox64Concat, EraIndex, Vec<OldUnappliedSlash<T>>, ValueQuery>;
#[frame_support::storage_alias]
pub type DisabledValidators<T: Config> =
StorageValue<Pallet<T>, BoundedVec<(u32, OffenceSeverity), ConstU32<100>>, ValueQuery>;
pub struct VersionUncheckedMigrateV16ToV17<T>(core::marker::PhantomData<T>);
impl<T: Config> UncheckedOnRuntimeUpgrade for VersionUncheckedMigrateV16ToV17<T> {
fn on_runtime_upgrade() -> Weight {
let mut weight: Weight = Weight::zero();
OldUnappliedSlashes::<T>::drain().for_each(|(era, slashes)| {
weight.saturating_accrue(T::DbWeight::get().reads(1));
for slash in slashes {
let validator = slash.validator.clone();
let new_slash = UnappliedSlash {
validator: validator.clone(),
own: slash.own,
others: WeakBoundedVec::force_from(slash.others, None),
payout: slash.payout,
reporter: slash.reporters.first().cloned(),
};
let slash_key = (validator, Perbill::from_percent(99), 9999);
UnappliedSlashes::<T>::insert(era, slash_key, new_slash);
weight.saturating_accrue(T::DbWeight::get().writes(1));
}
});
weight
}
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, sp_runtime::TryRuntimeError> {
let mut expected_slashes: u32 = 0;
OldUnappliedSlashes::<T>::iter().for_each(|(_, slashes)| {
expected_slashes += slashes.len() as u32;
});
Ok(expected_slashes.encode())
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(state: Vec<u8>) -> Result<(), TryRuntimeError> {
let expected_slash_count =
u32::decode(&mut state.as_slice()).expect("Failed to decode state");
let actual_slash_count = UnappliedSlashes::<T>::iter().count() as u32;
ensure!(expected_slash_count == actual_slash_count, "Slash count mismatch");
Ok(())
}
}
pub type MigrateV16ToV17<T> = VersionedMigration<
16,
17,
VersionUncheckedMigrateV16ToV17<T>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
pub struct MigrateDisabledToSession<T>(core::marker::PhantomData<T>);
impl<T: Config> pallet_session::migrations::v1::MigrateDisabledValidators
for MigrateDisabledToSession<T>
{
#[cfg(feature = "try-runtime")]
fn peek_disabled() -> Vec<(u32, OffenceSeverity)> {
DisabledValidators::<T>::get().into()
}
fn take_disabled() -> Vec<(u32, OffenceSeverity)> {
DisabledValidators::<T>::take().into()
}
}
}
pub mod v16 {
use super::*;
use frame_support::Twox64Concat;
use sp_staking::offence::OffenceSeverity;
#[frame_support::storage_alias]
pub(crate) type Invulnerables<T: Config> =
StorageValue<Pallet<T>, Vec<<T as frame_system::Config>::AccountId>, ValueQuery>;
#[frame_support::storage_alias]
pub(crate) type DisabledValidators<T: Config> =
StorageValue<Pallet<T>, Vec<(u32, OffenceSeverity)>, ValueQuery>;
#[frame_support::storage_alias]
pub(crate) type ErasStakers<T: Config> = StorageDoubleMap<
Pallet<T>,
Twox64Concat,
EraIndex,
Twox64Concat,
<T as frame_system::Config>::AccountId,
Exposure<<T as frame_system::Config>::AccountId, BalanceOf<T>>,
ValueQuery,
>;
#[frame_support::storage_alias]
pub(crate) type ErasStakersClipped<T: Config> = StorageDoubleMap<
Pallet<T>,
Twox64Concat,
EraIndex,
Twox64Concat,
<T as frame_system::Config>::AccountId,
Exposure<<T as frame_system::Config>::AccountId, BalanceOf<T>>,
ValueQuery,
>;
pub struct VersionUncheckedMigrateV15ToV16<T>(core::marker::PhantomData<T>);
impl<T: Config> UncheckedOnRuntimeUpgrade for VersionUncheckedMigrateV15ToV16<T> {
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, sp_runtime::TryRuntimeError> {
let old_disabled_validators = v15::DisabledValidators::<T>::get();
Ok(old_disabled_validators.encode())
}
fn on_runtime_upgrade() -> Weight {
let max_offence = OffenceSeverity(Perbill::from_percent(100));
let migrated = v15::DisabledValidators::<T>::take()
.into_iter()
.map(|v| (v, max_offence))
.collect::<Vec<_>>();
v16::DisabledValidators::<T>::set(migrated);
log!(info, "v16 applied successfully.");
T::DbWeight::get().reads_writes(1, 1)
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(state: Vec<u8>) -> Result<(), TryRuntimeError> {
let old_disabled_validators =
Vec::<u32>::decode(&mut state.as_slice()).expect("Failed to decode state");
let new_disabled_validators = v17::DisabledValidators::<T>::get();
frame_support::ensure!(
old_disabled_validators.len() == new_disabled_validators.len(),
"DisabledValidators length mismatch"
);
let new_disabled_validators =
new_disabled_validators.into_iter().map(|(v, _)| v).collect::<Vec<_>>();
frame_support::ensure!(
old_disabled_validators == new_disabled_validators,
"DisabledValidator ids mismatch"
);
let max_severity = OffenceSeverity(Perbill::from_percent(100));
let new_disabled_validators = v17::DisabledValidators::<T>::get();
for (_, severity) in new_disabled_validators {
frame_support::ensure!(severity == max_severity, "Severity mismatch");
}
Ok(())
}
}
pub type MigrateV15ToV16<T> = VersionedMigration<
15,
16,
VersionUncheckedMigrateV15ToV16<T>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
}
pub mod v15 {
use super::*;
type DefaultDisablingStrategy = pallet_session::disabling::UpToLimitDisablingStrategy;
#[storage_alias]
pub(crate) type DisabledValidators<T: Config> = StorageValue<Pallet<T>, Vec<u32>, ValueQuery>;
pub struct VersionUncheckedMigrateV14ToV15<T>(core::marker::PhantomData<T>);
impl<T: Config> UncheckedOnRuntimeUpgrade for VersionUncheckedMigrateV14ToV15<T> {
fn on_runtime_upgrade() -> Weight {
let mut migrated = v14::OffendingValidators::<T>::take()
.into_iter()
.filter(|p| p.1) .map(|p| p.0)
.collect::<Vec<_>>();
migrated.truncate(DefaultDisablingStrategy::disable_limit(
T::SessionInterface::validators().len(),
));
DisabledValidators::<T>::set(migrated);
log!(info, "v15 applied successfully.");
T::DbWeight::get().reads_writes(1, 1)
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_state: Vec<u8>) -> Result<(), TryRuntimeError> {
frame_support::ensure!(
v14::OffendingValidators::<T>::decode_len().is_none(),
"OffendingValidators is not empty after the migration"
);
Ok(())
}
}
pub type MigrateV14ToV15<T> = VersionedMigration<
14,
15,
VersionUncheckedMigrateV14ToV15<T>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
}
pub mod v14 {
use super::*;
#[frame_support::storage_alias]
pub(crate) type OffendingValidators<T: Config> =
StorageValue<Pallet<T>, Vec<(u32, bool)>, ValueQuery>;
pub struct MigrateToV14<T>(core::marker::PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for MigrateToV14<T> {
fn on_runtime_upgrade() -> Weight {
let in_code = Pallet::<T>::in_code_storage_version();
let on_chain = Pallet::<T>::on_chain_storage_version();
if in_code == 14 && on_chain == 13 {
in_code.put::<Pallet<T>>();
log!(info, "staking v14 applied successfully.");
T::DbWeight::get().reads_writes(1, 1)
} else {
log!(warn, "staking v14 not applied.");
T::DbWeight::get().reads(1)
}
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_state: Vec<u8>) -> Result<(), TryRuntimeError> {
frame_support::ensure!(
Pallet::<T>::on_chain_storage_version() >= 14,
"v14 not applied"
);
Ok(())
}
}
}
pub mod v13 {
use super::*;
pub struct MigrateToV13<T>(core::marker::PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for MigrateToV13<T> {
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, TryRuntimeError> {
frame_support::ensure!(
StorageVersion::<T>::get() == ObsoleteReleases::V12_0_0,
"Required v12 before upgrading to v13"
);
Ok(Default::default())
}
fn on_runtime_upgrade() -> Weight {
let in_code = Pallet::<T>::in_code_storage_version();
let onchain = StorageVersion::<T>::get();
if in_code == 13 && onchain == ObsoleteReleases::V12_0_0 {
StorageVersion::<T>::kill();
in_code.put::<Pallet<T>>();
log!(info, "v13 applied successfully");
T::DbWeight::get().reads_writes(1, 2)
} else {
log!(warn, "Skipping v13, should be removed");
T::DbWeight::get().reads(1)
}
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_state: Vec<u8>) -> Result<(), TryRuntimeError> {
frame_support::ensure!(
Pallet::<T>::on_chain_storage_version() == 13,
"v13 not applied"
);
frame_support::ensure!(
!StorageVersion::<T>::exists(),
"Storage version not migrated correctly"
);
Ok(())
}
}
}
pub mod v12 {
use super::*;
use frame_support::{pallet_prelude::ValueQuery, storage_alias};
#[storage_alias]
type HistoryDepth<T: Config> = StorageValue<Pallet<T>, u32, ValueQuery>;
pub struct MigrateToV12<T>(core::marker::PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for MigrateToV12<T> {
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, TryRuntimeError> {
frame_support::ensure!(
StorageVersion::<T>::get() == ObsoleteReleases::V11_0_0,
"Expected v11 before upgrading to v12"
);
if HistoryDepth::<T>::exists() {
frame_support::ensure!(
T::HistoryDepth::get() == HistoryDepth::<T>::get(),
"Provided value of HistoryDepth should be same as the existing storage value"
);
} else {
log::info!("No HistoryDepth in storage; nothing to remove");
}
Ok(Default::default())
}
fn on_runtime_upgrade() -> frame_support::weights::Weight {
if StorageVersion::<T>::get() == ObsoleteReleases::V11_0_0 {
HistoryDepth::<T>::kill();
StorageVersion::<T>::put(ObsoleteReleases::V12_0_0);
log!(info, "v12 applied successfully");
T::DbWeight::get().reads_writes(1, 2)
} else {
log!(warn, "Skipping v12, should be removed");
T::DbWeight::get().reads(1)
}
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_state: Vec<u8>) -> Result<(), TryRuntimeError> {
frame_support::ensure!(
StorageVersion::<T>::get() == ObsoleteReleases::V12_0_0,
"v12 not applied"
);
Ok(())
}
}
}
pub mod v11 {
use super::*;
use frame_support::{
storage::migration::move_pallet,
traits::{GetStorageVersion, PalletInfoAccess},
};
#[cfg(feature = "try-runtime")]
use sp_io::hashing::twox_128;
pub struct MigrateToV11<T, P, N>(core::marker::PhantomData<(T, P, N)>);
impl<T: Config, P: GetStorageVersion + PalletInfoAccess, N: Get<&'static str>> OnRuntimeUpgrade
for MigrateToV11<T, P, N>
{
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, TryRuntimeError> {
frame_support::ensure!(
StorageVersion::<T>::get() == ObsoleteReleases::V10_0_0,
"must upgrade linearly"
);
let old_pallet_prefix = twox_128(N::get().as_bytes());
frame_support::ensure!(
sp_io::storage::next_key(&old_pallet_prefix).is_some(),
"no data for the old pallet name has been detected"
);
Ok(Default::default())
}
fn on_runtime_upgrade() -> Weight {
let old_pallet_name = N::get();
let new_pallet_name = <P as PalletInfoAccess>::name();
if StorageVersion::<T>::get() == ObsoleteReleases::V10_0_0 {
StorageVersion::<T>::put(ObsoleteReleases::V11_0_0);
if new_pallet_name == old_pallet_name {
log!(
warn,
"new bags-list name is equal to the old one, only bumping the version"
);
return T::DbWeight::get().reads(1).saturating_add(T::DbWeight::get().writes(1))
}
move_pallet(old_pallet_name.as_bytes(), new_pallet_name.as_bytes());
<T as frame_system::Config>::BlockWeights::get().max_block
} else {
log!(warn, "v11::migrate should be removed.");
T::DbWeight::get().reads(1)
}
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_state: Vec<u8>) -> Result<(), TryRuntimeError> {
frame_support::ensure!(
StorageVersion::<T>::get() == ObsoleteReleases::V11_0_0,
"wrong version after the upgrade"
);
let old_pallet_name = N::get();
let new_pallet_name = <P as PalletInfoAccess>::name();
if new_pallet_name == old_pallet_name {
return Ok(())
}
let old_pallet_prefix = twox_128(N::get().as_bytes());
frame_support::ensure!(
sp_io::storage::next_key(&old_pallet_prefix).is_none(),
"old pallet data hasn't been removed"
);
let new_pallet_name = <P as PalletInfoAccess>::name();
let new_pallet_prefix = twox_128(new_pallet_name.as_bytes());
frame_support::ensure!(
sp_io::storage::next_key(&new_pallet_prefix).is_some(),
"new pallet data hasn't been created"
);
Ok(())
}
}
}