Update get blobs endpoint to return a list of BlobSidecars (#4109)

* Update get blobs endpoint to return BlobSidecarList

* Update code comment

* Update blob retrieval to return BlobSidecarList without Arc

* Remove usage of BlobSidecarList type alias to avoid code conflicts

* Add clippy allow exception
This commit is contained in:
Jimmy Chen 2023-03-22 01:56:32 +11:00 committed by GitHub
parent 78414333a2
commit b40dceaae9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 77 additions and 55 deletions

View File

@ -1057,6 +1057,23 @@ impl<T: BeaconChainTypes> BeaconChain<T> {
.map(Some) .map(Some)
} }
// FIXME(jimmy): temporary method added to unblock API work. This method will be replaced by
// the `get_blobs` method below once the new blob sidecar structure (`BlobSidecarList`) is
// implemented in that method.
#[allow(clippy::type_complexity)] // FIXME: this will be fixed by the `BlobSidecarList` alias in Sean's PR
pub fn get_blob_sidecar_list(
&self,
_block_root: &Hash256,
_data_availability_boundary: Epoch,
) -> Result<
Option<
VariableList<Arc<BlobSidecar<T::EthSpec>>, <T::EthSpec as EthSpec>::MaxBlobsPerBlock>,
>,
Error,
> {
unimplemented!("update to use the updated `get_blobs` method instead once this PR is merged: https://github.com/sigp/lighthouse/pull/4104")
}
/// Returns the blobs at the given root, if any. /// Returns the blobs at the given root, if any.
/// ///
/// Returns `Ok(None)` if the blobs and associated block are not found. /// Returns `Ok(None)` if the blobs and associated block are not found.

View File

@ -1,10 +1,10 @@
use crate::{state_id::checkpoint_slot_and_execution_optimistic, ExecutionOptimistic}; use crate::{state_id::checkpoint_slot_and_execution_optimistic, ExecutionOptimistic};
use beacon_chain::{BeaconChain, BeaconChainError, BeaconChainTypes, WhenSlotSkipped}; use beacon_chain::{BeaconChain, BeaconChainError, BeaconChainTypes, WhenSlotSkipped};
use eth2::types::BlockId as CoreBlockId; use eth2::types::{BlockId as CoreBlockId, VariableList};
use std::fmt; use std::fmt;
use std::str::FromStr; use std::str::FromStr;
use std::sync::Arc; use std::sync::Arc;
use types::{BlobsSidecar, Hash256, SignedBeaconBlock, SignedBlindedBeaconBlock, Slot}; use types::{BlobSidecar, EthSpec, Hash256, SignedBeaconBlock, SignedBlindedBeaconBlock, Slot};
/// Wraps `eth2::types::BlockId` and provides a simple way to obtain a block or root for a given /// Wraps `eth2::types::BlockId` and provides a simple way to obtain a block or root for a given
/// `BlockId`. /// `BlockId`.
@ -212,19 +212,22 @@ impl BlockId {
} }
} }
/// Return the `BlobsSidecar` identified by `self`. /// Return the `BlobSidecarList` identified by `self`.
pub async fn blobs_sidecar<T: BeaconChainTypes>( pub async fn blob_sidecar_list<T: BeaconChainTypes>(
&self, &self,
chain: &BeaconChain<T>, chain: &BeaconChain<T>,
) -> Result<Arc<BlobsSidecar<T::EthSpec>>, warp::Rejection> { ) -> Result<
VariableList<Arc<BlobSidecar<T::EthSpec>>, <T::EthSpec as EthSpec>::MaxBlobsPerBlock>,
warp::Rejection,
> {
let root = self.root(chain)?.0; let root = self.root(chain)?.0;
let Some(data_availability_boundary) = chain.data_availability_boundary() else { let Some(data_availability_boundary) = chain.data_availability_boundary() else {
return Err(warp_utils::reject::custom_not_found("Eip4844 fork disabled".into())); return Err(warp_utils::reject::custom_not_found("Deneb fork disabled".into()));
}; };
match chain.get_blobs(&root, data_availability_boundary) { match chain.get_blob_sidecar_list(&root, data_availability_boundary) {
Ok(Some(blob)) => Ok(Arc::new(blob)), Ok(Some(blobs)) => Ok(blobs),
Ok(None) => Err(warp_utils::reject::custom_not_found(format!( Ok(None) => Err(warp_utils::reject::custom_not_found(format!(
"Blob with block root {} is not in the store", "No blobs with block root {} found in the store",
root root
))), ))),
Err(e) => Err(warp_utils::reject::beacon_chain_error(e)), Err(e) => Err(warp_utils::reject::beacon_chain_error(e)),

View File

@ -1293,6 +1293,45 @@ pub fn serve<T: BeaconChainTypes>(
}, },
); );
/*
* beacon/blobs
*/
// GET beacon/blobs/{block_id}
let get_blobs = eth_v1
.and(warp::path("beacon"))
.and(warp::path("blobs"))
.and(block_id_or_err)
.and(warp::path::end())
.and(chain_filter.clone())
.and(warp::header::optional::<api_types::Accept>("accept"))
.and_then(
|block_id: BlockId,
chain: Arc<BeaconChain<T>>,
accept_header: Option<api_types::Accept>| {
async move {
let blob_sidecar_list = block_id.blob_sidecar_list(&chain).await?;
match accept_header {
Some(api_types::Accept::Ssz) => Response::builder()
.status(200)
.header("Content-Type", "application/octet-stream")
.body(blob_sidecar_list.as_ssz_bytes().into())
.map_err(|e| {
warp_utils::reject::custom_server_error(format!(
"failed to create response: {}",
e
))
}),
_ => Ok(warp::reply::json(&api_types::GenericResponse::from(
blob_sidecar_list,
))
.into_response()),
}
}
},
);
/* /*
* beacon/pool * beacon/pool
*/ */
@ -3498,41 +3537,6 @@ pub fn serve<T: BeaconChainTypes>(
) )
}); });
// GET lighthouse/beacon/blobs_sidecars/{block_id}
let get_lighthouse_blobs_sidecars = warp::path("lighthouse")
.and(warp::path("beacon"))
.and(warp::path("blobs_sidecars"))
.and(block_id_or_err)
.and(warp::path::end())
.and(chain_filter.clone())
.and(warp::header::optional::<api_types::Accept>("accept"))
.and_then(
|block_id: BlockId,
chain: Arc<BeaconChain<T>>,
accept_header: Option<api_types::Accept>| {
async move {
let blobs_sidecar = block_id.blobs_sidecar(&chain).await?;
match accept_header {
Some(api_types::Accept::Ssz) => Response::builder()
.status(200)
.header("Content-Type", "application/octet-stream")
.body(blobs_sidecar.as_ssz_bytes().into())
.map_err(|e| {
warp_utils::reject::custom_server_error(format!(
"failed to create response: {}",
e
))
}),
_ => Ok(warp::reply::json(&api_types::GenericResponse::from(
blobs_sidecar,
))
.into_response()),
}
}
},
);
let get_events = eth_v1 let get_events = eth_v1
.and(warp::path("events")) .and(warp::path("events"))
.and(warp::path::end()) .and(warp::path::end())
@ -3627,6 +3631,7 @@ pub fn serve<T: BeaconChainTypes>(
.uor(get_beacon_block_attestations) .uor(get_beacon_block_attestations)
.uor(get_beacon_blinded_block) .uor(get_beacon_blinded_block)
.uor(get_beacon_block_root) .uor(get_beacon_block_root)
.uor(get_blobs)
.uor(get_beacon_pool_attestations) .uor(get_beacon_pool_attestations)
.uor(get_beacon_pool_attester_slashings) .uor(get_beacon_pool_attester_slashings)
.uor(get_beacon_pool_proposer_slashings) .uor(get_beacon_pool_proposer_slashings)
@ -3672,7 +3677,6 @@ pub fn serve<T: BeaconChainTypes>(
.uor(get_lighthouse_attestation_performance) .uor(get_lighthouse_attestation_performance)
.uor(get_lighthouse_block_packing_efficiency) .uor(get_lighthouse_block_packing_efficiency)
.uor(get_lighthouse_merge_readiness) .uor(get_lighthouse_merge_readiness)
.uor(get_lighthouse_blobs_sidecars.boxed())
.uor(get_events) .uor(get_events)
.recover(warp_utils::reject::handle_rejection), .recover(warp_utils::reject::handle_rejection),
) )

View File

@ -658,15 +658,13 @@ impl BeaconNodeHttpClient {
Ok(path) Ok(path)
} }
/// Path for `lighthouse/beacon/blobs_sidecars/{block_id}` /// Path for `v1/beacon/blobs/{block_id}`
pub fn get_blobs_sidecar_path(&self, block_id: BlockId) -> Result<Url, Error> { pub fn get_blobs_path(&self, block_id: BlockId) -> Result<Url, Error> {
let mut path = self.server.full.clone(); let mut path = self.eth_path(V1)?;
path.path_segments_mut() path.path_segments_mut()
.map_err(|()| Error::InvalidUrl(self.server.clone()))? .map_err(|()| Error::InvalidUrl(self.server.clone()))?
.push("lighthouse")
.push("beacon") .push("beacon")
.push("blobs_sidecars") .push("blobs")
.push(&block_id.to_string()); .push(&block_id.to_string());
Ok(path) Ok(path)
} }
@ -698,14 +696,14 @@ impl BeaconNodeHttpClient {
Ok(Some(response.json().await?)) Ok(Some(response.json().await?))
} }
/// `GET lighthouse/beacon/blobs_sidecars/{block_id}` /// `GET v1/beacon/blobs/{block_id}`
/// ///
/// Returns `Ok(None)` on a 404 error. /// Returns `Ok(None)` on a 404 error.
pub async fn get_blobs_sidecar<T: EthSpec>( pub async fn get_blobs<T: EthSpec>(
&self, &self,
block_id: BlockId, block_id: BlockId,
) -> Result<Option<GenericResponse<BlobsSidecar<T>>>, Error> { ) -> Result<Option<GenericResponse<BlobSidecarList<T>>>, Error> {
let path = self.get_blobs_sidecar_path(block_id)?; let path = self.get_blobs_path(block_id)?;
let response = match self.get_response(path, |b| b).await.optional()? { let response = match self.get_response(path, |b| b).await.optional()? {
Some(res) => res, Some(res) => res,
None => return Ok(None), None => return Ok(None),