Skip to content

Commit ba2441f

Browse files
committed
Migrated pallet funding for block number provider
1 parent e07c1a9 commit ba2441f

File tree

4 files changed

+172
-6
lines changed

4 files changed

+172
-6
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

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
//! Migrations
2+
3+
mod storage_migrations;
4+
mod vesting_info;

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

+4-4
Original file line numberDiff line numberDiff line change
@@ -200,11 +200,11 @@ pub mod v5_storage_items {
200200
}
201201

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

0 commit comments

Comments
 (0)