1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
// This file is part of Astar.
// Copyright (C) Stake Technologies Pte.Ltd.
// SPDX-License-Identifier: GPL-3.0-or-later
// Astar is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// Astar is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with Astar. If not, see <http://www.gnu.org/licenses/>.
use super::*;
use core::marker::PhantomData;
use frame_support::{
migration::clear_storage_prefix,
migrations::{MigrationId, SteppedMigration, SteppedMigrationError},
traits::{GetStorageVersion, OnRuntimeUpgrade, UncheckedOnRuntimeUpgrade},
weights::WeightMeter,
};
#[cfg(feature = "try-runtime")]
use sp_std::vec::Vec;
#[cfg(feature = "try-runtime")]
use sp_runtime::TryRuntimeError;
/// Exports for versioned migration `type`s for this pallet.
pub mod versioned_migrations {
use super::*;
/// Migration V8 to V9 wrapped in a [`frame_support::migrations::VersionedMigration`], ensuring
/// the migration is only performed when on-chain version is 8.
pub type V8ToV9<T, InitArgs> = frame_support::migrations::VersionedMigration<
8,
9,
v9::VersionMigrateV8ToV9<T, InitArgs>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
}
mod v9 {
use super::*;
use frame_support::DefaultNoBound;
#[derive(
Encode,
Decode,
MaxEncodedLen,
RuntimeDebugNoBound,
PartialEqNoBound,
DefaultNoBound,
EqNoBound,
CloneNoBound,
TypeInfo,
)]
#[scale_info(skip_type_params(NT))]
pub struct TierParametersV8<NT: Get<u32>> {
/// Reward distribution per tier, in percentage.
/// First entry refers to the first tier, and so on.
/// The sum of all values must not exceed 100%.
/// In case it is less, portion of rewards will never be distributed.
pub(crate) reward_portion: BoundedVec<Permill, NT>,
/// Distribution of number of slots per tier, in percentage.
/// First entry refers to the first tier, and so on.
/// The sum of all values must not exceed 100%.
/// In case it is less, slot capacity will never be fully filled.
pub(crate) slot_distribution: BoundedVec<Permill, NT>,
/// Requirements for entry into each tier.
/// First entry refers to the first tier, and so on.
pub(crate) tier_thresholds: BoundedVec<TierThreshold, NT>,
}
// The loyal staker flag is updated to `u8` with the new MaxBonusSafeMovesPerPeriod from config
// for all already existing StakerInfo.
pub struct VersionMigrateV8ToV9<T, InitArgs>(PhantomData<(T, InitArgs)>);
impl<T: Config, InitArgs: Get<(u64, u64)>> UncheckedOnRuntimeUpgrade
for VersionMigrateV8ToV9<T, InitArgs>
{
fn on_runtime_upgrade() -> Weight {
let result = StaticTierParams::<T>::translate::<TierParametersV8<T::NumberOfTiers>, _>(
|maybe_old_params| match maybe_old_params {
Some(old_params) => Some(TierParameters {
slot_distribution: old_params.slot_distribution,
reward_portion: old_params.reward_portion,
tier_thresholds: old_params.tier_thresholds,
slot_number_args: InitArgs::get(),
}),
_ => None,
},
);
if result.is_err() {
log::error!("Failed to translate StaticTierParams from previous V8 type to current V9 type.");
// Enable maintenance mode.
ActiveProtocolState::<T>::mutate(|state| {
state.maintenance = true;
});
log::warn!("Maintenance mode enabled.");
return T::DbWeight::get().reads_writes(2, 1);
}
T::DbWeight::get().reads_writes(1, 1)
}
#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, TryRuntimeError> {
assert!(
!ActiveProtocolState::<T>::get().maintenance,
"Maintenance mode must be disabled before the runtime upgrade."
);
Ok(Vec::new())
}
#[cfg(feature = "try-runtime")]
fn post_upgrade(_: Vec<u8>) -> Result<(), TryRuntimeError> {
assert_eq!(
Pallet::<T>::on_chain_storage_version(),
EXPECTED_PALLET_DAPP_STAKING_VERSION,
"dapp-staking::migration::v9: wrong storage version"
);
assert!(
!ActiveProtocolState::<T>::get().maintenance,
"Maintenance mode must be disabled after the successful runtime upgrade."
);
let new_tier_params = StaticTierParams::<T>::get();
assert!(
new_tier_params.is_valid(),
"New tier params are invalid, re-check the values!"
);
assert_eq!(new_tier_params.slot_number_args, InitArgs::get());
Ok(())
}
}
}
const PALLET_MIGRATIONS_ID: &[u8; 16] = b"dapp-staking-mbm";
pub struct LazyMigration<T, W: WeightInfo>(PhantomData<(T, W)>);
impl<T: Config, W: WeightInfo> SteppedMigration for LazyMigration<T, W> {
type Cursor = <T as frame_system::Config>::AccountId;
// Without the explicit length here the construction of the ID would not be infallible.
type Identifier = MigrationId<16>;
/// The identifier of this migration. Which should be globally unique.
fn id() -> Self::Identifier {
MigrationId {
pallet_id: *PALLET_MIGRATIONS_ID,
version_from: 0,
version_to: 1,
}
}
fn step(
mut cursor: Option<Self::Cursor>,
meter: &mut WeightMeter,
) -> Result<Option<Self::Cursor>, SteppedMigrationError> {
let required = W::step();
// If there is not enough weight for a single step, return an error. This case can be
// problematic if it is the first migration that ran in this block. But there is nothing
// that we can do about it here.
if meter.remaining().any_lt(required) {
return Err(SteppedMigrationError::InsufficientWeight { required });
}
let mut count = 0u32;
let mut migrated = 0u32;
let current_block_number =
frame_system::Pallet::<T>::block_number().saturated_into::<u32>();
// We loop here to do as much progress as possible per step.
loop {
if meter.try_consume(required).is_err() {
break;
}
let mut iter = if let Some(last_key) = cursor {
// If a cursor is provided, start iterating from the stored value
// corresponding to the last key processed in the previous step.
// Note that this only works if the old and the new map use the same way to hash
// storage keys.
Ledger::<T>::iter_from(Ledger::<T>::hashed_key_for(last_key))
} else {
// If no cursor is provided, start iterating from the beginning.
Ledger::<T>::iter()
};
// If there's a next item in the iterator, perform the migration.
if let Some((ref last_key, mut ledger)) = iter.next() {
// inc count
count.saturating_inc();
if ledger.unlocking.is_empty() {
// no unlocking for this account, nothing to update
// Return the processed key as the new cursor.
cursor = Some(last_key.clone());
continue;
}
for chunk in ledger.unlocking.iter_mut() {
if current_block_number >= chunk.unlock_block {
continue; // chunk already unlocked
}
let remaining_blocks = chunk.unlock_block.saturating_sub(current_block_number);
chunk.unlock_block.saturating_accrue(remaining_blocks);
}
// Override ledger
Ledger::<T>::insert(last_key, ledger);
// inc migrated
migrated.saturating_inc();
// Return the processed key as the new cursor.
cursor = Some(last_key.clone())
} else {
// Signal that the migration is complete (no more items to process).
cursor = None;
break;
}
}
log::info!(target: LOG_TARGET, "🚚 iterated {count} entries, migrated {migrated}");
Ok(cursor)
}
}
/// Double the remaining block for next era start
pub struct AdjustEraMigration<T>(PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for AdjustEraMigration<T> {
fn on_runtime_upgrade() -> Weight {
log::info!("🚚 migrated to async backing, adjust next era start");
ActiveProtocolState::<T>::mutate_exists(|maybe| {
if let Some(state) = maybe {
let current_block_number =
frame_system::Pallet::<T>::block_number().saturated_into::<u32>();
let remaining = state.next_era_start.saturating_sub(current_block_number);
state.next_era_start.saturating_accrue(remaining);
}
});
T::DbWeight::get().reads_writes(1, 1)
}
}
pub const EXPECTED_PALLET_DAPP_STAKING_VERSION: u16 = 9;
pub struct DappStakingCleanupMigration<T>(PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for DappStakingCleanupMigration<T> {
fn on_runtime_upgrade() -> Weight {
let dapp_staking_storage_version =
<Pallet<T> as GetStorageVersion>::on_chain_storage_version();
if dapp_staking_storage_version != EXPECTED_PALLET_DAPP_STAKING_VERSION {
log::info!("Aborting migration due to unexpected on-chain storage versions for pallet-dapp-staking: {:?}. Expectation was: {:?}.", dapp_staking_storage_version, EXPECTED_PALLET_DAPP_STAKING_VERSION);
return T::DbWeight::get().reads(1);
}
let pallet_prefix: &[u8] = b"DappStaking";
let result =
clear_storage_prefix(pallet_prefix, b"ActiveBonusUpdateState", &[], None, None);
log::info!(
"cleanup dAppStaking migration result: {:?}",
result.deconstruct()
);
T::DbWeight::get().reads_writes(1, 1)
}
}