mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-30 19:51:02 +00:00
169 lines
5.5 KiB
Rust
169 lines
5.5 KiB
Rust
// Copyright 2019-2021 Parity Technologies (UK) Ltd.
|
|
// This file is part of Parity Bridges Common.
|
|
|
|
// Parity Bridges Common 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.
|
|
|
|
// Parity Bridges Common 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 Parity Bridges Common. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! Default generic implementation of finality source for basic Substrate client.
|
|
|
|
use crate::chain::{BlockWithJustification, Chain};
|
|
use crate::client::Client;
|
|
use crate::error::Error;
|
|
use crate::sync_header::SyncHeader;
|
|
|
|
use async_std::sync::{Arc, Mutex};
|
|
use async_trait::async_trait;
|
|
use bp_header_chain::justification::GrandpaJustification;
|
|
use codec::Decode;
|
|
use finality_relay::{FinalitySyncPipeline, SourceClient, SourceHeader};
|
|
use futures::stream::{unfold, Stream, StreamExt};
|
|
use relay_utils::relay_loop::Client as RelayClient;
|
|
use sp_runtime::traits::Header as HeaderT;
|
|
use std::{marker::PhantomData, pin::Pin};
|
|
|
|
/// Shared updatable reference to the maximal header number that we want to sync from the source.
|
|
pub type RequiredHeaderNumberRef<C> = Arc<Mutex<<C as bp_runtime::Chain>::BlockNumber>>;
|
|
|
|
/// Substrate node as finality source.
|
|
pub struct FinalitySource<C: Chain, P> {
|
|
client: Client<C>,
|
|
maximal_header_number: Option<RequiredHeaderNumberRef<C>>,
|
|
_phantom: PhantomData<P>,
|
|
}
|
|
|
|
impl<C: Chain, P> FinalitySource<C, P> {
|
|
/// Create new headers source using given client.
|
|
pub fn new(client: Client<C>, maximal_header_number: Option<RequiredHeaderNumberRef<C>>) -> Self {
|
|
FinalitySource {
|
|
client,
|
|
maximal_header_number,
|
|
_phantom: Default::default(),
|
|
}
|
|
}
|
|
|
|
/// Returns reference to the underlying RPC client.
|
|
pub fn client(&self) -> &Client<C> {
|
|
&self.client
|
|
}
|
|
|
|
/// Returns best finalized block number.
|
|
pub async fn on_chain_best_finalized_block_number(&self) -> Result<C::BlockNumber, Error> {
|
|
// we **CAN** continue to relay finality proofs if source node is out of sync, because
|
|
// target node may be missing proofs that are already available at the source
|
|
let finalized_header_hash = self.client.best_finalized_header_hash().await?;
|
|
let finalized_header = self.client.header_by_hash(finalized_header_hash).await?;
|
|
Ok(*finalized_header.number())
|
|
}
|
|
}
|
|
|
|
impl<C: Chain, P> Clone for FinalitySource<C, P> {
|
|
fn clone(&self) -> Self {
|
|
FinalitySource {
|
|
client: self.client.clone(),
|
|
maximal_header_number: self.maximal_header_number.clone(),
|
|
_phantom: Default::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl<C: Chain, P: FinalitySyncPipeline> RelayClient for FinalitySource<C, P> {
|
|
type Error = Error;
|
|
|
|
async fn reconnect(&mut self) -> Result<(), Error> {
|
|
self.client.reconnect().await
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl<C, P> SourceClient<P> for FinalitySource<C, P>
|
|
where
|
|
C: Chain,
|
|
C::BlockNumber: relay_utils::BlockNumberBase,
|
|
P: FinalitySyncPipeline<
|
|
Hash = C::Hash,
|
|
Number = C::BlockNumber,
|
|
Header = SyncHeader<C::Header>,
|
|
FinalityProof = GrandpaJustification<C::Header>,
|
|
>,
|
|
P::Header: SourceHeader<C::BlockNumber>,
|
|
{
|
|
type FinalityProofsStream = Pin<Box<dyn Stream<Item = GrandpaJustification<C::Header>> + Send>>;
|
|
|
|
async fn best_finalized_block_number(&self) -> Result<P::Number, Error> {
|
|
let mut finalized_header_number = self.on_chain_best_finalized_block_number().await?;
|
|
// never return block number larger than requested. This way we'll never sync headers
|
|
// past `maximal_header_number`
|
|
if let Some(ref maximal_header_number) = self.maximal_header_number {
|
|
let maximal_header_number = *maximal_header_number.lock().await;
|
|
if finalized_header_number > maximal_header_number {
|
|
finalized_header_number = maximal_header_number;
|
|
}
|
|
}
|
|
Ok(finalized_header_number)
|
|
}
|
|
|
|
async fn header_and_finality_proof(
|
|
&self,
|
|
number: P::Number,
|
|
) -> Result<(P::Header, Option<P::FinalityProof>), Error> {
|
|
let header_hash = self.client.block_hash_by_number(number).await?;
|
|
let signed_block = self.client.get_block(Some(header_hash)).await?;
|
|
|
|
let justification = signed_block
|
|
.justification()
|
|
.map(|raw_justification| GrandpaJustification::<C::Header>::decode(&mut raw_justification.as_slice()))
|
|
.transpose()
|
|
.map_err(Error::ResponseParseFailed)?;
|
|
|
|
Ok((signed_block.header().into(), justification))
|
|
}
|
|
|
|
async fn finality_proofs(&self) -> Result<Self::FinalityProofsStream, Error> {
|
|
Ok(unfold(
|
|
self.client.clone().subscribe_justifications().await?,
|
|
move |subscription| async move {
|
|
loop {
|
|
let log_error = |err| {
|
|
log::error!(
|
|
target: "bridge",
|
|
"Failed to read justification target from the {} justifications stream: {:?}",
|
|
P::SOURCE_NAME,
|
|
err,
|
|
);
|
|
};
|
|
|
|
let next_justification = subscription
|
|
.next()
|
|
.await
|
|
.map_err(|err| log_error(err.to_string()))
|
|
.ok()??;
|
|
let decoded_justification =
|
|
GrandpaJustification::<C::Header>::decode(&mut &next_justification.0[..]);
|
|
|
|
let justification = match decoded_justification {
|
|
Ok(j) => j,
|
|
Err(err) => {
|
|
log_error(format!("decode failed with error {:?}", err));
|
|
continue;
|
|
}
|
|
};
|
|
|
|
return Some((justification, subscription));
|
|
}
|
|
},
|
|
)
|
|
.boxed())
|
|
}
|
|
}
|