mod block_checker;
mod equivocation_loop;
mod mock;
mod reporter;
use async_trait::async_trait;
use bp_header_chain::{FinalityProof, FindEquivocations};
use finality_relay::{FinalityPipeline, SourceClientBase};
use relay_utils::{relay_loop::Client as RelayClient, MaybeConnectionError, TransactionTracker};
use std::{fmt::Debug, time::Duration};
pub use equivocation_loop::run;
#[cfg(not(test))]
const RECONNECT_DELAY: Duration = relay_utils::relay_loop::RECONNECT_DELAY;
#[cfg(test)]
const RECONNECT_DELAY: Duration = mock::TEST_RECONNECT_DELAY;
pub trait EquivocationDetectionPipeline: FinalityPipeline {
type TargetNumber: relay_utils::BlockNumberBase;
type FinalityVerificationContext: Debug + Send;
type EquivocationProof: Clone + Debug + Send + Sync;
type EquivocationsFinder: FindEquivocations<
Self::FinalityProof,
Self::FinalityVerificationContext,
Self::EquivocationProof,
>;
}
type HeaderFinalityInfo<P> = bp_header_chain::HeaderFinalityInfo<
<P as FinalityPipeline>::FinalityProof,
<P as EquivocationDetectionPipeline>::FinalityVerificationContext,
>;
#[async_trait]
pub trait SourceClient<P: EquivocationDetectionPipeline>: SourceClientBase<P> {
type TransactionTracker: TransactionTracker;
async fn report_equivocation(
&self,
at: P::Hash,
equivocation: P::EquivocationProof,
) -> Result<Self::TransactionTracker, Self::Error>;
}
#[async_trait]
pub trait TargetClient<P: EquivocationDetectionPipeline>: RelayClient {
async fn best_finalized_header_number(&self) -> Result<P::TargetNumber, Self::Error>;
async fn best_synced_header_hash(
&self,
at: P::TargetNumber,
) -> Result<Option<P::Hash>, Self::Error>;
async fn finality_verification_context(
&self,
at: P::TargetNumber,
) -> Result<P::FinalityVerificationContext, Self::Error>;
async fn synced_headers_finality_info(
&self,
at: P::TargetNumber,
) -> Result<Vec<HeaderFinalityInfo<P>>, Self::Error>;
}
#[derive(Debug, PartialEq)]
struct EquivocationReportingContext<P: EquivocationDetectionPipeline> {
pub synced_header_hash: P::Hash,
pub synced_verification_context: P::FinalityVerificationContext,
}
impl<P: EquivocationDetectionPipeline> EquivocationReportingContext<P> {
pub async fn try_read_from_target<TC: TargetClient<P>>(
target_client: &TC,
at: P::TargetNumber,
) -> Result<Option<Self>, TC::Error> {
let maybe_best_synced_header_hash = target_client.best_synced_header_hash(at).await?;
Ok(match maybe_best_synced_header_hash {
Some(best_synced_header_hash) => Some(EquivocationReportingContext {
synced_header_hash: best_synced_header_hash,
synced_verification_context: target_client
.finality_verification_context(at)
.await?,
}),
None => None,
})
}
pub fn update(&mut self, info: HeaderFinalityInfo<P>) {
if let Some(new_verification_context) = info.new_verification_context {
self.synced_header_hash = info.finality_proof.target_header_hash();
self.synced_verification_context = new_verification_context;
}
}
}
async fn handle_client_error<C: RelayClient>(client: &mut C, e: C::Error) {
if e.is_connection_error() {
client.reconnect_until_success(RECONNECT_DELAY).await;
} else {
async_std::task::sleep(RECONNECT_DELAY).await;
}
}