mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-12 21:41:12 +00:00
6ac8537436
* move migrate sequence to config * remove commented out code * Update frame/contracts/src/lib.rs Co-authored-by: PG Herveou <pgherveou@gmail.com> * remove Migrations generic * make runtime use noop migrations * restrict is_upgrade_supported * Update contract multi-block migration Ensure that we do as many steps as possible given the weight limit passed to on_idle * undo is_upgrade_supported change * Update bin/node/runtime/src/lib.rs Co-authored-by: PG Herveou <pgherveou@gmail.com> * wip * fix comment (#14316) * fix test * fix * Update frame/contracts/src/migration.rs Co-authored-by: Juan <juangirini@gmail.com> * fix test doc * Apply suggestions from code review Co-authored-by: Sasha Gryaznov <hi@agryaznov.com> * Fix compilation with feature runtime-benchmarks * fix example * fix cargo doc --document-private-items * private links * Remove dup comment * add doc for MigrationInProgress * PR review remove duplicate asserts * simplify upper bound * fix link * typo * typo * no unwrap() * correct log message * missing * fix typo * PR comment * Add example with single element tuple * Improve migration message * Update frame/contracts/src/benchmarking/mod.rs Co-authored-by: Sasha Gryaznov <hi@agryaznov.com> * Update frame/contracts/src/migration.rs Co-authored-by: Sasha Gryaznov <hi@agryaznov.com> * Update frame/contracts/src/migration.rs Co-authored-by: Sasha Gryaznov <hi@agryaznov.com> * use saturating_accrue instead of += * add more doc * Contracts: Better migration types (#14418) * Add explicit error, if try-runtime runs a noop migration * use mut remaining_weight --------- Co-authored-by: Juan Girini <juangirini@gmail.com> Co-authored-by: Sasha Gryaznov <hi@agryaznov.com>
145 lines
4.4 KiB
Rust
145 lines
4.4 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.
|
|
|
|
//! Update `CodeStorage` with the new `determinism` field.
|
|
|
|
use crate::{
|
|
migration::{IsFinished, MigrationStep},
|
|
weights::WeightInfo,
|
|
CodeHash, Config, Determinism, Pallet, Weight, LOG_TARGET,
|
|
};
|
|
use codec::{Decode, Encode};
|
|
use frame_support::{codec, pallet_prelude::*, storage_alias, DefaultNoBound, Identity};
|
|
#[cfg(feature = "try-runtime")]
|
|
use sp_runtime::TryRuntimeError;
|
|
use sp_std::prelude::*;
|
|
|
|
mod old {
|
|
use super::*;
|
|
|
|
#[derive(Encode, Decode)]
|
|
pub struct PrefabWasmModule {
|
|
#[codec(compact)]
|
|
pub instruction_weights_version: u32,
|
|
#[codec(compact)]
|
|
pub initial: u32,
|
|
#[codec(compact)]
|
|
pub maximum: u32,
|
|
pub code: Vec<u8>,
|
|
}
|
|
|
|
#[storage_alias]
|
|
pub type CodeStorage<T: Config> =
|
|
StorageMap<Pallet<T>, Identity, CodeHash<T>, PrefabWasmModule>;
|
|
}
|
|
|
|
#[cfg(feature = "runtime-benchmarks")]
|
|
pub fn store_old_dummy_code<T: Config>(len: usize) {
|
|
use sp_runtime::traits::Hash;
|
|
let module = old::PrefabWasmModule {
|
|
instruction_weights_version: 0,
|
|
initial: 0,
|
|
maximum: 0,
|
|
code: vec![42u8; len],
|
|
};
|
|
let hash = T::Hashing::hash(&module.code);
|
|
old::CodeStorage::<T>::insert(hash, module);
|
|
}
|
|
|
|
#[derive(Encode, Decode)]
|
|
struct PrefabWasmModule {
|
|
#[codec(compact)]
|
|
pub instruction_weights_version: u32,
|
|
#[codec(compact)]
|
|
pub initial: u32,
|
|
#[codec(compact)]
|
|
pub maximum: u32,
|
|
pub code: Vec<u8>,
|
|
pub determinism: Determinism,
|
|
}
|
|
|
|
#[storage_alias]
|
|
type CodeStorage<T: Config> = StorageMap<Pallet<T>, Identity, CodeHash<T>, PrefabWasmModule>;
|
|
|
|
#[derive(Encode, Decode, MaxEncodedLen, DefaultNoBound)]
|
|
pub struct Migration<T: Config> {
|
|
last_code_hash: Option<CodeHash<T>>,
|
|
}
|
|
|
|
impl<T: Config> MigrationStep for Migration<T> {
|
|
const VERSION: u16 = 9;
|
|
|
|
fn max_step_weight() -> Weight {
|
|
T::WeightInfo::v9_migration_step(T::MaxCodeLen::get())
|
|
}
|
|
|
|
fn step(&mut self) -> (IsFinished, Weight) {
|
|
let mut iter = if let Some(last_key) = self.last_code_hash.take() {
|
|
old::CodeStorage::<T>::iter_from(old::CodeStorage::<T>::hashed_key_for(last_key))
|
|
} else {
|
|
old::CodeStorage::<T>::iter()
|
|
};
|
|
|
|
if let Some((key, old)) = iter.next() {
|
|
log::debug!(target: LOG_TARGET, "Migrating contract code {:?}", key);
|
|
let len = old.code.len() as u32;
|
|
let module = PrefabWasmModule {
|
|
instruction_weights_version: old.instruction_weights_version,
|
|
initial: old.initial,
|
|
maximum: old.maximum,
|
|
code: old.code,
|
|
determinism: Determinism::Enforced,
|
|
};
|
|
CodeStorage::<T>::insert(key, module);
|
|
self.last_code_hash = Some(key);
|
|
(IsFinished::No, T::WeightInfo::v9_migration_step(len))
|
|
} else {
|
|
log::debug!(target: LOG_TARGET, "No more contracts code to migrate");
|
|
(IsFinished::Yes, T::WeightInfo::v9_migration_step(0))
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn pre_upgrade_step() -> Result<Vec<u8>, TryRuntimeError> {
|
|
let sample: Vec<_> = old::CodeStorage::<T>::iter().take(100).collect();
|
|
|
|
log::debug!(target: LOG_TARGET, "Taking sample of {} contract codes", sample.len());
|
|
Ok(sample.encode())
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn post_upgrade_step(state: Vec<u8>) -> Result<(), TryRuntimeError> {
|
|
let sample = <Vec<(CodeHash<T>, old::PrefabWasmModule)> as Decode>::decode(&mut &state[..])
|
|
.expect("pre_upgrade_step provides a valid state; qed");
|
|
|
|
log::debug!(target: LOG_TARGET, "Validating sample of {} contract codes", sample.len());
|
|
for (code_hash, old) in sample {
|
|
let module = CodeStorage::<T>::get(&code_hash).unwrap();
|
|
ensure!(
|
|
module.instruction_weights_version == old.instruction_weights_version,
|
|
"invalid isntruction weights version"
|
|
);
|
|
ensure!(module.determinism == Determinism::Enforced, "invalid determinism");
|
|
ensure!(module.initial == old.initial, "invalid initial");
|
|
ensure!(module.maximum == old.maximum, "invalid maximum");
|
|
ensure!(module.code == old.code, "invalid code");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|