mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-19 22:51:03 +00:00
002d9260f9
**Update:** Pushed additional changes based on the review comments. **This pull request fixes various spelling mistakes in this repository.** Most of the changes are contained in the first **3** commits: - `Fix spelling mistakes in comments and docs` - `Fix spelling mistakes in test names` - `Fix spelling mistakes in error messages, panic messages, logs and tracing` Other source code spelling mistakes are separated into individual commits for easier reviewing: - `Fix the spelling of 'authority'` - `Fix the spelling of 'REASONABLE_HEADERS_IN_JUSTIFICATION_ANCESTRY'` - `Fix the spelling of 'prev_enqueud_messages'` - `Fix the spelling of 'endpoint'` - `Fix the spelling of 'children'` - `Fix the spelling of 'PenpalSiblingSovereignAccount'` - `Fix the spelling of 'PenpalSudoAccount'` - `Fix the spelling of 'insufficient'` - `Fix the spelling of 'PalletXcmExtrinsicsBenchmark'` - `Fix the spelling of 'subtracted'` - `Fix the spelling of 'CandidatePendingAvailability'` - `Fix the spelling of 'exclusive'` - `Fix the spelling of 'until'` - `Fix the spelling of 'discriminator'` - `Fix the spelling of 'nonexistent'` - `Fix the spelling of 'subsystem'` - `Fix the spelling of 'indices'` - `Fix the spelling of 'committed'` - `Fix the spelling of 'topology'` - `Fix the spelling of 'response'` - `Fix the spelling of 'beneficiary'` - `Fix the spelling of 'formatted'` - `Fix the spelling of 'UNKNOWN_PROOF_REQUEST'` - `Fix the spelling of 'succeeded'` - `Fix the spelling of 'reopened'` - `Fix the spelling of 'proposer'` - `Fix the spelling of 'InstantiationNonce'` - `Fix the spelling of 'depositor'` - `Fix the spelling of 'expiration'` - `Fix the spelling of 'phantom'` - `Fix the spelling of 'AggregatedKeyValue'` - `Fix the spelling of 'randomness'` - `Fix the spelling of 'defendant'` - `Fix the spelling of 'AquaticMammal'` - `Fix the spelling of 'transactions'` - `Fix the spelling of 'PassingTracingSubscriber'` - `Fix the spelling of 'TxSignaturePayload'` - `Fix the spelling of 'versioning'` - `Fix the spelling of 'descendant'` - `Fix the spelling of 'overridden'` - `Fix the spelling of 'network'` Let me know if this structure is adequate. **Note:** The usage of the words `Merkle`, `Merkelize`, `Merklization`, `Merkelization`, `Merkleization`, is somewhat inconsistent but I left it as it is. ~~**Note:** In some places the term `Receival` is used to refer to message reception, IMO `Reception` is the correct word here, but I left it as it is.~~ ~~**Note:** In some places the term `Overlayed` is used instead of the more acceptable version `Overlaid` but I also left it as it is.~~ ~~**Note:** In some places the term `Applyable` is used instead of the correct version `Applicable` but I also left it as it is.~~ **Note:** Some usage of British vs American english e.g. `judgement` vs `judgment`, `initialise` vs `initialize`, `optimise` vs `optimize` etc. are both present in different places, but I suppose that's understandable given the number of contributors. ~~**Note:** There is a spelling mistake in `.github/CODEOWNERS` but it triggers errors in CI when I make changes to it, so I left it as it is.~~
178 lines
5.6 KiB
Rust
178 lines
5.6 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.
|
|
|
|
//! Rpc for state migration.
|
|
|
|
use jsonrpsee::{
|
|
core::RpcResult,
|
|
proc_macros::rpc,
|
|
types::error::{ErrorCode, ErrorObject, ErrorObjectOwned},
|
|
};
|
|
use sc_rpc_api::DenyUnsafe;
|
|
use serde::{Deserialize, Serialize};
|
|
use sp_runtime::traits::Block as BlockT;
|
|
use std::sync::Arc;
|
|
|
|
use sp_core::{
|
|
storage::{ChildInfo, ChildType, PrefixedStorageKey},
|
|
Hasher,
|
|
};
|
|
use sp_state_machine::backend::AsTrieBackend;
|
|
use sp_trie::{
|
|
trie_types::{TrieDB, TrieDBBuilder},
|
|
KeySpacedDB, Trie,
|
|
};
|
|
use trie_db::{
|
|
node::{NodePlan, ValuePlan},
|
|
TrieDBNodeIterator,
|
|
};
|
|
|
|
fn count_migrate<'a, H: Hasher>(
|
|
storage: &'a dyn trie_db::HashDBRef<H, Vec<u8>>,
|
|
root: &'a H::Out,
|
|
) -> std::result::Result<(u64, u64, TrieDB<'a, 'a, H>), String> {
|
|
let mut nb = 0u64;
|
|
let mut total_nb = 0u64;
|
|
let trie = TrieDBBuilder::new(storage, root).build();
|
|
let iter_node =
|
|
TrieDBNodeIterator::new(&trie).map_err(|e| format!("TrieDB node iterator error: {}", e))?;
|
|
for node in iter_node {
|
|
let node = node.map_err(|e| format!("TrieDB node iterator error: {}", e))?;
|
|
match node.2.node_plan() {
|
|
NodePlan::Leaf { value, .. } | NodePlan::NibbledBranch { value: Some(value), .. } => {
|
|
total_nb += 1;
|
|
if let ValuePlan::Inline(range) = value {
|
|
if (range.end - range.start) as u32 >=
|
|
sp_core::storage::TRIE_VALUE_NODE_THRESHOLD
|
|
{
|
|
nb += 1;
|
|
}
|
|
}
|
|
},
|
|
_ => (),
|
|
}
|
|
}
|
|
Ok((nb, total_nb, trie))
|
|
}
|
|
|
|
/// Check trie migration status.
|
|
pub fn migration_status<H, B>(backend: &B) -> std::result::Result<MigrationStatusResult, String>
|
|
where
|
|
H: Hasher,
|
|
H::Out: codec::Codec,
|
|
B: AsTrieBackend<H>,
|
|
{
|
|
let trie_backend = backend.as_trie_backend();
|
|
let essence = trie_backend.essence();
|
|
let (top_remaining_to_migrate, total_top, trie) = count_migrate(essence, essence.root())?;
|
|
|
|
let mut child_remaining_to_migrate = 0;
|
|
let mut total_child = 0;
|
|
let mut child_roots: Vec<(ChildInfo, Vec<u8>)> = Vec::new();
|
|
// get all child trie roots
|
|
for key_value in trie.iter().map_err(|e| format!("TrieDB node iterator error: {}", e))? {
|
|
let (key, value) = key_value.map_err(|e| format!("TrieDB node iterator error: {}", e))?;
|
|
if key[..].starts_with(sp_core::storage::well_known_keys::DEFAULT_CHILD_STORAGE_KEY_PREFIX)
|
|
{
|
|
let prefixed_key = PrefixedStorageKey::new(key);
|
|
let (_type, unprefixed) = ChildType::from_prefixed_key(&prefixed_key).unwrap();
|
|
child_roots.push((ChildInfo::new_default(unprefixed), value));
|
|
}
|
|
}
|
|
for (child_info, root) in child_roots {
|
|
let mut child_root = H::Out::default();
|
|
let storage = KeySpacedDB::new(essence, child_info.keyspace());
|
|
|
|
child_root.as_mut()[..].copy_from_slice(&root[..]);
|
|
let (nb, total_top, _) = count_migrate(&storage, &child_root)?;
|
|
child_remaining_to_migrate += nb;
|
|
total_child += total_top;
|
|
}
|
|
|
|
Ok(MigrationStatusResult {
|
|
top_remaining_to_migrate,
|
|
child_remaining_to_migrate,
|
|
total_top,
|
|
total_child,
|
|
})
|
|
}
|
|
|
|
/// Current state migration status.
|
|
#[derive(Serialize, Deserialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
#[serde(deny_unknown_fields)]
|
|
pub struct MigrationStatusResult {
|
|
/// Number of top items that should migrate.
|
|
pub top_remaining_to_migrate: u64,
|
|
/// Number of child items that should migrate.
|
|
pub child_remaining_to_migrate: u64,
|
|
/// Number of top items that we will iterate on.
|
|
pub total_top: u64,
|
|
/// Number of child items that we will iterate on.
|
|
pub total_child: u64,
|
|
}
|
|
|
|
/// Migration RPC methods.
|
|
#[rpc(server)]
|
|
pub trait StateMigrationApi<BlockHash> {
|
|
/// Check current migration state.
|
|
///
|
|
/// This call is performed locally without submitting any transactions. Thus executing this
|
|
/// won't change any state. Nonetheless it is a VERY costly call that should be
|
|
/// only exposed to trusted peers.
|
|
#[method(name = "state_trieMigrationStatus")]
|
|
fn call(&self, at: Option<BlockHash>) -> RpcResult<MigrationStatusResult>;
|
|
}
|
|
|
|
/// An implementation of state migration specific RPC methods.
|
|
pub struct StateMigration<C, B, BA> {
|
|
client: Arc<C>,
|
|
backend: Arc<BA>,
|
|
deny_unsafe: DenyUnsafe,
|
|
_marker: std::marker::PhantomData<(B, BA)>,
|
|
}
|
|
|
|
impl<C, B, BA> StateMigration<C, B, BA> {
|
|
/// Create new state migration rpc for the given reference to the client.
|
|
pub fn new(client: Arc<C>, backend: Arc<BA>, deny_unsafe: DenyUnsafe) -> Self {
|
|
StateMigration { client, backend, deny_unsafe, _marker: Default::default() }
|
|
}
|
|
}
|
|
|
|
impl<C, B, BA> StateMigrationApiServer<<B as BlockT>::Hash> for StateMigration<C, B, BA>
|
|
where
|
|
B: BlockT,
|
|
C: Send + Sync + 'static + sc_client_api::HeaderBackend<B>,
|
|
BA: 'static + sc_client_api::backend::Backend<B>,
|
|
{
|
|
fn call(&self, at: Option<<B as BlockT>::Hash>) -> RpcResult<MigrationStatusResult> {
|
|
self.deny_unsafe.check_if_safe()?;
|
|
|
|
let hash = at.unwrap_or_else(|| self.client.info().best_hash);
|
|
let state = self.backend.state_at(hash).map_err(error_into_rpc_err)?;
|
|
migration_status(&state).map_err(error_into_rpc_err)
|
|
}
|
|
}
|
|
|
|
fn error_into_rpc_err(err: impl std::fmt::Display) -> ErrorObjectOwned {
|
|
ErrorObject::owned(
|
|
ErrorCode::InternalError.code(),
|
|
"Error while checking migration state",
|
|
Some(err.to_string()),
|
|
)
|
|
}
|