Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MBM try-runtime support #4251

Merged
merged 49 commits into from
Sep 25, 2024
Merged
Show file tree
Hide file tree
Changes from 36 commits
Commits
Show all changes
49 commits
Select commit Hold shift + click to select a range
7ec7cc9
wip mbm try-runtime
liamaharon Apr 23, 2024
1c8416d
Merge branch 'master' into liam-mbm-try-runtime
liamaharon Apr 23, 2024
623c2b8
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 21, 2024
675338a
update cargo.lock
liamaharon May 21, 2024
d36ce71
refactor try-runtime logic
liamaharon May 27, 2024
45ce6e4
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 27, 2024
1403cbc
taplo
liamaharon May 27, 2024
f5b9210
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 28, 2024
92053e0
wip upgrade check select
liamaharon May 28, 2024
311d994
Merge branch 'liam-mbm-try-runtime' of github.com:paritytech/polkadot…
liamaharon May 28, 2024
e0a8152
impl_for_tuples MockedMigrations
liamaharon May 28, 2024
bf3fbfb
wip revert upgradecheckselect changes
liamaharon May 29, 2024
279e4cf
revert upgradecheckselect changes
liamaharon May 29, 2024
1710410
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 29, 2024
38dd259
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 29, 2024
1a8e86c
use Vec
liamaharon May 29, 2024
b10534f
Merge branch 'liam-mbm-try-runtime' of github.com:paritytech/polkadot…
liamaharon May 29, 2024
30121c5
prdoc
liamaharon May 29, 2024
b1ccef1
address comments
liamaharon May 30, 2024
6fbe843
address comment
liamaharon May 30, 2024
1677cd6
address comment
liamaharon May 30, 2024
1a61af3
fix newlines
liamaharon May 30, 2024
f57b50a
clippy
liamaharon May 30, 2024
95b7969
Merge branch 'master' into liam-mbm-try-runtime
liamaharon May 30, 2024
7dcad83
Merge branch 'master' into liam-mbm-try-runtime
liamaharon Jun 3, 2024
8fa87f2
Merge branch 'master' into liam-mbm-try-runtime
liamaharon Jun 3, 2024
49cd4e1
Merge branch 'master' into liam-mbm-try-runtime
liamaharon Jun 4, 2024
028aa60
ci: test without try-runtime feature
liamaharon Jun 7, 2024
e01c9e4
Merge branch 'liam-mbm-try-runtime' of github.com:paritytech/polkadot…
liamaharon Jun 7, 2024
b38ff1a
Merge branch 'master' into liam-mbm-try-runtime
liamaharon Jun 7, 2024
06eb247
cfg_attr
liamaharon Jun 7, 2024
96860b3
Merge branch 'liam-mbm-try-runtime' of github.com:paritytech/polkadot…
liamaharon Jun 7, 2024
b9bf013
Merge remote-tracking branch 'origin/master' into liam-mbm-try-runtime
ggwpez Sep 11, 2024
ba768dc
Cleanup
ggwpez Sep 12, 2024
97a1cbd
replace sp-std
claravanstaden Sep 12, 2024
f610140
adds external crate alloc
claravanstaden Sep 12, 2024
1c969d8
Cleanup
ggwpez Sep 12, 2024
ad1811a
cleanup
ggwpez Sep 12, 2024
666db60
Add PrDoc (auto generated)
ggwpez Sep 12, 2024
e437426
review
ggwpez Sep 12, 2024
00de16a
prdoc
ggwpez Sep 12, 2024
6e0539f
prdoc
ggwpez Sep 12, 2024
f26c07b
Fix clippy
georgepisaltu Sep 12, 2024
475a976
Add missing migration to people-rococo
ggwpez Sep 16, 2024
ea00e35
Derive PartialEq on some enums
ggwpez Sep 16, 2024
7c597b6
Merge branch 'master' into liam-mbm-try-runtime
ggwpez Sep 16, 2024
fbda868
Merge branch 'master' into liam-mbm-try-runtime
ggwpez Sep 24, 2024
d6ff835
Merge branch 'master' into liam-mbm-try-runtime
ggwpez Sep 24, 2024
9113c2b
Merge branch 'master' into liam-mbm-try-runtime
ggwpez Sep 25, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 24 additions & 0 deletions .gitlab/pipeline/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,30 @@ test-linux-stable-codecov:
codecovcli -v do-upload -f target/coverage/result/report-${CI_NODE_INDEX}.lcov --disable-search -t ${CODECOV_TOKEN} -r paritytech/polkadot-sdk --commit-sha ${CI_COMMIT_SHA} --fail-on-error --git-service github;
fi

# some tests do not run with `try-runtime` feature enabled
# https://github.com/paritytech/polkadot-sdk/pull/4251#discussion_r1624282143
test-linux-stable-no-try-runtime:
stage: test
extends:
- .docker-env
- .common-refs
- .run-immediately
- .pipeline-stopper-artifacts
variables:
RUST_TOOLCHAIN: stable
# Enable debug assertions since we are running optimized builds for testing
# but still want to have debug assertions.
RUSTFLAGS: "-Cdebug-assertions=y -Dwarnings"
script:
- >
time cargo nextest run \
--workspace \
--locked \
--release \
--no-fail-fast \
--cargo-quiet \
--features experimental,riscv,ci-only-tests

test-doc:
stage: test
extends:
Expand Down
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

21 changes: 21 additions & 0 deletions prdoc/pr_4251.prdoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
# Schema: Polkadot SDK PRDoc Schema (prdoc) v1.0.0
# See doc at https://raw.githubusercontent.com/paritytech/polkadot-sdk/master/prdoc/schema_user.json

title: MBM try-runtime support

doc:
- audience: Runtime Dev
description: |
- Adds pre_upgrade post_upgrade hook usage to MBM example
- Adds try-runtime gated methods pre_upgrade and post_upgrade on SteppedMigration
- Adds try-runtime gated methods nth_pre_upgrade and nth_post_upgrade on SteppedMigrations
- Modifies pallet_migrations implementation to run pre_upgrade and post_upgrade steps at the appropriate times, and panic in the event of migration failure
- Derives `impl_for_tuples` for `MockedMigrations`

crates:
- name: pallet-example-mbm
bump: minor
- name: pallet-migrations
bump: minor
- name: frame-support
bump: minor
1 change: 1 addition & 0 deletions substrate/frame/examples/multi-block-migrations/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ frame-benchmarking = { optional = true, workspace = true }
log = { workspace = true }
scale-info = { workspace = true }
sp-io = { workspace = true }
sp-std = { workspace = true }
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should be removed, right?


[features]
default = ["std"]
Expand Down
2 changes: 2 additions & 0 deletions substrate/frame/examples/multi-block-migrations/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,8 @@
pub mod migrations;
mod mock;

extern crate alloc;

pub use pallet::*;

#[frame_support::pallet]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,12 @@ use frame_support::{
weights::WeightMeter,
};

#[cfg(feature = "try-runtime")]
use alloc::collections::btree_map::BTreeMap;

#[cfg(feature = "try-runtime")]
use alloc::vec::Vec;

mod benchmarks;
mod tests;
pub mod weights;
Expand Down Expand Up @@ -115,4 +121,38 @@ impl<T: Config, W: weights::WeightInfo> SteppedMigration for LazyMigrationV1<T,
}
Ok(cursor)
}

#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, frame_support::sp_runtime::TryRuntimeError> {
use codec::Encode;

// Return the state of the storage before the migration.
Ok(v0::MyMap::<T>::iter().collect::<BTreeMap<_, _>>().encode())
}

#[cfg(feature = "try-runtime")]
fn post_upgrade(prev: Vec<u8>) -> Result<(), frame_support::sp_runtime::TryRuntimeError> {
use codec::Decode;

// Check the state of the storage after the migration.
let prev_map = BTreeMap::<u32, u32>::decode(&mut &prev[..])
.expect("Failed to decode the previous storage state");

// Check the len of prev and post are the same.
assert_eq!(
MyMap::<T>::iter().count(),
prev_map.len(),
"Migration failed: the number of items in the storage after the migration is not the same as before"
);

for (key, value) in prev_map {
let value = MyMap::<T>::get(key).expect("Failed to get the value after the migration");
assert_eq!(
value, value as u64,
"Migration failed: the value after the migration is not the same as before"
);
}

Ok(())
}
}
1 change: 1 addition & 0 deletions substrate/frame/migrations/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ targets = ["x86_64-unknown-linux-gnu"]

[dependencies]
codec = { features = ["derive"], workspace = true }
cfg-if = { workspace = true }
docify = { workspace = true }
impl-trait-for-tuples = { workspace = true }
log = { workspace = true, default-features = true }
Expand Down
71 changes: 64 additions & 7 deletions substrate/frame/migrations/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -70,21 +70,26 @@
//! points to the currently active migration and stores its inner cursor. The inner cursor can then
//! be used by the migration to store its inner state and advance. Each time when the migration
//! returns `Some(cursor)`, it signals the pallet that it is not done yet.
liamaharon marked this conversation as resolved.
Show resolved Hide resolved
//!
//! The cursor is reset on each runtime upgrade. This ensures that it starts to execute at the
//! first migration in the vector. The pallets cursor is only ever incremented or set to `Stuck`
//! once it encounters an error (Goal 4). Once in the stuck state, the pallet will stay stuck until
//! it is fixed through manual governance intervention.
//!
//! As soon as the cursor of the pallet becomes `Some(_)`; [`MultiStepMigrator::ongoing`] returns
//! `true` (Goal 2). This can be used by upstream code to possibly pause transactions.
//! In `on_initialize` the pallet will load the current migration and check whether it was already
//! executed in the past by checking for membership of its ID in the [`Historic`] set. Historic
//! migrations are skipped without causing an error. Each successfully executed migration is added
//! to this set (Goal 5).
//!
//! This proceeds until no more migrations remain. At that point, the event `UpgradeCompleted` is
//! emitted (Goal 1).
//!
//! The execution of each migration happens by calling [`SteppedMigration::transactional_step`].
//! This function wraps the inner `step` function into a transactional layer to allow rollback in
//! the error case (Goal 6).
//!
//! Weight limits must be checked by the migration itself. The pallet provides a [`WeightMeter`] for
//! that purpose. The pallet may return [`SteppedMigrationError::InsufficientWeight`] at any point.
//! In that scenario, one of two things will happen: if that migration was exclusively executed
Expand Down Expand Up @@ -156,11 +161,15 @@ use core::ops::ControlFlow;
use frame_support::{
defensive, defensive_assert,
migrations::*,
pallet_prelude::*,
traits::Get,
weights::{Weight, WeightMeter},
BoundedVec,
};
use frame_system::{pallet_prelude::BlockNumberFor, Pallet as System};
use frame_system::{
pallet_prelude::{BlockNumberFor, *},
Pallet as System,
};
use sp_runtime::Saturating;

/// Points to the next migration to execute.
Expand Down Expand Up @@ -262,18 +271,39 @@ pub type IdentifierOf<T> = BoundedVec<u8, <T as Config>::IdentifierMaxLen>;
pub type ActiveCursorOf<T> = ActiveCursor<RawCursorOf<T>, BlockNumberFor<T>>;

/// Trait for a tuple of No-OP migrations with one element.
#[impl_trait_for_tuples::impl_for_tuples(30)]
pub trait MockedMigrations: SteppedMigrations {
/// The migration should fail after `n` steps.
fn set_fail_after(n: u32);
/// The migration should succeed after `n` steps.
fn set_success_after(n: u32);
}

#[cfg(feature = "try-runtime")]
/// Wrapper for pre-upgrade bytes, allowing us to impl MEL on it.
liamaharon marked this conversation as resolved.
Show resolved Hide resolved
///
/// For `try-runtime` testing only.
#[derive(Debug, Clone, Eq, PartialEq, Encode, Decode, scale_info::TypeInfo, Default)]
struct PreUpgradeBytesWrapper(pub Vec<u8>);

#[cfg(feature = "try-runtime")]
impl MaxEncodedLen for PreUpgradeBytesWrapper {
fn max_encoded_len() -> usize {
0
}
}

/// Data stored by the pre-upgrade hook of the MBMs. Only used for `try-runtime` testing.
///
/// Define this outside of the pallet so it is not confused with actual storage.
#[cfg(feature = "try-runtime")]
#[frame_support::storage_alias]
type PreUpgradeBytes<T: Config> =
StorageMap<Pallet<T>, Twox64Concat, IdentifierOf<T>, PreUpgradeBytesWrapper, ValueQuery>;

#[frame_support::pallet]
pub mod pallet {
use super::*;
use frame_support::pallet_prelude::*;
use frame_system::pallet_prelude::*;

#[pallet::pallet]
pub struct Pallet<T>(_);
Expand Down Expand Up @@ -700,6 +730,16 @@ impl<T: Config> Pallet<T> {
}

let max_steps = T::Migrations::nth_max_steps(cursor.index);

// If this is the first time running this migration, exec the pre-upgrade hook.
ggwpez marked this conversation as resolved.
Show resolved Hide resolved
#[cfg(feature = "try-runtime")]
if !PreUpgradeBytes::<T>::contains_key(&bounded_id) {
let bytes = T::Migrations::nth_pre_upgrade(cursor.index)
.expect("Invalid cursor.index")
.expect("Pre-upgrade failed");
PreUpgradeBytes::<T>::insert(&bounded_id, PreUpgradeBytesWrapper(bytes));
Copy link
Member

@ggwpez ggwpez Sep 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This part is a bit ugly - since it puts the intermediary state into storage. But i dont think there is a spot where we can put intermediary data that is persisted between blocks.
The threadlocal stuff gets reset after each block i think...

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's ok, it's just try-runtime storage.

}

let next_cursor = T::Migrations::nth_transactional_step(
cursor.index,
cursor.inner_cursor.clone().map(|c| c.into_inner()),
Expand Down Expand Up @@ -734,6 +774,16 @@ impl<T: Config> Pallet<T> {
},
Ok(None) => {
// A migration is done when it returns cursor `None`.

// Run post-upgrade checks.
#[cfg(feature = "try-runtime")]
T::Migrations::nth_post_upgrade(
cursor.index,
PreUpgradeBytes::<T>::get(&bounded_id).0,
)
.expect("Invalid cursor.index.")
.expect("Post-upgrade failed.");

Self::deposit_event(Event::MigrationCompleted { index: cursor.index, took });
Historic::<T>::insert(&bounded_id, ());
cursor.goto_next_migration(System::<T>::block_number());
Expand All @@ -758,14 +808,21 @@ impl<T: Config> Pallet<T> {
}

/// Fail the current runtime upgrade, caused by `migration`.
///
/// When the `try-runtime` feature is enabled, this function will panic.
// Allow unreachable code so it can compile without warnings when `try-runtime` is enabled.
fn upgrade_failed(migration: Option<u32>) {
use FailedMigrationHandling::*;
Self::deposit_event(Event::UpgradeFailed);

match T::FailedMigrationHandler::failed(migration) {
KeepStuck => Cursor::<T>::set(Some(MigrationCursor::Stuck)),
ForceUnstuck => Cursor::<T>::kill(),
Ignore => {},
if cfg!(feature = "try-runtime") {
panic!("Migration with index {:?} failed.", migration);
} else {
match T::FailedMigrationHandler::failed(migration) {
KeepStuck => Cursor::<T>::set(Some(MigrationCursor::Stuck)),
ForceUnstuck => Cursor::<T>::kill(),
Ignore => {},
}
}
}

Expand Down
33 changes: 33 additions & 0 deletions substrate/frame/migrations/src/mock_helpers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,12 @@ pub enum MockedMigrationKind {
/// Cause an [`SteppedMigrationError::InsufficientWeight`] error after its number of steps
/// elapsed.
HighWeightAfter(Weight),
/// PreUpgrade should fail.
#[cfg(feature = "try-runtime")]
PreUpgradeFail,
/// PostUpgrade should fail.
#[cfg(feature = "try-runtime")]
PostUpgradeFail,
}
use MockedMigrationKind::*; // C style

Expand Down Expand Up @@ -99,6 +105,8 @@ impl SteppedMigrations for MockedMigrations {
Err(SteppedMigrationError::Failed)
},
TimeoutAfter => unreachable!(),
#[cfg(feature = "try-runtime")]
PreUpgradeFail | PostUpgradeFail => Ok(None),
})
}

Expand All @@ -115,6 +123,31 @@ impl SteppedMigrations for MockedMigrations {
MIGRATIONS::get().get(n as usize).map(|(_, s)| Some(*s))
}

#[cfg(feature = "try-runtime")]
fn nth_pre_upgrade(n: u32) -> Option<Result<Vec<u8>, sp_runtime::TryRuntimeError>> {
let (kind, _) = MIGRATIONS::get()[n as usize];

if let PreUpgradeFail = kind {
return Some(Err("Some pre-upgrade error".into()))
}

Some(Ok(vec![]))
}

#[cfg(feature = "try-runtime")]
fn nth_post_upgrade(
n: u32,
_state: Vec<u8>,
) -> Option<Result<(), sp_runtime::TryRuntimeError>> {
let (kind, _) = MIGRATIONS::get()[n as usize];

if let PostUpgradeFail = kind {
return Some(Err("Some post-upgrade error".into()))
}

Some(Ok(()))
}

fn cursor_max_encoded_len() -> usize {
65_536
}
Expand Down
Loading
Loading