mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-29 06:47:57 +00:00
7b56ab15b4
* Run cargo fmt on the whole code base * Second run * Add CI check * Fix compilation * More unnecessary braces * Handle weights * Use --all * Use correct attributes... * Fix UI tests * AHHHHHHHHH * 🤦 * Docs * Fix compilation * 🤷 * Please stop * 🤦 x 2 * More * make rustfmt.toml consistent with polkadot Co-authored-by: André Silva <andrerfosilva@gmail.com>
138 lines
4.5 KiB
Rust
138 lines
4.5 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2020-2021 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
//! A RPC handler to create sync states for light clients.
|
|
//! Currently only usable with BABE + GRANDPA.
|
|
|
|
#![deny(unused_crate_dependencies)]
|
|
|
|
use sp_blockchain::HeaderBackend;
|
|
use sp_runtime::{
|
|
generic::BlockId,
|
|
traits::{Block as BlockT, NumberFor},
|
|
};
|
|
use std::sync::Arc;
|
|
|
|
use jsonrpc_derive::rpc;
|
|
|
|
type SharedAuthoritySet<TBl> =
|
|
sc_finality_grandpa::SharedAuthoritySet<<TBl as BlockT>::Hash, NumberFor<TBl>>;
|
|
type SharedEpochChanges<TBl> =
|
|
sc_consensus_epochs::SharedEpochChanges<TBl, sc_consensus_babe::Epoch>;
|
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
#[allow(missing_docs)]
|
|
enum Error<Block: BlockT> {
|
|
#[error(transparent)]
|
|
Blockchain(#[from] sp_blockchain::Error),
|
|
|
|
#[error("Failed to load the block weight for block {0:?}")]
|
|
LoadingBlockWeightFailed(<Block as BlockT>::Hash),
|
|
|
|
#[error("JsonRpc error: {0}")]
|
|
JsonRpc(String),
|
|
}
|
|
|
|
impl<Block: BlockT> From<Error<Block>> for jsonrpc_core::Error {
|
|
fn from(error: Error<Block>) -> Self {
|
|
let message = match error {
|
|
Error::JsonRpc(s) => s,
|
|
_ => error.to_string(),
|
|
};
|
|
jsonrpc_core::Error { message, code: jsonrpc_core::ErrorCode::ServerError(1), data: None }
|
|
}
|
|
}
|
|
|
|
/// An api for sync state RPC calls.
|
|
#[rpc]
|
|
pub trait SyncStateRpcApi {
|
|
/// Returns the json-serialized chainspec running the node, with a sync state.
|
|
#[rpc(name = "sync_state_genSyncSpec", returns = "jsonrpc_core::Value")]
|
|
fn system_gen_sync_spec(&self, raw: bool) -> jsonrpc_core::Result<jsonrpc_core::Value>;
|
|
}
|
|
|
|
/// The handler for sync state RPC calls.
|
|
pub struct SyncStateRpcHandler<TBl: BlockT, TCl> {
|
|
chain_spec: Box<dyn sc_chain_spec::ChainSpec>,
|
|
client: Arc<TCl>,
|
|
shared_authority_set: SharedAuthoritySet<TBl>,
|
|
shared_epoch_changes: SharedEpochChanges<TBl>,
|
|
deny_unsafe: sc_rpc_api::DenyUnsafe,
|
|
}
|
|
|
|
impl<TBl, TCl> SyncStateRpcHandler<TBl, TCl>
|
|
where
|
|
TBl: BlockT,
|
|
TCl: HeaderBackend<TBl> + sc_client_api::AuxStore + 'static,
|
|
{
|
|
/// Create a new handler.
|
|
pub fn new(
|
|
chain_spec: Box<dyn sc_chain_spec::ChainSpec>,
|
|
client: Arc<TCl>,
|
|
shared_authority_set: SharedAuthoritySet<TBl>,
|
|
shared_epoch_changes: SharedEpochChanges<TBl>,
|
|
deny_unsafe: sc_rpc_api::DenyUnsafe,
|
|
) -> Self {
|
|
Self { chain_spec, client, shared_authority_set, shared_epoch_changes, deny_unsafe }
|
|
}
|
|
|
|
fn build_sync_state(&self) -> Result<sc_chain_spec::LightSyncState<TBl>, Error<TBl>> {
|
|
let finalized_hash = self.client.info().finalized_hash;
|
|
let finalized_header = self
|
|
.client
|
|
.header(BlockId::Hash(finalized_hash))?
|
|
.ok_or_else(|| sp_blockchain::Error::MissingHeader(finalized_hash.to_string()))?;
|
|
|
|
let finalized_block_weight =
|
|
sc_consensus_babe::aux_schema::load_block_weight(&*self.client, finalized_hash)?
|
|
.ok_or_else(|| Error::LoadingBlockWeightFailed(finalized_hash))?;
|
|
|
|
Ok(sc_chain_spec::LightSyncState {
|
|
finalized_block_header: finalized_header,
|
|
babe_epoch_changes: self.shared_epoch_changes.shared_data().clone(),
|
|
babe_finalized_block_weight: finalized_block_weight,
|
|
grandpa_authority_set: self.shared_authority_set.clone_inner(),
|
|
})
|
|
}
|
|
}
|
|
|
|
impl<TBl, TCl> SyncStateRpcApi for SyncStateRpcHandler<TBl, TCl>
|
|
where
|
|
TBl: BlockT,
|
|
TCl: HeaderBackend<TBl> + sc_client_api::AuxStore + 'static,
|
|
{
|
|
fn system_gen_sync_spec(&self, raw: bool) -> jsonrpc_core::Result<jsonrpc_core::Value> {
|
|
if let Err(err) = self.deny_unsafe.check_if_safe() {
|
|
return Err(err.into())
|
|
}
|
|
|
|
let mut chain_spec = self.chain_spec.cloned_box();
|
|
|
|
let sync_state = self.build_sync_state().map_err(map_error::<TBl, Error<TBl>>)?;
|
|
|
|
chain_spec.set_light_sync_state(sync_state.to_serializable());
|
|
let string = chain_spec.as_json(raw).map_err(map_error::<TBl, _>)?;
|
|
|
|
serde_json::from_str(&string).map_err(|err| map_error::<TBl, _>(err))
|
|
}
|
|
}
|
|
|
|
fn map_error<Block: BlockT, S: ToString>(error: S) -> jsonrpc_core::Error {
|
|
Error::<Block>::JsonRpc(error.to_string()).into()
|
|
}
|