Only recheck blocks with non-fatal inherent transaction errors

This commit is contained in:
Luke Parker 2022-11-11 02:17:10 -05:00
parent 7d46daa36e
commit 6f74bade8b
No known key found for this signature in database
GPG key ID: F9F1386DB1E119B6
2 changed files with 23 additions and 7 deletions

View file

@ -1,6 +1,7 @@
use std::{
sync::{Arc, RwLock},
time::{UNIX_EPOCH, SystemTime, Duration},
collections::HashSet,
};
use async_trait::async_trait;
@ -318,8 +319,7 @@ impl<T: TendermintValidator> Network for TendermintAuthority<T> {
origin: None,
allow_missing_state: false,
skip_execution: false,
// TODO: Only set to true if block was rejected due to its inherents
import_existing: true,
import_existing: self.import.recheck.read().unwrap().contains(&hash),
state: None,
}],
);
@ -354,6 +354,9 @@ impl<T: TendermintValidator> Network for TendermintAuthority<T> {
.map_err(|_| Error::InvalidJustification)
.unwrap();
// Clear any blocks for the previous height we were willing to recheck
*self.import.recheck.write().unwrap() = HashSet::new();
let number: u64 = match (*block.header().number()).try_into() {
Ok(number) => number,
Err(_) => panic!("BlockNumber exceeded u64"),

View file

@ -1,4 +1,7 @@
use std::sync::{Arc, RwLock};
use std::{
sync::{Arc, RwLock},
collections::HashSet,
};
use log::warn;
@ -31,6 +34,7 @@ pub struct TendermintImport<T: TendermintValidator> {
pub(crate) providers: Arc<AsyncRwLock<Option<T::CIDP>>>,
pub(crate) importing_block: Arc<RwLock<Option<<T::Block as Block>::Hash>>>,
pub(crate) recheck: Arc<RwLock<HashSet<<T::Block as Block>::Hash>>>,
pub(crate) client: Arc<T::Client>,
pub(crate) queue:
@ -44,6 +48,7 @@ impl<T: TendermintValidator> Clone for TendermintImport<T> {
providers: self.providers.clone(),
importing_block: self.importing_block.clone(),
recheck: self.recheck.clone(),
client: self.client.clone(),
queue: self.queue.clone(),
@ -58,6 +63,7 @@ impl<T: TendermintValidator> TendermintImport<T> {
providers: Arc::new(AsyncRwLock::new(None)),
importing_block: Arc::new(RwLock::new(None)),
recheck: Arc::new(RwLock::new(HashSet::new())),
client,
queue: Arc::new(AsyncRwLock::new(None)),
@ -89,7 +95,11 @@ impl<T: TendermintValidator> TendermintImport<T> {
.unwrap_or_else(InherentData::new)
}
async fn check_inherents(&self, block: T::Block) -> Result<(), Error> {
async fn check_inherents(
&mut self,
hash: <T::Block as Block>::Hash,
block: T::Block,
) -> Result<(), Error> {
let inherent_data = self.inherent_data(*block.header().parent_hash()).await;
let err = self
.client
@ -98,10 +108,12 @@ impl<T: TendermintValidator> TendermintImport<T> {
.map_err(|_| Error::Other(BlockError::Fatal.into()))?;
if err.ok() {
self.recheck.write().unwrap().remove(&hash);
Ok(())
} else if err.fatal_error() {
Err(Error::Other(BlockError::Fatal.into()))
} else {
self.recheck.write().unwrap().insert(hash);
Err(Error::Other(BlockError::Temporal.into()))
}
}
@ -173,7 +185,7 @@ impl<T: TendermintValidator> TendermintImport<T> {
}
pub(crate) async fn check<BT>(
&self,
&mut self,
block: &mut BlockImportParams<T::Block, BT>,
) -> Result<(), Error> {
if block.finalized {
@ -194,9 +206,10 @@ impl<T: TendermintValidator> TendermintImport<T> {
// If the block wasn't finalized, verify the origin and validity of its inherents
if !block.finalized {
self.verify_origin(block.header.hash())?;
let hash = block.header.hash();
self.verify_origin(hash)?;
if let Some(body) = block.body.clone() {
self.check_inherents(T::Block::new(block.header.clone(), body)).await?;
self.check_inherents(hash, T::Block::new(block.header.clone(), body)).await?;
}
}