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:

![image](https://github.com/paritytech/polkadot-sdk/assets/10380170/73052d4f-4da5-4b21-a8dd-b17004e5965e)

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:
Piet
2023-11-06 19:40:14 +01:00
committed by GitHub
parent 15df7f54d2
commit 32a974088c
19 changed files with 807 additions and 36 deletions
@@ -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] {
+4 -1
View File
@@ -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,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")]