Skip to content

Commit 3a47582

Browse files
committed
Migrated pallet funding for block number provider
1 parent 4317213 commit 3a47582

File tree

4 files changed

+161
-10
lines changed

4 files changed

+161
-10
lines changed

pallets/funding/src/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,7 @@ pub use types::*;
9595
use xcm::v5::prelude::*;
9696

9797
pub mod functions;
98-
pub mod storage_migrations;
98+
pub mod migrations;
9999
pub mod traits;
100100
pub mod types;
101101
pub mod weights;
@@ -166,7 +166,7 @@ pub mod pallet {
166166
}
167167

168168
#[pallet::pallet]
169-
#[pallet::storage_version(storage_migrations::STORAGE_VERSION)]
169+
#[pallet::storage_version(migrations::STORAGE_VERSION)]
170170
pub struct Pallet<T>(_);
171171

172172
#[pallet::config]

pallets/funding/src/migrations/mod.rs

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
//! Migrations
2+
3+
use frame_support::traits::StorageVersion;
4+
5+
mod storage_migrations;
6+
mod vesting_info;
7+
8+
/// Current storage version
9+
pub const STORAGE_VERSION: StorageVersion = StorageVersion::new(7);

pallets/funding/src/storage_migrations.rs pallets/funding/src/migrations/storage_migrations.rs

+4-8
Original file line numberDiff line numberDiff line change
@@ -15,10 +15,6 @@ use alloc::vec::Vec;
1515
use polimec_common::migration_types::{MigrationInfo, ParticipationType};
1616
use xcm::v5::Location;
1717

18-
/// The current storage version
19-
pub const STORAGE_VERSION: StorageVersion = StorageVersion::new(6);
20-
pub const LOG: &str = "runtime::funding::migration";
21-
2218
pub mod v5_storage_items {
2319

2420
#[derive(Clone, Copy, Encode, Decode, Eq, PartialEq, RuntimeDebug, TypeInfo, MaxEncodedLen)]
@@ -200,11 +196,11 @@ pub mod v5_storage_items {
200196
}
201197

202198
pub mod v6 {
203-
use super::*;
204-
use crate::{
205-
storage_migrations::v5_storage_items::{OldMigration, OldProjectStatus, MAX_PARTICIPATIONS_PER_USER},
206-
EvaluationRoundInfo, ProjectDetailsOf, ProjectStatus, TicketSize,
199+
use super::{
200+
v5_storage_items::{OldMigration, OldProjectStatus, MAX_PARTICIPATIONS_PER_USER},
201+
*,
207202
};
203+
use crate::{EvaluationRoundInfo, ProjectDetailsOf, ProjectStatus, TicketSize};
208204
use frame_system::pallet_prelude::BlockNumberFor;
209205
use polimec_common::{
210206
credentials::Did,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
// storage_migrations.rs
2+
3+
use crate::Config;
4+
use alloc::vec::Vec;
5+
use frame_support::{
6+
pallet_prelude::*,
7+
traits::{StorageVersion, UncheckedOnRuntimeUpgrade},
8+
weights::Weight,
9+
};
10+
use polimec_common::migration_types::{Migration, MigrationInfo, MigrationStatus};
11+
use sp_runtime::{traits::ConstU32, WeakBoundedVec};
12+
13+
pub mod v7 {
14+
use crate::{AccountIdOf, ProjectId, UserMigrations};
15+
16+
use super::*;
17+
18+
const LOG: &str = "funding::migration::v7";
19+
pub const MAX_PARTICIPATIONS_PER_USER: u32 = 10_000;
20+
21+
pub struct UncheckedMigrationToV7<T: Config>(PhantomData<T>);
22+
impl<T: Config> UncheckedOnRuntimeUpgrade for UncheckedMigrationToV7<T> {
23+
fn on_runtime_upgrade() -> Weight {
24+
let mut items = 0u64;
25+
log::info!(target: LOG, "Starting vesting time migration to V7");
26+
27+
let translate_migration = |project_user: (ProjectId, AccountIdOf<T>),
28+
(status, migrations): (
29+
MigrationStatus,
30+
WeakBoundedVec<Migration, ConstU32<MAX_PARTICIPATIONS_PER_USER>>,
31+
)|
32+
-> Option<(
33+
MigrationStatus,
34+
WeakBoundedVec<Migration, ConstU32<MAX_PARTICIPATIONS_PER_USER>>,
35+
)> {
36+
let mut migrated = Vec::new();
37+
38+
for mut migration in migrations.into_iter() {
39+
items = items.saturating_add(1);
40+
// Halve the number of blocks while maintaining same real-time duration
41+
// Assumes block time increases from 6s to 12s
42+
if let MigrationInfo { vesting_time, contribution_token_amount } = migration.info {
43+
migration.info =
44+
MigrationInfo { vesting_time: vesting_time.saturating_div(2), contribution_token_amount };
45+
}
46+
migrated.push(migration);
47+
}
48+
49+
let bounded = WeakBoundedVec::try_from(migrated)
50+
.unwrap_or_else(|_| panic!("MaxParticipationsPerUser exceeded during migration"));
51+
52+
Some((status, bounded))
53+
};
54+
55+
UserMigrations::<T>::translate(translate_migration);
56+
57+
log::info!(target: LOG, "Migrated {} vesting entries", items);
58+
T::DbWeight::get().reads_writes(items, items)
59+
}
60+
61+
#[cfg(feature = "try-runtime")]
62+
fn pre_upgrade() -> Result<Vec<u8>, DispatchError> {
63+
let migration_count = UserMigrations::<T>::iter().count() as u32;
64+
let vesting_count: u32 = UserMigrations::<T>::iter_values()
65+
.map(|(_, migrations)| {
66+
migrations.into_iter().filter(|m| matches!(m.info, MigrationInfo { .. })).count() as u32
67+
})
68+
.sum();
69+
log::info!(target: LOG, "Pre-upgrade: {} UserMigrations entries, {} vesting entries", migration_count, vesting_count);
70+
71+
// Encode counts and sample pre-migration vesting times for validation
72+
let mut state = Vec::new();
73+
state.extend(migration_count.to_le_bytes());
74+
state.extend(vesting_count.to_le_bytes());
75+
for ((project_id, account), (_, migrations)) in UserMigrations::<T>::iter() {
76+
for migration in migrations {
77+
if let MigrationInfo::Vesting { vesting_time, .. } = migration.info {
78+
state.extend(project_id.to_le_bytes());
79+
state.extend(T::AccountId32Conversion::convert(account.clone()));
80+
state.extend(vesting_time.to_le_bytes());
81+
break; // Sample one per entry for simplicity
82+
}
83+
}
84+
}
85+
Ok(state)
86+
}
87+
88+
#[cfg(feature = "try-runtime")]
89+
fn post_upgrade(pre_state: Vec<u8>) -> Result<(), DispatchError> {
90+
let pre_migration_count = u32::from_le_bytes(pre_state[0..4].try_into().unwrap());
91+
let pre_vesting_count = u32::from_le_bytes(pre_state[4..8].try_into().unwrap());
92+
let post_migration_count = UserMigrations::<T>::iter().count() as u32;
93+
let post_vesting_count: u32 = UserMigrations::<T>::iter_values()
94+
.map(|(_, migrations)| {
95+
migrations.into_iter().filter(|m| matches!(m.info, MigrationInfo { .. })).count() as u32
96+
})
97+
.sum();
98+
99+
assert_eq!(pre_migration_count, post_migration_count, "UserMigrations count mismatch",);
100+
assert_eq!(pre_vesting_count, post_vesting_count, "Vesting entries count mismatch",);
101+
102+
// Validate vesting times are halved
103+
let mut offset = 8;
104+
while offset + 44 < pre_state.len() {
105+
// 4 (project_id) + 32 (account) + 8 (vesting_time)
106+
let project_id = ProjectId::from_le_bytes(pre_state[offset..offset + 4].try_into().unwrap());
107+
let account_bytes: [u8; 32] = pre_state[offset + 4..offset + 36].try_into().unwrap();
108+
let account = T::AccountId32Conversion::convert_back(account_bytes);
109+
let pre_vesting_time = u64::from_le_bytes(pre_state[offset + 36..offset + 44].try_into().unwrap());
110+
offset += 44;
111+
112+
let (_, migrations) = UserMigrations::<T>::get((project_id, account.clone()))
113+
.ok_or_else(|| {
114+
format!("Migration entry for project {} account {:?} not found", project_id, account)
115+
})
116+
.unwrap();
117+
let post_vesting_time = migrations
118+
.into_iter()
119+
.find_map(
120+
|m| {
121+
if let MigrationInfo { vesting_time, .. } = m.info {
122+
Some(vesting_time)
123+
} else {
124+
None
125+
}
126+
},
127+
)
128+
.ok_or_else(|| format!("Vesting info not found for project {} account {:?}", project_id, account))
129+
.unwrap();
130+
131+
assert_eq!(post_vesting_time, pre_vesting_time / 2, "Vesting time not halved",);
132+
}
133+
134+
log::info!(target: LOG, "Post-upgrade checks passed: {} migrations, {} vesting entries", post_migration_count, post_vesting_count);
135+
Ok(())
136+
}
137+
}
138+
139+
pub type MigrationToV7<T> = frame_support::migrations::VersionedMigration<
140+
6,
141+
7,
142+
UncheckedMigrationToV7<T>,
143+
crate::Pallet<T>,
144+
<T as frame_system::Config>::DbWeight,
145+
>;
146+
}

0 commit comments

Comments
 (0)