mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-23 16:41:06 +00:00
TryDecodeEntireState check for storage types and pallets (#1805)
### This PR is a port of this [PR for substrate](https://github.com/paritytech/substrate/pull/13013) by @kianenigma Add infrastructure needed to have a Pallet::decode_entire_state(), which makes sure all "typed" storage items defined in the pallet are decode-able. This is not enforced in any way at the moment. Teams who wish to integrate/use this in the try-runtime feature flag should add frame_support::storage::migration::EnsureStateDecodes as the LAST ITEM of the runtime's custom migrations, and pass it to frame-executive. This will make it usable in try-runtime on-runtime-upgrade. This now catches cases like https://github.com/paritytech/polkadot-sdk/pull/1969: ```pre ERROR runtime::executive] failed to decode the value at key: Failed to decode value at key: 0x94eadf0156a8ad5156507773d0471e4ab8ebad86f546c7e0b135a4212aace339. Storage info StorageInfo { pallet_name: Ok("ParaScheduler"), storage_name: Ok("AvailabilityCores"), prefix: Err(Utf8Error { valid_up_to: 0, error_len: Some(1) }), max_values: Some(1), max_size: None }. Raw value: Some("0x0c010101010101") ``` ... or:  Closes #241 --------- Signed-off-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> Co-authored-by: Oliver Tale-Yazdi <oliver.tale-yazdi@parity.io> Co-authored-by: Liam Aharon <liam.aharon@hotmail.com>
This commit is contained in:
@@ -24,10 +24,10 @@
|
||||
//!
|
||||
//! This is internal api and is subject to change.
|
||||
|
||||
mod double_map;
|
||||
pub(crate) mod double_map;
|
||||
pub(crate) mod map;
|
||||
mod nmap;
|
||||
mod value;
|
||||
pub(crate) mod nmap;
|
||||
pub(crate) mod value;
|
||||
|
||||
pub use double_map::StorageDoubleMap;
|
||||
pub use map::StorageMap;
|
||||
|
||||
@@ -119,7 +119,11 @@ impl<P: CountedStorageMapInstance, H, K, V, Q, O, M> MapWrapper
|
||||
type Map = StorageMap<P, H, K, V, Q, O, M>;
|
||||
}
|
||||
|
||||
type CounterFor<P> = StorageValue<<P as CountedStorageMapInstance>::CounterPrefix, u32, ValueQuery>;
|
||||
/// The numeric counter type.
|
||||
pub type Counter = u32;
|
||||
|
||||
type CounterFor<P> =
|
||||
StorageValue<<P as CountedStorageMapInstance>::CounterPrefix, Counter, ValueQuery>;
|
||||
|
||||
/// On removal logic for updating counter while draining upon some prefix with
|
||||
/// [`crate::storage::PrefixIterator`].
|
||||
@@ -423,14 +427,14 @@ where
|
||||
/// can be very heavy, so use with caution.
|
||||
///
|
||||
/// Returns the number of items in the map which is used to set the counter.
|
||||
pub fn initialize_counter() -> u32 {
|
||||
let count = Self::iter_values().count() as u32;
|
||||
pub fn initialize_counter() -> Counter {
|
||||
let count = Self::iter_values().count() as Counter;
|
||||
CounterFor::<Prefix>::set(count);
|
||||
count
|
||||
}
|
||||
|
||||
/// Return the count.
|
||||
pub fn count() -> u32 {
|
||||
pub fn count() -> Counter {
|
||||
CounterFor::<Prefix>::get()
|
||||
}
|
||||
}
|
||||
@@ -1207,7 +1211,7 @@ mod test {
|
||||
StorageEntryMetadataIR {
|
||||
name: "counter_for_foo",
|
||||
modifier: StorageEntryModifierIR::Default,
|
||||
ty: StorageEntryTypeIR::Plain(scale_info::meta_type::<u32>()),
|
||||
ty: StorageEntryTypeIR::Plain(scale_info::meta_type::<Counter>()),
|
||||
default: vec![0, 0, 0, 0],
|
||||
docs: if cfg!(feature = "no-metadata-docs") {
|
||||
vec![]
|
||||
|
||||
@@ -114,8 +114,10 @@ impl<P: CountedStorageNMapInstance, K, V, Q, O, M> MapWrapper
|
||||
type Map = StorageNMap<P, K, V, Q, O, M>;
|
||||
}
|
||||
|
||||
type Counter = super::counted_map::Counter;
|
||||
|
||||
type CounterFor<P> =
|
||||
StorageValue<<P as CountedStorageNMapInstance>::CounterPrefix, u32, ValueQuery>;
|
||||
StorageValue<<P as CountedStorageNMapInstance>::CounterPrefix, Counter, ValueQuery>;
|
||||
|
||||
/// On removal logic for updating counter while draining upon some prefix with
|
||||
/// [`crate::storage::PrefixIterator`].
|
||||
@@ -472,7 +474,7 @@ where
|
||||
}
|
||||
|
||||
/// Return the count.
|
||||
pub fn count() -> u32 {
|
||||
pub fn count() -> Counter {
|
||||
CounterFor::<Prefix>::get()
|
||||
}
|
||||
}
|
||||
|
||||
@@ -30,7 +30,7 @@ mod map;
|
||||
mod nmap;
|
||||
mod value;
|
||||
|
||||
pub use counted_map::{CountedStorageMap, CountedStorageMapInstance};
|
||||
pub use counted_map::{CountedStorageMap, CountedStorageMapInstance, Counter};
|
||||
pub use counted_nmap::{CountedStorageNMap, CountedStorageNMapInstance};
|
||||
pub use double_map::StorageDoubleMap;
|
||||
pub use key::{
|
||||
|
||||
@@ -23,7 +23,7 @@ use crate::{
|
||||
types::{OptionQuery, QueryKindTrait, StorageEntryMetadataBuilder},
|
||||
StorageAppend, StorageDecodeLength, StorageTryAppend,
|
||||
},
|
||||
traits::{GetDefault, StorageInfo, StorageInstance},
|
||||
traits::{Get, GetDefault, StorageInfo, StorageInstance},
|
||||
};
|
||||
use codec::{Decode, Encode, EncodeLike, FullCodec, MaxEncodedLen};
|
||||
use frame_support::storage::StorageDecodeNonDedupLength;
|
||||
@@ -72,7 +72,7 @@ where
|
||||
Prefix: StorageInstance,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: crate::traits::Get<QueryKind::Query> + 'static,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
{
|
||||
type Query = QueryKind::Query;
|
||||
fn pallet_prefix() -> &'static [u8] {
|
||||
|
||||
@@ -126,4 +126,7 @@ pub use tx_pause::{TransactionPause, TransactionPauseError};
|
||||
#[cfg(feature = "try-runtime")]
|
||||
mod try_runtime;
|
||||
#[cfg(feature = "try-runtime")]
|
||||
pub use try_runtime::{Select as TryStateSelect, TryState, UpgradeCheckSelect};
|
||||
pub use try_runtime::{
|
||||
Select as TryStateSelect, TryDecodeEntireStorage, TryDecodeEntireStorageError, TryState,
|
||||
UpgradeCheckSelect,
|
||||
};
|
||||
|
||||
@@ -93,9 +93,7 @@ pub trait StorageInstance {
|
||||
}
|
||||
|
||||
/// Metadata about storage from the runtime.
|
||||
#[derive(
|
||||
codec::Encode, codec::Decode, RuntimeDebug, Eq, PartialEq, Clone, scale_info::TypeInfo,
|
||||
)]
|
||||
#[derive(Debug, codec::Encode, codec::Decode, Eq, PartialEq, Clone, scale_info::TypeInfo)]
|
||||
pub struct StorageInfo {
|
||||
/// Encoded string of pallet name.
|
||||
pub pallet_name: Vec<u8>,
|
||||
|
||||
@@ -0,0 +1,498 @@
|
||||
// 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.
|
||||
|
||||
//! Types to check that the entire storage can be decoded.
|
||||
|
||||
use super::StorageInstance;
|
||||
use crate::{
|
||||
storage::types::{
|
||||
CountedStorageMapInstance, CountedStorageNMapInstance, Counter, KeyGenerator,
|
||||
QueryKindTrait,
|
||||
},
|
||||
traits::{PartialStorageInfoTrait, StorageInfo},
|
||||
StorageHasher,
|
||||
};
|
||||
use codec::{Decode, DecodeAll, FullCodec};
|
||||
use impl_trait_for_tuples::impl_for_tuples;
|
||||
use sp_core::Get;
|
||||
use sp_std::prelude::*;
|
||||
|
||||
/// Decode the entire data under the given storage type.
|
||||
///
|
||||
/// For values, this is trivial. For all kinds of maps, it should decode all the values associated
|
||||
/// with all keys existing in the map.
|
||||
///
|
||||
/// Tuple implementations are provided and simply decode each type in the tuple, summing up the
|
||||
/// decoded bytes if `Ok(_)`.
|
||||
pub trait TryDecodeEntireStorage {
|
||||
/// Decode the entire data under the given storage, returning `Ok(bytes_decoded)` if success.
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>>;
|
||||
}
|
||||
|
||||
#[cfg_attr(all(not(feature = "tuples-96"), not(feature = "tuples-128")), impl_for_tuples(64))]
|
||||
#[cfg_attr(all(feature = "tuples-96", not(feature = "tuples-128")), impl_for_tuples(96))]
|
||||
#[cfg_attr(feature = "tuples-128", impl_for_tuples(128))]
|
||||
impl TryDecodeEntireStorage for Tuple {
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let mut errors = Vec::new();
|
||||
let mut len = 0usize;
|
||||
|
||||
for_tuples!(#(
|
||||
match Tuple::try_decode_entire_state() {
|
||||
Ok(bytes) => len += bytes,
|
||||
Err(errs) => errors.extend(errs),
|
||||
}
|
||||
)*);
|
||||
|
||||
if errors.is_empty() {
|
||||
Ok(len)
|
||||
} else {
|
||||
Err(errors)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// A value could not be decoded.
|
||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||
pub struct TryDecodeEntireStorageError {
|
||||
/// The key of the undecodable value.
|
||||
pub key: Vec<u8>,
|
||||
/// The raw value.
|
||||
pub raw: Option<Vec<u8>>,
|
||||
/// The storage info of the key.
|
||||
pub info: StorageInfo,
|
||||
}
|
||||
|
||||
impl core::fmt::Display for TryDecodeEntireStorageError {
|
||||
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"Failed to decode storage item `{}::{}`",
|
||||
&sp_std::str::from_utf8(&self.info.pallet_name).unwrap_or("<invalid>"),
|
||||
&sp_std::str::from_utf8(&self.info.storage_name).unwrap_or("<invalid>"),
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/// Decode all the values based on the prefix of `info` to `V`.
|
||||
///
|
||||
/// Basically, it decodes and sums up all the values who's key start with `info.prefix`. For values,
|
||||
/// this would be the value itself. For all sorts of maps, this should be all map items in the
|
||||
/// absence of key collision.
|
||||
fn decode_storage_info<V: Decode>(
|
||||
info: StorageInfo,
|
||||
) -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let mut next_key = info.prefix.clone();
|
||||
let mut decoded = 0;
|
||||
|
||||
let decode_key = |key: &[u8]| match sp_io::storage::get(key) {
|
||||
None => Ok(0),
|
||||
Some(bytes) => {
|
||||
let len = bytes.len();
|
||||
let _ = <V as DecodeAll>::decode_all(&mut bytes.as_ref()).map_err(|_| {
|
||||
vec![TryDecodeEntireStorageError {
|
||||
key: key.to_vec(),
|
||||
raw: Some(bytes.to_vec()),
|
||||
info: info.clone(),
|
||||
}]
|
||||
})?;
|
||||
|
||||
Ok::<usize, Vec<_>>(len)
|
||||
},
|
||||
};
|
||||
|
||||
decoded += decode_key(&next_key)?;
|
||||
loop {
|
||||
match sp_io::storage::next_key(&next_key) {
|
||||
Some(key) if key.starts_with(&info.prefix) => {
|
||||
decoded += decode_key(&key)?;
|
||||
next_key = key;
|
||||
},
|
||||
_ => break,
|
||||
}
|
||||
}
|
||||
|
||||
Ok(decoded)
|
||||
}
|
||||
|
||||
impl<Prefix, Value, QueryKind, OnEmpty> TryDecodeEntireStorage
|
||||
for crate::storage::types::StorageValue<Prefix, Value, QueryKind, OnEmpty>
|
||||
where
|
||||
Prefix: StorageInstance,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let info = Self::partial_storage_info()
|
||||
.first()
|
||||
.cloned()
|
||||
.expect("Value has only one storage info; qed");
|
||||
decode_storage_info::<Value>(info)
|
||||
}
|
||||
}
|
||||
|
||||
impl<Prefix, Hasher, Key, Value, QueryKind, OnEmpty, MaxValues> TryDecodeEntireStorage
|
||||
for crate::storage::types::StorageMap<Prefix, Hasher, Key, Value, QueryKind, OnEmpty, MaxValues>
|
||||
where
|
||||
Prefix: StorageInstance,
|
||||
Hasher: StorageHasher,
|
||||
Key: FullCodec,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
MaxValues: Get<Option<u32>>,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let info = Self::partial_storage_info()
|
||||
.first()
|
||||
.cloned()
|
||||
.expect("Map has only one storage info; qed");
|
||||
decode_storage_info::<Value>(info)
|
||||
}
|
||||
}
|
||||
|
||||
impl<Prefix, Hasher, Key, Value, QueryKind, OnEmpty, MaxValues> TryDecodeEntireStorage
|
||||
for crate::storage::types::CountedStorageMap<
|
||||
Prefix,
|
||||
Hasher,
|
||||
Key,
|
||||
Value,
|
||||
QueryKind,
|
||||
OnEmpty,
|
||||
MaxValues,
|
||||
> where
|
||||
Prefix: CountedStorageMapInstance,
|
||||
Hasher: StorageHasher,
|
||||
Key: FullCodec,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
MaxValues: Get<Option<u32>>,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let (map_info, counter_info) = match &Self::partial_storage_info()[..] {
|
||||
[a, b] => (a.clone(), b.clone()),
|
||||
_ => panic!("Counted map has two storage info items; qed"),
|
||||
};
|
||||
let mut decoded = decode_storage_info::<Counter>(counter_info)?;
|
||||
decoded += decode_storage_info::<Value>(map_info)?;
|
||||
Ok(decoded)
|
||||
}
|
||||
}
|
||||
|
||||
impl<Prefix, Hasher1, Key1, Hasher2, Key2, Value, QueryKind, OnEmpty, MaxValues>
|
||||
TryDecodeEntireStorage
|
||||
for crate::storage::types::StorageDoubleMap<
|
||||
Prefix,
|
||||
Hasher1,
|
||||
Key1,
|
||||
Hasher2,
|
||||
Key2,
|
||||
Value,
|
||||
QueryKind,
|
||||
OnEmpty,
|
||||
MaxValues,
|
||||
> where
|
||||
Prefix: StorageInstance,
|
||||
Hasher1: StorageHasher,
|
||||
Key1: FullCodec,
|
||||
Hasher2: StorageHasher,
|
||||
Key2: FullCodec,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
MaxValues: Get<Option<u32>>,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let info = Self::partial_storage_info()
|
||||
.first()
|
||||
.cloned()
|
||||
.expect("Double-map has only one storage info; qed");
|
||||
decode_storage_info::<Value>(info)
|
||||
}
|
||||
}
|
||||
|
||||
impl<Prefix, Key, Value, QueryKind, OnEmpty, MaxValues> TryDecodeEntireStorage
|
||||
for crate::storage::types::StorageNMap<Prefix, Key, Value, QueryKind, OnEmpty, MaxValues>
|
||||
where
|
||||
Prefix: StorageInstance,
|
||||
Key: KeyGenerator,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
MaxValues: Get<Option<u32>>,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let info = Self::partial_storage_info()
|
||||
.first()
|
||||
.cloned()
|
||||
.expect("N-map has only one storage info; qed");
|
||||
decode_storage_info::<Value>(info)
|
||||
}
|
||||
}
|
||||
|
||||
impl<Prefix, Key, Value, QueryKind, OnEmpty, MaxValues> TryDecodeEntireStorage
|
||||
for crate::storage::types::CountedStorageNMap<Prefix, Key, Value, QueryKind, OnEmpty, MaxValues>
|
||||
where
|
||||
Prefix: CountedStorageNMapInstance,
|
||||
Key: KeyGenerator,
|
||||
Value: FullCodec,
|
||||
QueryKind: QueryKindTrait<Value, OnEmpty>,
|
||||
OnEmpty: Get<QueryKind::Query> + 'static,
|
||||
MaxValues: Get<Option<u32>>,
|
||||
{
|
||||
fn try_decode_entire_state() -> Result<usize, Vec<TryDecodeEntireStorageError>> {
|
||||
let (map_info, counter_info) = match &Self::partial_storage_info()[..] {
|
||||
[a, b] => (a.clone(), b.clone()),
|
||||
_ => panic!("Counted NMap has two storage info items; qed"),
|
||||
};
|
||||
|
||||
let mut decoded = decode_storage_info::<Counter>(counter_info)?;
|
||||
decoded += decode_storage_info::<Value>(map_info)?;
|
||||
Ok(decoded)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use crate::{
|
||||
storage::types::{self, CountedStorageMapInstance, CountedStorageNMapInstance, Key},
|
||||
Blake2_128Concat,
|
||||
};
|
||||
|
||||
type H = Blake2_128Concat;
|
||||
|
||||
macro_rules! build_prefix {
|
||||
($name:ident) => {
|
||||
struct $name;
|
||||
impl StorageInstance for $name {
|
||||
fn pallet_prefix() -> &'static str {
|
||||
"test_pallet"
|
||||
}
|
||||
const STORAGE_PREFIX: &'static str = stringify!($name);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
build_prefix!(ValuePrefix);
|
||||
type Value = types::StorageValue<ValuePrefix, u32>;
|
||||
|
||||
build_prefix!(MapPrefix);
|
||||
type Map = types::StorageMap<MapPrefix, H, u32, u32>;
|
||||
|
||||
build_prefix!(CMapCounterPrefix);
|
||||
build_prefix!(CMapPrefix);
|
||||
impl CountedStorageMapInstance for CMapPrefix {
|
||||
type CounterPrefix = CMapCounterPrefix;
|
||||
}
|
||||
type CMap = types::CountedStorageMap<CMapPrefix, H, u8, u16>;
|
||||
|
||||
build_prefix!(DMapPrefix);
|
||||
type DMap = types::StorageDoubleMap<DMapPrefix, H, u32, H, u32, u32>;
|
||||
|
||||
build_prefix!(NMapPrefix);
|
||||
type NMap = types::StorageNMap<NMapPrefix, (Key<H, u8>, Key<H, u8>), u128>;
|
||||
|
||||
build_prefix!(CountedNMapCounterPrefix);
|
||||
build_prefix!(CountedNMapPrefix);
|
||||
impl CountedStorageNMapInstance for CountedNMapPrefix {
|
||||
type CounterPrefix = CountedNMapCounterPrefix;
|
||||
}
|
||||
type CNMap = types::CountedStorageNMap<CountedNMapPrefix, (Key<H, u8>, Key<H, u8>), u128>;
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_value_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(Value::try_decode_entire_state(), Ok(0));
|
||||
|
||||
Value::put(42);
|
||||
assert_eq!(Value::try_decode_entire_state(), Ok(4));
|
||||
|
||||
Value::kill();
|
||||
assert_eq!(Value::try_decode_entire_state(), Ok(0));
|
||||
|
||||
// two bytes, cannot be decoded into u32.
|
||||
sp_io::storage::set(&Value::hashed_key(), &[0u8, 1]);
|
||||
assert!(Value::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_map_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(0));
|
||||
|
||||
Map::insert(0, 42);
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(4));
|
||||
|
||||
Map::insert(0, 42);
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(4));
|
||||
|
||||
Map::insert(1, 42);
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(8));
|
||||
|
||||
Map::remove(0);
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(4));
|
||||
|
||||
// two bytes, cannot be decoded into u32.
|
||||
sp_io::storage::set(&Map::hashed_key_for(2), &[0u8, 1]);
|
||||
assert!(Map::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_counted_map_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
// counter is not even initialized;
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(0 + 0));
|
||||
|
||||
let counter = 4;
|
||||
let value_size = std::mem::size_of::<u16>();
|
||||
|
||||
CMap::insert(0, 42);
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(value_size + counter));
|
||||
|
||||
CMap::insert(0, 42);
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(value_size + counter));
|
||||
|
||||
CMap::insert(1, 42);
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(value_size * 2 + counter));
|
||||
|
||||
CMap::remove(0);
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(value_size + counter));
|
||||
|
||||
// counter is cleared again.
|
||||
let _ = CMap::clear(u32::MAX, None);
|
||||
assert_eq!(CMap::try_decode_entire_state(), Ok(0 + 0));
|
||||
|
||||
// 1 bytes, cannot be decoded into u16.
|
||||
sp_io::storage::set(&CMap::hashed_key_for(2), &[0u8]);
|
||||
assert!(CMap::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_double_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(0));
|
||||
|
||||
DMap::insert(0, 0, 42);
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(4));
|
||||
|
||||
DMap::insert(0, 0, 42);
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(4));
|
||||
|
||||
DMap::insert(0, 1, 42);
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(8));
|
||||
|
||||
DMap::insert(1, 0, 42);
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(12));
|
||||
|
||||
DMap::remove(0, 0);
|
||||
assert_eq!(DMap::try_decode_entire_state(), Ok(8));
|
||||
|
||||
// two bytes, cannot be decoded into u32.
|
||||
sp_io::storage::set(&DMap::hashed_key_for(1, 1), &[0u8, 1]);
|
||||
assert!(DMap::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_n_map_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(0));
|
||||
|
||||
let value_size = std::mem::size_of::<u128>();
|
||||
|
||||
NMap::insert((0u8, 0), 42);
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(value_size));
|
||||
|
||||
NMap::insert((0, 0), 42);
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(value_size));
|
||||
|
||||
NMap::insert((0, 1), 42);
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(value_size * 2));
|
||||
|
||||
NMap::insert((1, 0), 42);
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(value_size * 3));
|
||||
|
||||
NMap::remove((0, 0));
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(value_size * 2));
|
||||
|
||||
// two bytes, cannot be decoded into u128.
|
||||
sp_io::storage::set(&NMap::hashed_key_for((1, 1)), &[0u8, 1]);
|
||||
assert!(NMap::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_counted_n_map_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(NMap::try_decode_entire_state(), Ok(0));
|
||||
|
||||
let value_size = std::mem::size_of::<u128>();
|
||||
let counter = 4;
|
||||
|
||||
CNMap::insert((0u8, 0), 42);
|
||||
assert_eq!(CNMap::try_decode_entire_state(), Ok(value_size + counter));
|
||||
|
||||
CNMap::insert((0, 0), 42);
|
||||
assert_eq!(CNMap::try_decode_entire_state(), Ok(value_size + counter));
|
||||
|
||||
CNMap::insert((0, 1), 42);
|
||||
assert_eq!(CNMap::try_decode_entire_state(), Ok(value_size * 2 + counter));
|
||||
|
||||
CNMap::insert((1, 0), 42);
|
||||
assert_eq!(CNMap::try_decode_entire_state(), Ok(value_size * 3 + counter));
|
||||
|
||||
CNMap::remove((0, 0));
|
||||
assert_eq!(CNMap::try_decode_entire_state(), Ok(value_size * 2 + counter));
|
||||
|
||||
// two bytes, cannot be decoded into u128.
|
||||
sp_io::storage::set(&CNMap::hashed_key_for((1, 1)), &[0u8, 1]);
|
||||
assert!(CNMap::try_decode_entire_state().is_err());
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn extra_bytes_are_rejected() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(Map::try_decode_entire_state(), Ok(0));
|
||||
|
||||
// 6bytes, too many to fit in u32, should be rejected.
|
||||
sp_io::storage::set(&Map::hashed_key_for(2), &[0u8, 1, 3, 4, 5, 6]);
|
||||
assert!(Map::try_decode_entire_state().is_err());
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn try_decode_entire_state_tuple_of_storage_works() {
|
||||
sp_io::TestExternalities::new_empty().execute_with(|| {
|
||||
assert_eq!(<(Value, Map) as TryDecodeEntireStorage>::try_decode_entire_state(), Ok(0));
|
||||
|
||||
Value::put(42);
|
||||
assert_eq!(<(Value, Map) as TryDecodeEntireStorage>::try_decode_entire_state(), Ok(4));
|
||||
|
||||
Map::insert(0, 42);
|
||||
assert_eq!(<(Value, Map) as TryDecodeEntireStorage>::try_decode_entire_state(), Ok(8));
|
||||
});
|
||||
}
|
||||
}
|
||||
+17
@@ -17,6 +17,11 @@
|
||||
|
||||
//! Try-runtime specific traits and types.
|
||||
|
||||
pub mod decode_entire_state;
|
||||
pub use decode_entire_state::{TryDecodeEntireStorage, TryDecodeEntireStorageError};
|
||||
|
||||
use super::StorageInstance;
|
||||
|
||||
use impl_trait_for_tuples::impl_for_tuples;
|
||||
use sp_arithmetic::traits::AtLeast32BitUnsigned;
|
||||
use sp_runtime::TryRuntimeError;
|
||||
@@ -37,6 +42,13 @@ pub enum Select {
|
||||
Only(Vec<Vec<u8>>),
|
||||
}
|
||||
|
||||
impl Select {
|
||||
/// Whether to run any checks at all.
|
||||
pub fn any(&self) -> bool {
|
||||
!matches!(self, Select::None)
|
||||
}
|
||||
}
|
||||
|
||||
impl Default for Select {
|
||||
fn default() -> Self {
|
||||
Select::None
|
||||
@@ -105,6 +117,11 @@ impl UpgradeCheckSelect {
|
||||
pub fn try_state(&self) -> bool {
|
||||
matches!(self, Self::All | Self::TryState)
|
||||
}
|
||||
|
||||
/// Whether to run any checks at all.
|
||||
pub fn any(&self) -> bool {
|
||||
!matches!(self, Self::None)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(feature = "std")]
|
||||
Reference in New Issue
Block a user