diff --git a/rpc-client-api/src/config.rs b/rpc-client-api/src/config.rs index 50176d309c0c3f..5d49f75bb9d210 100644 --- a/rpc-client-api/src/config.rs +++ b/rpc-client-api/src/config.rs @@ -360,3 +360,12 @@ pub struct RpcContextConfig { pub struct RpcRecentPrioritizationFeesConfig { pub percentile: Option, } + +#[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RpcLatestBlockhashConfig { + #[serde(flatten)] + pub context: RpcContextConfig, + #[serde(default)] + pub rollback: usize, +} diff --git a/rpc-client/src/nonblocking/rpc_client.rs b/rpc-client/src/nonblocking/rpc_client.rs index 7543b4f926bdeb..8e2c7bcf08bf13 100644 --- a/rpc-client/src/nonblocking/rpc_client.rs +++ b/rpc-client/src/nonblocking/rpc_client.rs @@ -4578,6 +4578,26 @@ impl RpcClient { Ok(blockhash) } + pub async fn get_latest_blockhash_with_config( + &self, + config: RpcLatestBlockhashConfig, + ) -> ClientResult<(Hash, u64)> { + let RpcBlockhash { + blockhash, + last_valid_block_height, + } = self + .send::>(RpcRequest::GetLatestBlockhash, json!([config])) + .await? + .value; + let blockhash = blockhash.parse().map_err(|_| { + ClientError::new_with_request( + RpcError::ParseError("Hash".to_string()).into(), + RpcRequest::GetLatestBlockhash, + ) + })?; + Ok((blockhash, last_valid_block_height)) + } + pub async fn get_latest_blockhash_with_commitment( &self, commitment: CommitmentConfig, diff --git a/rpc/src/rpc.rs b/rpc/src/rpc.rs index 4f83ea5bb10be3..96cc9f5b2c33a5 100644 --- a/rpc/src/rpc.rs +++ b/rpc/src/rpc.rs @@ -2184,8 +2184,20 @@ impl JsonRpcRequestProcessor { } } - fn get_latest_blockhash(&self, config: RpcContextConfig) -> Result> { - let bank = self.get_bank_with_config(config)?; + fn get_latest_blockhash( + &self, + config: RpcLatestBlockhashConfig, + ) -> Result> { + let mut bank = self.get_bank_with_config(config.context)?; + if config.rollback > 300 { + return Err(Error::invalid_params("rollback exceeds 300")); + } + for _ in 0..config.rollback { + bank = match bank.parent() { + Some(bank) => bank, + None => return Err(Error::invalid_params("failed to rollback block")), + }; + } let blockhash = bank.last_blockhash(); let last_valid_block_height = bank .get_blockhash_last_valid_block_height(&blockhash) @@ -3422,7 +3434,7 @@ pub mod rpc_full { fn get_latest_blockhash( &self, meta: Self::Metadata, - config: Option, + config: Option, ) -> Result>; #[rpc(meta, name = "isBlockhashValid")] @@ -4098,7 +4110,7 @@ pub mod rpc_full { fn get_latest_blockhash( &self, meta: Self::Metadata, - config: Option, + config: Option, ) -> Result> { debug!("get_latest_blockhash rpc request received"); meta.get_latest_blockhash(config.unwrap_or_default())