Newer
Older
// Copyright 2019-2021 Parity Technologies (UK) Ltd.
Svyatoslav Nikolsky
committed
// 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/>.
//! Substrate client as Substrate finality proof target. The chain we connect to should have
//! bridge GRANDPA pallet deployed and provide `<BridgedChainName>FinalityApi` to allow bridging
//! with <BridgedName> chain.
Svyatoslav Nikolsky
committed
use crate::{
finality_pipeline::{
FinalitySyncPipelineAdapter, SubmitFinalityProofCallBuilder, SubstrateFinalitySyncPipeline,
},
Svyatoslav Nikolsky
committed
use async_trait::async_trait;
Svyatoslav Nikolsky
committed
use bp_header_chain::{justification::GrandpaJustification, storage_keys::is_halted_key};
use codec::Encode;
use finality_relay::TargetClient;
use relay_substrate_client::{
Svyatoslav Nikolsky
committed
AccountIdOf, AccountKeyPairOf, BlockNumberOf, Chain, ChainWithGrandpa, Client, Error, HashOf,
HeaderOf, SignParam, SyncHeader, TransactionEra, TransactionSignScheme, UnsignedTransaction,
Svyatoslav Nikolsky
committed
use relay_utils::relay_loop::Client as RelayClient;
use sp_core::{Bytes, Pair};
Svyatoslav Nikolsky
committed
/// Substrate client as Substrate finality target.
pub struct SubstrateFinalityTarget<P: SubstrateFinalitySyncPipeline> {
client: Client<P::TargetChain>,
transaction_params: TransactionParams<AccountKeyPairOf<P::TransactionSignScheme>>,
Svyatoslav Nikolsky
committed
}
impl<P: SubstrateFinalitySyncPipeline> SubstrateFinalityTarget<P> {
Svyatoslav Nikolsky
committed
/// Create new Substrate headers target.
pub fn new(
client: Client<P::TargetChain>,
transaction_params: TransactionParams<AccountKeyPairOf<P::TransactionSignScheme>>,
) -> Self {
SubstrateFinalityTarget { client, transaction_params }
Svyatoslav Nikolsky
committed
}
Svyatoslav Nikolsky
committed
/// Ensure that the GRANDPA pallet at target chain is active.
async fn ensure_pallet_active(&self) -> Result<(), Error> {
let is_halted = self
.client
.storage_value(is_halted_key(P::SourceChain::WITH_CHAIN_GRANDPA_PALLET_NAME), None)
.await?;
if is_halted.unwrap_or(false) {
Err(Error::BridgePalletIsHalted)
} else {
Ok(())
}
}
Svyatoslav Nikolsky
committed
}
impl<P: SubstrateFinalitySyncPipeline> Clone for SubstrateFinalityTarget<P> {
Svyatoslav Nikolsky
committed
fn clone(&self) -> Self {
SubstrateFinalityTarget {
client: self.client.clone(),
transaction_params: self.transaction_params.clone(),
Svyatoslav Nikolsky
committed
}
}
}
#[async_trait]
impl<P: SubstrateFinalitySyncPipeline> RelayClient for SubstrateFinalityTarget<P> {
type Error = Error;
Svyatoslav Nikolsky
committed
async fn reconnect(&mut self) -> Result<(), Error> {
Svyatoslav Nikolsky
committed
self.client.reconnect().await
}
}
#[async_trait]
impl<P: SubstrateFinalitySyncPipeline> TargetClient<FinalitySyncPipelineAdapter<P>>
for SubstrateFinalityTarget<P>
Svyatoslav Nikolsky
committed
where
AccountIdOf<P::TargetChain>: From<<AccountKeyPairOf<P::TransactionSignScheme> as Pair>::Public>,
P::TransactionSignScheme: TransactionSignScheme<Chain = P::TargetChain>,
Svyatoslav Nikolsky
committed
{
async fn best_finalized_source_block_number(
&self,
) -> Result<BlockNumberOf<P::SourceChain>, Error> {
Svyatoslav Nikolsky
committed
// we can't continue to relay finality if target node is out of sync, because
// it may have already received (some of) headers that we're going to relay
self.client.ensure_synced().await?;
Svyatoslav Nikolsky
committed
// we can't relay finality if GRANDPA pallet at target chain is halted
self.ensure_pallet_active().await?;
Svyatoslav Nikolsky
committed
Ok(crate::messages_source::read_client_state::<
P::TargetChain,
HashOf<P::SourceChain>,
BlockNumberOf<P::SourceChain>,
>(&self.client, P::SourceChain::BEST_FINALIZED_HEADER_ID_METHOD)
Svyatoslav Nikolsky
committed
.await?
.best_finalized_peer_at_best_self
.0)
}
async fn submit_finality_proof(
&self,
header: SyncHeader<HeaderOf<P::SourceChain>>,
proof: GrandpaJustification<HeaderOf<P::SourceChain>>,
) -> Result<(), Error> {
let genesis_hash = *self.client.genesis_hash();
let transaction_params = self.transaction_params.clone();
let call =
P::SubmitFinalityProofCallBuilder::build_submit_finality_proof_call(header, proof);
fewensa
committed
let (spec_version, transaction_version) = self.client.simple_runtime_version().await?;
self.transaction_params.signer.public().into(),
fewensa
committed
P::TransactionSignScheme::sign_transaction(SignParam {
spec_version,
transaction_version,
fewensa
committed
signer: transaction_params.signer.clone(),
era: TransactionEra::new(best_block_id, transaction_params.mortality),
unsigned: UnsignedTransaction::new(call, transaction_nonce),
})
Svyatoslav Nikolsky
committed
.await
.map(drop)
}
}