Cleanup XCMP QueueConfigData (#2142)

Removes obsolete fields from the `QueueConfigData` structure. For the
remaining fields, if they use the old defaults, we replace them with the
new defaults.

Resolves: https://github.com/paritytech/polkadot-sdk/issues/1795
This commit is contained in:
Serban Iorga
2023-12-04 13:16:20 +01:00
committed by GitHub
parent 35c39c9608
commit 1266de3919
9 changed files with 311 additions and 133 deletions
+2 -22
View File
@@ -60,7 +60,7 @@ use cumulus_primitives_core::{
use frame_support::{ use frame_support::{
defensive, defensive_assert, defensive, defensive_assert,
traits::{EnqueueMessage, EnsureOrigin, Get, QueueFootprint, QueuePausedQuery}, traits::{EnqueueMessage, EnsureOrigin, Get, QueueFootprint, QueuePausedQuery},
weights::{constants::WEIGHT_REF_TIME_PER_MILLIS, Weight, WeightMeter}, weights::{Weight, WeightMeter},
BoundedVec, BoundedVec,
}; };
use pallet_message_queue::OnQueueChanged; use pallet_message_queue::OnQueueChanged;
@@ -255,7 +255,7 @@ pub mod pallet {
return meter.consumed() return meter.consumed()
} }
migration::lazy_migrate_inbound_queue::<T>(); migration::v3::lazy_migrate_inbound_queue::<T>();
meter.consumed() meter.consumed()
} }
@@ -387,36 +387,16 @@ pub struct QueueConfigData {
/// The number of pages which the queue must be reduced to before it signals that /// The number of pages which the queue must be reduced to before it signals that
/// message sending may recommence after it has been suspended. /// message sending may recommence after it has been suspended.
resume_threshold: u32, resume_threshold: u32,
/// UNUSED - The amount of remaining weight under which we stop processing messages.
#[deprecated(note = "Will be removed")]
threshold_weight: Weight,
/// UNUSED - The speed to which the available weight approaches the maximum weight. A lower
/// number results in a faster progression. A value of 1 makes the entire weight available
/// initially.
#[deprecated(note = "Will be removed")]
weight_restrict_decay: Weight,
/// UNUSED - The maximum amount of weight any individual message may consume. Messages above
/// this weight go into the overweight queue and may only be serviced explicitly.
#[deprecated(note = "Will be removed")]
xcmp_max_individual_weight: Weight,
} }
impl Default for QueueConfigData { impl Default for QueueConfigData {
fn default() -> Self { fn default() -> Self {
// NOTE that these default values are only used on genesis. They should give a rough idea of // NOTE that these default values are only used on genesis. They should give a rough idea of
// what to set these values to, but is in no way a requirement. // what to set these values to, but is in no way a requirement.
#![allow(deprecated)]
Self { Self {
drop_threshold: 48, // 64KiB * 48 = 3MiB drop_threshold: 48, // 64KiB * 48 = 3MiB
suspend_threshold: 32, // 64KiB * 32 = 2MiB suspend_threshold: 32, // 64KiB * 32 = 2MiB
resume_threshold: 8, // 64KiB * 8 = 512KiB resume_threshold: 8, // 64KiB * 8 = 512KiB
// unused:
threshold_weight: Weight::from_parts(100_000, 0),
weight_restrict_decay: Weight::from_parts(2, 0),
xcmp_max_individual_weight: Weight::from_parts(
20u64 * WEIGHT_REF_TIME_PER_MILLIS,
DEFAULT_POV_SIZE,
),
} }
} }
} }
+279 -108
View File
@@ -16,7 +16,7 @@
//! A module that is responsible for migration of storage. //! A module that is responsible for migration of storage.
use crate::{Config, OverweightIndex, Pallet, ParaId, QueueConfig, DEFAULT_POV_SIZE}; use crate::{Config, OverweightIndex, Pallet, QueueConfig, QueueConfigData, DEFAULT_POV_SIZE};
use cumulus_primitives_core::XcmpMessageFormat; use cumulus_primitives_core::XcmpMessageFormat;
use frame_support::{ use frame_support::{
pallet_prelude::*, pallet_prelude::*,
@@ -25,37 +25,17 @@ use frame_support::{
}; };
/// The current storage version. /// The current storage version.
pub const STORAGE_VERSION: StorageVersion = StorageVersion::new(3); pub const STORAGE_VERSION: StorageVersion = StorageVersion::new(4);
pub const LOG: &str = "runtime::xcmp-queue-migration"; pub const LOG: &str = "runtime::xcmp-queue-migration";
/// Migrates the pallet storage to the most recent version.
pub struct MigrationToV3<T: Config>(PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for MigrationToV3<T> {
fn on_runtime_upgrade() -> Weight {
let mut weight = T::DbWeight::get().reads(1);
if StorageVersion::get::<Pallet<T>>() == 1 {
weight.saturating_accrue(migrate_to_v2::<T>());
StorageVersion::new(2).put::<Pallet<T>>();
weight.saturating_accrue(T::DbWeight::get().writes(1));
}
if StorageVersion::get::<Pallet<T>>() == 2 {
weight.saturating_accrue(migrate_to_v3::<T>());
StorageVersion::new(3).put::<Pallet<T>>();
weight.saturating_accrue(T::DbWeight::get().writes(1));
}
weight
}
}
mod v1 { mod v1 {
use super::*; use super::*;
use codec::{Decode, Encode}; use codec::{Decode, Encode};
#[frame_support::storage_alias]
pub(crate) type QueueConfig<T: Config> = StorageValue<Pallet<T>, QueueConfigData, ValueQuery>;
#[derive(Encode, Decode, Debug)] #[derive(Encode, Decode, Debug)]
pub struct QueueConfigData { pub struct QueueConfigData {
pub suspend_threshold: u32, pub suspend_threshold: u32,
@@ -80,6 +60,84 @@ mod v1 {
} }
} }
pub mod v2 {
use super::*;
#[frame_support::storage_alias]
pub(crate) type QueueConfig<T: Config> = StorageValue<Pallet<T>, QueueConfigData, ValueQuery>;
#[derive(Copy, Clone, Eq, PartialEq, Encode, Decode, RuntimeDebug, TypeInfo)]
pub struct QueueConfigData {
pub suspend_threshold: u32,
pub drop_threshold: u32,
pub resume_threshold: u32,
pub threshold_weight: Weight,
pub weight_restrict_decay: Weight,
pub xcmp_max_individual_weight: Weight,
}
impl Default for QueueConfigData {
fn default() -> Self {
Self {
suspend_threshold: 2,
drop_threshold: 5,
resume_threshold: 1,
threshold_weight: Weight::from_parts(100_000, 0),
weight_restrict_decay: Weight::from_parts(2, 0),
xcmp_max_individual_weight: Weight::from_parts(
20u64 * WEIGHT_REF_TIME_PER_MILLIS,
DEFAULT_POV_SIZE,
),
}
}
}
/// Migrates `QueueConfigData` from v1 (using only reference time weights) to v2 (with
/// 2D weights).
pub struct UncheckedMigrationToV2<T: Config>(PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for UncheckedMigrationToV2<T> {
#[allow(deprecated)]
fn on_runtime_upgrade() -> Weight {
let translate = |pre: v1::QueueConfigData| -> v2::QueueConfigData {
v2::QueueConfigData {
suspend_threshold: pre.suspend_threshold,
drop_threshold: pre.drop_threshold,
resume_threshold: pre.resume_threshold,
threshold_weight: Weight::from_parts(pre.threshold_weight, 0),
weight_restrict_decay: Weight::from_parts(pre.weight_restrict_decay, 0),
xcmp_max_individual_weight: Weight::from_parts(
pre.xcmp_max_individual_weight,
DEFAULT_POV_SIZE,
),
}
};
if v2::QueueConfig::<T>::translate(|pre| pre.map(translate)).is_err() {
log::error!(
target: crate::LOG_TARGET,
"unexpected error when performing translation of the QueueConfig type \
during storage upgrade to v2"
);
}
T::DbWeight::get().reads_writes(1, 1)
}
}
/// [`UncheckedMigrationToV2`] wrapped in a
/// [`VersionedMigration`](frame_support::migrations::VersionedMigration), ensuring the
/// migration is only performed when on-chain version is 1.
#[allow(dead_code)]
pub type MigrationToV2<T> = frame_support::migrations::VersionedMigration<
1,
2,
UncheckedMigrationToV2<T>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
}
pub mod v3 { pub mod v3 {
use super::*; use super::*;
use crate::*; use crate::*;
@@ -101,6 +159,10 @@ pub mod v3 {
OptionQuery, OptionQuery,
>; >;
#[frame_support::storage_alias]
pub(crate) type QueueConfig<T: Config> =
StorageValue<Pallet<T>, v2::QueueConfigData, ValueQuery>;
#[derive(Clone, Eq, PartialEq, Ord, PartialOrd, Encode, Decode, TypeInfo)] #[derive(Clone, Eq, PartialEq, Ord, PartialOrd, Encode, Decode, TypeInfo)]
pub struct InboundChannelDetails { pub struct InboundChannelDetails {
/// The `ParaId` of the parachain that this channel is connected with. /// The `ParaId` of the parachain that this channel is connected with.
@@ -121,98 +183,135 @@ pub mod v3 {
Ok, Ok,
Suspended, Suspended,
} }
}
/// Migrates `QueueConfigData` from v1 (using only reference time weights) to v2 (with /// Migrates the pallet storage to v3.
/// 2D weights). pub struct UncheckedMigrationToV3<T: Config>(PhantomData<T>);
///
/// NOTE: Only use this function if you know what you're doing. Default to using impl<T: Config> OnRuntimeUpgrade for UncheckedMigrationToV3<T> {
/// `migrate_to_latest`. fn on_runtime_upgrade() -> Weight {
#[allow(deprecated)] #[frame_support::storage_alias]
pub fn migrate_to_v2<T: Config>() -> Weight { type Overweight<T: Config> =
let translate = |pre: v1::QueueConfigData| -> super::QueueConfigData { CountedStorageMap<Pallet<T>, Twox64Concat, OverweightIndex, ParaId>;
super::QueueConfigData { let overweight_messages = Overweight::<T>::initialize_counter() as u64;
suspend_threshold: pre.suspend_threshold,
drop_threshold: pre.drop_threshold, T::DbWeight::get().reads_writes(overweight_messages, 1)
resume_threshold: pre.resume_threshold,
threshold_weight: Weight::from_parts(pre.threshold_weight, 0),
weight_restrict_decay: Weight::from_parts(pre.weight_restrict_decay, 0),
xcmp_max_individual_weight: Weight::from_parts(
pre.xcmp_max_individual_weight,
DEFAULT_POV_SIZE,
),
} }
};
if QueueConfig::<T>::translate(|pre| pre.map(translate)).is_err() {
log::error!(
target: super::LOG_TARGET,
"unexpected error when performing translation of the QueueConfig type during storage upgrade to v2"
);
} }
T::DbWeight::get().reads_writes(1, 1) /// [`UncheckedMigrationToV3`] wrapped in a
} /// [`VersionedMigration`](frame_support::migrations::VersionedMigration), ensuring the
/// migration is only performed when on-chain version is 2.
pub type MigrationToV3<T> = frame_support::migrations::VersionedMigration<
2,
3,
UncheckedMigrationToV3<T>,
Pallet<T>,
<T as frame_system::Config>::DbWeight,
>;
pub fn migrate_to_v3<T: Config>() -> Weight { pub fn lazy_migrate_inbound_queue<T: Config>() {
#[frame_support::storage_alias] let Some(mut states) = v3::InboundXcmpStatus::<T>::get() else {
type Overweight<T: Config> = log::debug!(target: LOG, "Lazy migration finished: item gone");
CountedStorageMap<Pallet<T>, Twox64Concat, OverweightIndex, ParaId>; return
let overweight_messages = Overweight::<T>::initialize_counter() as u64; };
let Some(ref mut next) = states.first_mut() else {
T::DbWeight::get().reads_writes(overweight_messages, 1) log::debug!(target: LOG, "Lazy migration finished: item empty");
} v3::InboundXcmpStatus::<T>::kill();
return
pub fn lazy_migrate_inbound_queue<T: Config>() { };
let Some(mut states) = v3::InboundXcmpStatus::<T>::get() else { log::debug!(
log::debug!(target: LOG, "Lazy migration finished: item gone"); "Migrating inbound HRMP channel with sibling {:?}, msgs left {}.",
return next.sender,
}; next.message_metadata.len()
let Some(ref mut next) = states.first_mut() else {
log::debug!(target: LOG, "Lazy migration finished: item empty");
v3::InboundXcmpStatus::<T>::kill();
return
};
log::debug!(
"Migrating inbound HRMP channel with sibling {:?}, msgs left {}.",
next.sender,
next.message_metadata.len()
);
// We take the last element since the MQ is a FIFO and we want to keep the order.
let Some((block_number, format)) = next.message_metadata.pop() else {
states.remove(0);
v3::InboundXcmpStatus::<T>::put(states);
return
};
if format != XcmpMessageFormat::ConcatenatedVersionedXcm {
log::warn!(target: LOG,
"Dropping message with format {:?} (not ConcatenatedVersionedXcm)",
format
); );
v3::InboundXcmpMessages::<T>::remove(&next.sender, &block_number); // We take the last element since the MQ is a FIFO and we want to keep the order.
let Some((block_number, format)) = next.message_metadata.pop() else {
states.remove(0);
v3::InboundXcmpStatus::<T>::put(states);
return
};
if format != XcmpMessageFormat::ConcatenatedVersionedXcm {
log::warn!(target: LOG,
"Dropping message with format {:?} (not ConcatenatedVersionedXcm)",
format
);
v3::InboundXcmpMessages::<T>::remove(&next.sender, &block_number);
v3::InboundXcmpStatus::<T>::put(states);
return
}
let Some(msg) = v3::InboundXcmpMessages::<T>::take(&next.sender, &block_number) else {
defensive!("Storage corrupted: HRMP message missing:", (next.sender, block_number));
v3::InboundXcmpStatus::<T>::put(states);
return
};
let Ok(msg): Result<BoundedVec<_, _>, _> = msg.try_into() else {
log::error!(target: LOG, "Message dropped: too big");
v3::InboundXcmpStatus::<T>::put(states);
return
};
// Finally! We have a proper message.
T::XcmpQueue::enqueue_message(msg.as_bounded_slice(), next.sender);
log::debug!(target: LOG, "Migrated HRMP message to MQ: {:?}", (next.sender, block_number));
v3::InboundXcmpStatus::<T>::put(states); v3::InboundXcmpStatus::<T>::put(states);
return }
}
pub mod v4 {
use super::*;
/// Migrates `QueueConfigData` to v4, removing deprecated fields and bumping page
/// thresholds to at least the default values.
pub struct UncheckedMigrationToV4<T: Config>(PhantomData<T>);
impl<T: Config> OnRuntimeUpgrade for UncheckedMigrationToV4<T> {
fn on_runtime_upgrade() -> Weight {
let translate = |pre: v2::QueueConfigData| -> QueueConfigData {
let pre_default = v2::QueueConfigData::default();
// If the previous values are the default ones, let's replace them with the new
// default.
if pre.suspend_threshold == pre_default.suspend_threshold &&
pre.drop_threshold == pre_default.drop_threshold &&
pre.resume_threshold == pre_default.resume_threshold
{
return QueueConfigData::default()
}
// If the previous values are not the default ones, let's leave them as they are.
QueueConfigData {
suspend_threshold: pre.suspend_threshold,
drop_threshold: pre.drop_threshold,
resume_threshold: pre.resume_threshold,
}
};
if QueueConfig::<T>::translate(|pre| pre.map(translate)).is_err() {
log::error!(
target: crate::LOG_TARGET,
"unexpected error when performing translation of the QueueConfig type \
during storage upgrade to v4"
);
}
T::DbWeight::get().reads_writes(1, 1)
}
} }
let Some(msg) = v3::InboundXcmpMessages::<T>::take(&next.sender, &block_number) else { /// [`UncheckedMigrationToV4`] wrapped in a
defensive!("Storage corrupted: HRMP message missing:", (next.sender, block_number)); /// [`VersionedMigration`](frame_support::migrations::VersionedMigration), ensuring the
v3::InboundXcmpStatus::<T>::put(states); /// migration is only performed when on-chain version is 3.
return pub type MigrationToV4<T> = frame_support::migrations::VersionedMigration<
}; 3,
4,
let Ok(msg): Result<BoundedVec<_, _>, _> = msg.try_into() else { UncheckedMigrationToV4<T>,
log::error!(target: LOG, "Message dropped: too big"); Pallet<T>,
v3::InboundXcmpStatus::<T>::put(states); <T as frame_system::Config>::DbWeight,
return >;
};
// Finally! We have a proper message.
T::XcmpQueue::enqueue_message(msg.as_bounded_slice(), next.sender);
log::debug!(target: LOG, "Migrated HRMP message to MQ: {:?}", (next.sender, block_number));
v3::InboundXcmpStatus::<T>::put(states);
} }
#[cfg(test)] #[cfg(all(feature = "try-runtime", test))]
mod tests { mod tests {
use super::*; use super::*;
use crate::mock::{new_test_ext, Test}; use crate::mock::{new_test_ext, Test};
@@ -230,14 +329,20 @@ mod tests {
}; };
new_test_ext().execute_with(|| { new_test_ext().execute_with(|| {
let storage_version = StorageVersion::new(1);
storage_version.put::<Pallet<Test>>();
frame_support::storage::unhashed::put_raw( frame_support::storage::unhashed::put_raw(
&crate::QueueConfig::<Test>::hashed_key(), &crate::QueueConfig::<Test>::hashed_key(),
&v1.encode(), &v1.encode(),
); );
migrate_to_v2::<Test>(); let bytes = v2::MigrationToV2::<Test>::pre_upgrade();
assert!(bytes.is_ok());
v2::MigrationToV2::<Test>::on_runtime_upgrade();
assert!(v2::MigrationToV2::<Test>::post_upgrade(bytes.unwrap()).is_ok());
let v2 = crate::QueueConfig::<Test>::get(); let v2 = v2::QueueConfig::<Test>::get();
assert_eq!(v1.suspend_threshold, v2.suspend_threshold); assert_eq!(v1.suspend_threshold, v2.suspend_threshold);
assert_eq!(v1.drop_threshold, v2.drop_threshold); assert_eq!(v1.drop_threshold, v2.drop_threshold);
@@ -247,4 +352,70 @@ mod tests {
assert_eq!(v1.xcmp_max_individual_weight, v2.xcmp_max_individual_weight.ref_time()); assert_eq!(v1.xcmp_max_individual_weight, v2.xcmp_max_individual_weight.ref_time());
}); });
} }
#[test]
#[allow(deprecated)]
fn test_migration_to_v4() {
new_test_ext().execute_with(|| {
let storage_version = StorageVersion::new(3);
storage_version.put::<Pallet<Test>>();
let v2 = v2::QueueConfigData {
drop_threshold: 5,
suspend_threshold: 2,
resume_threshold: 1,
..Default::default()
};
frame_support::storage::unhashed::put_raw(
&crate::QueueConfig::<Test>::hashed_key(),
&v2.encode(),
);
let bytes = v4::MigrationToV4::<Test>::pre_upgrade();
assert!(bytes.is_ok());
v4::MigrationToV4::<Test>::on_runtime_upgrade();
assert!(v4::MigrationToV4::<Test>::post_upgrade(bytes.unwrap()).is_ok());
let v4 = QueueConfig::<Test>::get();
assert_eq!(
v4,
QueueConfigData { suspend_threshold: 32, drop_threshold: 48, resume_threshold: 8 }
);
});
new_test_ext().execute_with(|| {
let storage_version = StorageVersion::new(3);
storage_version.put::<Pallet<Test>>();
let v2 = v2::QueueConfigData {
drop_threshold: 100,
suspend_threshold: 50,
resume_threshold: 40,
..Default::default()
};
frame_support::storage::unhashed::put_raw(
&crate::QueueConfig::<Test>::hashed_key(),
&v2.encode(),
);
let bytes = v4::MigrationToV4::<Test>::pre_upgrade();
assert!(bytes.is_ok());
v4::MigrationToV4::<Test>::on_runtime_upgrade();
assert!(v4::MigrationToV4::<Test>::post_upgrade(bytes.unwrap()).is_ok());
let v4 = QueueConfig::<Test>::get();
assert_eq!(
v4,
QueueConfigData {
suspend_threshold: 50,
drop_threshold: 100,
resume_threshold: 40
}
);
});
}
} }
@@ -954,8 +954,12 @@ pub type SignedExtra = (
pub type UncheckedExtrinsic = pub type UncheckedExtrinsic =
generic::UncheckedExtrinsic<Address, RuntimeCall, Signature, SignedExtra>; generic::UncheckedExtrinsic<Address, RuntimeCall, Signature, SignedExtra>;
/// Migrations to apply on runtime upgrade. /// Migrations to apply on runtime upgrade.
pub type Migrations = pub type Migrations = (
(pallet_collator_selection::migration::v1::MigrateToV1<Runtime>, InitStorageVersions); pallet_collator_selection::migration::v1::MigrateToV1<Runtime>,
InitStorageVersions,
// unreleased
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
);
/// Migration to initialize storage versions for pallets added after genesis. /// Migration to initialize storage versions for pallets added after genesis.
/// ///
@@ -941,6 +941,8 @@ pub type Migrations = (
InitStorageVersions, InitStorageVersions,
// unreleased // unreleased
DeleteUndecodableStorage, DeleteUndecodableStorage,
// unreleased
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
); );
/// Asset Hub Westend has some undecodable storage, delete it. /// Asset Hub Westend has some undecodable storage, delete it.
@@ -118,6 +118,7 @@ pub type Migrations = (
pallet_collator_selection::migration::v1::MigrateToV1<Runtime>, pallet_collator_selection::migration::v1::MigrateToV1<Runtime>,
pallet_multisig::migrations::v1::MigrateToV1<Runtime>, pallet_multisig::migrations::v1::MigrateToV1<Runtime>,
InitStorageVersions, InitStorageVersions,
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
); );
/// Migration to initialize storage versions for pallets added after genesis. /// Migration to initialize storage versions for pallets added after genesis.
@@ -118,6 +118,8 @@ pub type Migrations = (
pallet_collator_selection::migration::v1::MigrateToV1<Runtime>, pallet_collator_selection::migration::v1::MigrateToV1<Runtime>,
pallet_multisig::migrations::v1::MigrateToV1<Runtime>, pallet_multisig::migrations::v1::MigrateToV1<Runtime>,
InitStorageVersions, InitStorageVersions,
// unreleased
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
); );
/// Migration to initialize storage versions for pallets added after genesis. /// Migration to initialize storage versions for pallets added after genesis.
@@ -707,6 +707,8 @@ pub type UncheckedExtrinsic =
type Migrations = ( type Migrations = (
// unreleased // unreleased
pallet_collator_selection::migration::v1::MigrateToV1<Runtime>, pallet_collator_selection::migration::v1::MigrateToV1<Runtime>,
// unreleased
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
); );
/// Executive: handles dispatch to the various modules. /// Executive: handles dispatch to the various modules.
@@ -100,9 +100,11 @@ pub type UncheckedExtrinsic =
/// Migrations to apply on runtime upgrade. /// Migrations to apply on runtime upgrade.
pub type Migrations = ( pub type Migrations = (
cumulus_pallet_parachain_system::migration::Migration<Runtime>, cumulus_pallet_parachain_system::migration::Migration<Runtime>,
cumulus_pallet_xcmp_queue::migration::MigrationToV3<Runtime>, cumulus_pallet_xcmp_queue::migration::v2::MigrationToV2<Runtime>,
cumulus_pallet_xcmp_queue::migration::v3::MigrationToV3<Runtime>,
pallet_contracts::Migration<Runtime>, pallet_contracts::Migration<Runtime>,
// unreleased // unreleased
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<Runtime>,
); );
type EventRecord = frame_system::EventRecord< type EventRecord = frame_system::EventRecord<
+14
View File
@@ -0,0 +1,14 @@
title: Cleanup XCMP `QueueConfigData`
doc:
- audience: Parachain Dev
description: Removes obsolete fields from the `QueueConfigData` structure. For the remaining fields, if they use the old defaults, we replace them with the new defaults.
migrations:
runtime:
- pallet: "cumulus_pallet_xcmp_queue"
description: "v4: Removes obsolete fields from the `QueueConfigData` structure. For the remaining fields, if they use the old defaults, we replace them with the new defaults."
crates: []
host_functions: []