mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-28 07:27:55 +00:00
bcb4d137c9
## 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>
163 lines
5.2 KiB
Rust
163 lines
5.2 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.
|
|
|
|
//! Mocked runtime for testing the migrations pallet.
|
|
|
|
#![cfg(test)]
|
|
|
|
use crate::{mock_helpers::*, Event, Historic};
|
|
|
|
use frame_support::{
|
|
derive_impl,
|
|
migrations::*,
|
|
traits::{OnFinalize, OnInitialize},
|
|
weights::Weight,
|
|
};
|
|
use frame_system::EventRecord;
|
|
use sp_core::H256;
|
|
|
|
type Block = frame_system::mocking::MockBlock<Test>;
|
|
|
|
// Configure a mock runtime to test the pallet.
|
|
frame_support::construct_runtime!(
|
|
pub enum Test {
|
|
System: frame_system,
|
|
Migrations: crate,
|
|
}
|
|
);
|
|
|
|
#[derive_impl(frame_system::config_preludes::TestDefaultConfig)]
|
|
impl frame_system::Config for Test {
|
|
type Block = Block;
|
|
type PalletInfo = PalletInfo;
|
|
type MultiBlockMigrator = Migrations;
|
|
}
|
|
|
|
frame_support::parameter_types! {
|
|
pub const MaxServiceWeight: Weight = Weight::MAX.div(10);
|
|
}
|
|
|
|
#[derive_impl(crate::config_preludes::TestDefaultConfig)]
|
|
impl crate::Config for Test {
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
type Migrations = crate::mock_helpers::MockedMigrations;
|
|
#[cfg(not(feature = "runtime-benchmarks"))]
|
|
type Migrations = MockedMigrations;
|
|
type MigrationStatusHandler = MockedMigrationStatusHandler;
|
|
type FailedMigrationHandler = MockedFailedMigrationHandler;
|
|
}
|
|
|
|
frame_support::parameter_types! {
|
|
/// The number of started upgrades.
|
|
pub static UpgradesStarted: u32 = 0;
|
|
/// The number of completed upgrades.
|
|
pub static UpgradesCompleted: u32 = 0;
|
|
/// The migrations that failed.
|
|
pub static UpgradesFailed: Vec<Option<u32>> = vec![];
|
|
/// Return value of [`MockedFailedMigrationHandler::failed`].
|
|
pub static FailedUpgradeResponse: FailedMigrationHandling = FailedMigrationHandling::KeepStuck;
|
|
}
|
|
|
|
/// Records all started and completed upgrades in `UpgradesStarted` and `UpgradesCompleted`.
|
|
pub struct MockedMigrationStatusHandler;
|
|
impl MigrationStatusHandler for MockedMigrationStatusHandler {
|
|
fn started() {
|
|
log::info!("MigrationStatusHandler started");
|
|
UpgradesStarted::mutate(|v| *v += 1);
|
|
}
|
|
|
|
fn completed() {
|
|
log::info!("MigrationStatusHandler completed");
|
|
UpgradesCompleted::mutate(|v| *v += 1);
|
|
}
|
|
}
|
|
|
|
/// Records all failed upgrades in `UpgradesFailed`.
|
|
pub struct MockedFailedMigrationHandler;
|
|
impl FailedMigrationHandler for MockedFailedMigrationHandler {
|
|
fn failed(migration: Option<u32>) -> FailedMigrationHandling {
|
|
UpgradesFailed::mutate(|v| v.push(migration));
|
|
let res = FailedUpgradeResponse::get();
|
|
log::error!("FailedMigrationHandler failed at: {migration:?}, handling as {res:?}");
|
|
res
|
|
}
|
|
}
|
|
|
|
/// Returns the number of `(started, completed, failed)` upgrades and resets their numbers.
|
|
pub fn upgrades_started_completed_failed() -> (u32, u32, u32) {
|
|
(UpgradesStarted::take(), UpgradesCompleted::take(), UpgradesFailed::take().len() as u32)
|
|
}
|
|
|
|
/// Build genesis storage according to the mock runtime.
|
|
pub fn new_test_ext() -> sp_io::TestExternalities {
|
|
sp_io::TestExternalities::new(Default::default())
|
|
}
|
|
|
|
/// Run this closure in test externalities.
|
|
pub fn test_closure<R>(f: impl FnOnce() -> R) -> R {
|
|
let mut ext = new_test_ext();
|
|
ext.execute_with(f)
|
|
}
|
|
|
|
pub fn run_to_block(n: u32) {
|
|
while System::block_number() < n as u64 {
|
|
log::debug!("Block {}", System::block_number());
|
|
System::set_block_number(System::block_number() + 1);
|
|
System::on_initialize(System::block_number());
|
|
Migrations::on_initialize(System::block_number());
|
|
// Executive calls this:
|
|
<Migrations as MultiStepMigrator>::step();
|
|
|
|
Migrations::on_finalize(System::block_number());
|
|
System::on_finalize(System::block_number());
|
|
}
|
|
}
|
|
|
|
/// Returns the historic migrations, sorted by their identifier.
|
|
pub fn historic() -> Vec<MockedIdentifier> {
|
|
let mut historic = Historic::<Test>::iter_keys().collect::<Vec<_>>();
|
|
historic.sort();
|
|
historic
|
|
}
|
|
|
|
// Traits to make using events less insufferable:
|
|
pub trait IntoRecord {
|
|
fn into_record(self) -> EventRecord<<Test as frame_system::Config>::RuntimeEvent, H256>;
|
|
}
|
|
|
|
impl IntoRecord for Event<Test> {
|
|
fn into_record(self) -> EventRecord<<Test as frame_system::Config>::RuntimeEvent, H256> {
|
|
let re: <Test as frame_system::Config>::RuntimeEvent = self.into();
|
|
EventRecord { phase: frame_system::Phase::Initialization, event: re, topics: vec![] }
|
|
}
|
|
}
|
|
|
|
pub trait IntoRecords {
|
|
fn into_records(self) -> Vec<EventRecord<<Test as frame_system::Config>::RuntimeEvent, H256>>;
|
|
}
|
|
|
|
impl<E: IntoRecord> IntoRecords for Vec<E> {
|
|
fn into_records(self) -> Vec<EventRecord<<Test as frame_system::Config>::RuntimeEvent, H256>> {
|
|
self.into_iter().map(|e| e.into_record()).collect()
|
|
}
|
|
}
|
|
|
|
pub fn assert_events<E: IntoRecord>(events: Vec<E>) {
|
|
pretty_assertions::assert_eq!(events.into_records(), System::events());
|
|
System::reset_events();
|
|
}
|