Files
pezkuwi-subxt/substrate/frame/migrations/src/mock_helpers.rs
T
Juan Girini bcb4d137c9 [doc] Example MBM pallet (#2119)
## Basic example showcasing a migration using the MBM framework

This PR has been built on top of
https://github.com/paritytech/polkadot-sdk/pull/1781 and adds two new
example crates to the `examples` pallet

### Changes Made:

Added the `pallet-example-mbm` crate: This crate provides a minimal
example of a pallet that uses MBM. It showcases a storage migration
where values are migrated from a `u32` to a `u64`.

---------

Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io>
Co-authored-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io>
Co-authored-by: Liam Aharon <liam.aharon@hotmail.com>
2024-04-04 11:47:24 +00:00

143 lines
4.0 KiB
Rust

// This file is part of Substrate.
// Copyright (C) Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: Apache-2.0
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//! Test helpers for internal and external usage.
#![allow(missing_docs)]
use codec::{Decode, Encode};
use frame_support::{
migrations::*,
weights::{Weight, WeightMeter},
};
use sp_core::ConstU32;
use sp_runtime::BoundedVec;
use sp_std::{vec, vec::Vec};
/// Opaque identifier of a migration.
pub type MockedIdentifier = BoundedVec<u8, ConstU32<256>>;
/// How a mocked migration should behave.
#[derive(Debug, Clone, Copy, Encode, Decode)]
pub enum MockedMigrationKind {
/// Succeed after its number of steps elapsed.
SucceedAfter,
/// Fail after its number of steps elapsed.
FailAfter,
/// Never terminate.
TimeoutAfter,
/// Cause an [`SteppedMigrationError::InsufficientWeight`] error after its number of steps
/// elapsed.
HighWeightAfter(Weight),
}
use MockedMigrationKind::*; // C style
/// Creates a migration identifier with a specific `kind` and `steps`.
pub fn mocked_id(kind: MockedMigrationKind, steps: u32) -> MockedIdentifier {
(b"MockedMigration", kind, steps).encode().try_into().unwrap()
}
frame_support::parameter_types! {
/// The configs for the migrations to run.
storage MIGRATIONS: Vec<(MockedMigrationKind, u32)> = vec![];
}
/// Allows to set the migrations to run at runtime instead of compile-time.
///
/// It achieves this by using the storage to store the migrations to run.
pub struct MockedMigrations;
impl SteppedMigrations for MockedMigrations {
fn len() -> u32 {
MIGRATIONS::get().len() as u32
}
fn nth_id(n: u32) -> Option<Vec<u8>> {
let k = MIGRATIONS::get().get(n as usize).copied();
k.map(|(kind, steps)| mocked_id(kind, steps).into_inner())
}
fn nth_step(
n: u32,
cursor: Option<Vec<u8>>,
_meter: &mut WeightMeter,
) -> Option<Result<Option<Vec<u8>>, SteppedMigrationError>> {
let (kind, steps) = MIGRATIONS::get()[n as usize];
let mut count: u32 =
cursor.as_ref().and_then(|c| Decode::decode(&mut &c[..]).ok()).unwrap_or(0);
log::debug!("MockedMigration: Step {count} vs max {steps}");
if count != steps || matches!(kind, TimeoutAfter) {
count += 1;
return Some(Ok(Some(count.encode())))
}
Some(match kind {
SucceedAfter => {
log::debug!("MockedMigration: Succeeded after {} steps", count);
Ok(None)
},
HighWeightAfter(required) => {
log::debug!("MockedMigration: Not enough weight after {} steps", count);
Err(SteppedMigrationError::InsufficientWeight { required })
},
FailAfter => {
log::debug!("MockedMigration: Failed after {} steps", count);
Err(SteppedMigrationError::Failed)
},
TimeoutAfter => unreachable!(),
})
}
fn nth_transactional_step(
n: u32,
cursor: Option<Vec<u8>>,
meter: &mut WeightMeter,
) -> Option<Result<Option<Vec<u8>>, SteppedMigrationError>> {
// This is a hack but should be fine. We don't need it in testing.
Self::nth_step(n, cursor, meter)
}
fn nth_max_steps(n: u32) -> Option<Option<u32>> {
MIGRATIONS::get().get(n as usize).map(|(_, s)| Some(*s))
}
fn cursor_max_encoded_len() -> usize {
65_536
}
fn identifier_max_encoded_len() -> usize {
256
}
}
impl MockedMigrations {
/// Set the migrations to run.
pub fn set(migrations: Vec<(MockedMigrationKind, u32)>) {
MIGRATIONS::set(&migrations);
}
}
impl crate::MockedMigrations for MockedMigrations {
fn set_fail_after(steps: u32) {
MIGRATIONS::set(&vec![(FailAfter, steps)]);
}
fn set_success_after(steps: u32) {
MIGRATIONS::set(&vec![(SucceedAfter, steps)]);
}
}