celestia-rpc 1.0.0

A collection of traits for interacting with Celestia data availability nodes RPC
Documentation
//! celestia-node rpc types and methods related to shares

use std::future::Future;
use std::marker::{Send, Sync};

use celestia_types::namespace_data::NamespaceData;
use celestia_types::nmt::Namespace;
use celestia_types::sample::{RawSample, Sample, SampleId};
use celestia_types::{ExtendedDataSquare, RawShare, Share, ShareProof};
use jsonrpsee::core::client::{ClientT, Error};
use jsonrpsee::proc_macros::rpc;
use serde::{Deserialize, Serialize};

use crate::custom_client_error;

/// Response type for [`ShareClient::share_get_range`].
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "PascalCase")]
pub struct GetRangeResponse {
    /// Shares contained in given range.
    pub shares: Vec<Share>,
    /// Proof of inclusion of the shares.
    pub proof: ShareProof,
}

/// Side of a row within the EDS.
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "UPPERCASE")]
pub enum RowSide {
    /// The row data is on the left of the EDS (i.e. in the ODS).
    Left,
    /// The row data is on the right of the EDS (i.e. it's parity data).
    Right,
    /// The row contains both the original data and the parity data.
    Both,
}

/// Response type for [`ShareClient::share_get_row`].
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
pub struct GetRowResponse {
    /// Shares contained in given range.
    pub shares: Vec<Share>,
    /// Side of the row within the EDS.
    pub side: RowSide,
}

/// Position of a sample in a 2D grid.
#[derive(Debug, Copy, Clone, PartialEq, Serialize, Deserialize)]
pub struct SampleCoordinates {
    /// Row index of the sample.
    pub row: u16,
    /// Column index of the sample.
    #[serde(rename = "col")]
    pub column: u16,
}

impl SampleCoordinates {
    /// Create new `SampleCoordinates` base on row and column.
    pub fn new(row: u16, column: u16) -> Self {
        SampleCoordinates { row, column }
    }
}

impl From<(u16, u16)> for SampleCoordinates {
    fn from(value: (u16, u16)) -> Self {
        SampleCoordinates::new(value.0, value.1)
    }
}

mod rpc {
    use super::*;
    use celestia_types::eds::RawExtendedDataSquare;
    use jsonrpsee::core::RpcResult;

    #[derive(Debug, Clone, PartialEq, Deserialize, Serialize)]
    pub struct RawGetRowResponse {
        pub(crate) shares: Vec<RawShare>,
        pub(crate) side: RowSide,
    }

    /// Share RPC methods.
    #[rpc(client, server, namespace = "share", namespace_separator = ".")]
    trait Share {
        /// See [`crate::ShareClient::share_get_eds`].
        #[method(name = "GetEDS")]
        async fn share_get_eds(&self, height: u64) -> RpcResult<RawExtendedDataSquare>;

        /// See [`crate::ShareClient::share_get_range`].
        #[method(name = "GetRange")]
        async fn share_get_range(
            &self,
            height: u64,
            start: u64,
            end: u64,
        ) -> RpcResult<GetRangeResponse>; // was Result<_, Error>

        /// See [`crate::ShareClient::share_get_samples`].
        #[method(name = "GetSamples")]
        async fn share_get_samples(
            &self,
            height: u64,
            indices: Vec<SampleCoordinates>, // was &[SampleCoordinates]
        ) -> RpcResult<Vec<RawSample>>; // was Result<_, Error>

        /// See [`crate::ShareClient::share_get_row`].
        #[method(name = "GetRow")]
        async fn share_get_row(&self, height: u64, row: u16) -> RpcResult<RawGetRowResponse>;

        /// See [`crate::ShareClient::share_get_share`].
        #[method(name = "GetShare")]
        async fn share_get_share(&self, height: u64, row: u16, col: u16) -> RpcResult<RawShare>;

        /// See [`crate::ShareClient::share_get_namespace_data`].
        #[method(name = "GetNamespaceData")]
        async fn share_get_namespace_data(
            &self,
            height: u64,
            namespace: Namespace,
        ) -> RpcResult<NamespaceData>;

        /// See [`crate::ShareClient::share_shares_available`].
        #[method(name = "SharesAvailable")]
        async fn share_shares_available(&self, height: u64) -> RpcResult<()>;
    }
}

/// Client implementation for the `Share` RPC API.
pub trait ShareClient: ClientT {
    /// GetEDS gets the full EDS identified at the specified height.
    fn share_get_eds<'a, 'fut>(
        &'a self,
        height: u64,
    ) -> impl Future<Output = Result<ExtendedDataSquare, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        async move {
            let raw_eds = rpc::ShareClient::share_get_eds(self, height).await?;
            // Correct `Share` construction and validation is done inside `ExtendedDataSquare::from_raw`
            ExtendedDataSquare::from_raw(raw_eds).map_err(custom_client_error)
        }
    }

    /// Retrieves a list of shares and their corresponding proof.
    ///
    /// The start and end index ignores parity shares and corresponds to ODS.
    fn share_get_range<'a, 'fut>(
        &'a self,
        height: u64,
        start: u64,
        end: u64,
    ) -> impl Future<Output = Result<GetRangeResponse, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        rpc::ShareClient::share_get_range(self, height, start, end)
    }
    /// Retrieves multiple shares from the [`ExtendedDataSquare`] at the given height
    /// at the given sample coordinates.
    ///
    /// `coordinates` is a list of `(row, column)`.
    fn share_get_samples<'a, 'fut, I, C>(
        &'a self,
        height: u64,
        coordinates: I,
    ) -> impl Future<Output = Result<Vec<Sample>, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
        I: IntoIterator<Item = C>,
        C: Into<SampleCoordinates>,
    {
        let coordinates = coordinates
            .into_iter()
            .map(|c| c.into())
            .collect::<Vec<_>>();

        async move {
            let raw_samples =
                rpc::ShareClient::share_get_samples(self, height, coordinates.clone()).await?;
            let mut samples = Vec::with_capacity(raw_samples.len());

            for (coords, raw_sample) in coordinates.iter().zip(raw_samples.into_iter()) {
                let sample_id = SampleId::new(coords.row, coords.column, height)
                    .map_err(custom_client_error)?;

                // Correct `Share` construction is done inside `Sample::from_raw`
                let sample =
                    Sample::from_raw(sample_id, raw_sample).map_err(custom_client_error)?;

                samples.push(sample);
            }

            Ok(samples)
        }
    }

    /// GetShare gets the list of shares in a single row.
    fn share_get_row<'a, 'fut>(
        &'a self,
        height: u64,
        square_width: u16,
        row: u16,
    ) -> impl Future<Output = Result<GetRowResponse, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        async move {
            let resp = rpc::ShareClient::share_get_row(self, height, row).await?;

            let expected_len = match resp.side {
                RowSide::Left | RowSide::Right => square_width / 2,
                RowSide::Both => square_width,
            };

            if resp.shares.len() != usize::from(expected_len) {
                return Err(Error::Custom(format!(
                    "GetRowResponse::shares should have length of {} but received {}",
                    expected_len,
                    resp.shares.len(),
                )));
            }

            let shares = resp
                .shares
                .into_iter()
                .enumerate()
                .map(|(relative_col, raw_share)| {
                    let relative_col = u16::try_from(relative_col)
                        .expect("square_width and expected_len already validated this");

                    let col = match resp.side {
                        RowSide::Left | RowSide::Both => relative_col,
                        RowSide::Right => (square_width / 2) + relative_col,
                    };

                    let share = if is_ods_square(row, col, square_width) {
                        Share::from_raw(&raw_share.data)
                    } else {
                        Share::parity(&raw_share.data)
                    }
                    .map_err(custom_client_error)?;

                    Ok(share)
                })
                .collect::<Result<Vec<_>, Error>>()?;

            Ok(GetRowResponse {
                shares,
                side: resp.side,
            })
        }
    }

    /// GetShare gets a Share by coordinates in EDS.
    fn share_get_share<'a, 'fut>(
        &'a self,
        height: u64,
        square_width: u16,
        row: u16,
        col: u16,
    ) -> impl Future<Output = Result<Share, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        async move {
            let share = rpc::ShareClient::share_get_share(self, height, row, col).await?;
            let share = if is_ods_square(row, col, square_width) {
                Share::from_raw(&share.data)
            } else {
                Share::parity(&share.data)
            }
            .map_err(custom_client_error)?;

            Ok(share)
        }
    }

    /// GetNamespaceData gets all shares from an EDS within the given namespace.
    ///
    /// Shares are returned in a row-by-row order if the namespace spans multiple rows.
    ///
    /// PARITY and TAIL PADDING namespaces are not allowed.
    fn share_get_namespace_data<'a, 'fut>(
        &'a self,
        height: u64,
        namespace: Namespace,
    ) -> impl Future<Output = Result<NamespaceData, Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        rpc::ShareClient::share_get_namespace_data(self, height, namespace)
    }

    /// SharesAvailable subjectively validates if Shares committed to the given Root are available on the Network.
    fn share_shares_available<'a, 'fut>(
        &'a self,
        height: u64,
    ) -> impl Future<Output = Result<(), Error>> + Send + 'fut
    where
        'a: 'fut,
        Self: Sized + Sync + 'fut,
    {
        rpc::ShareClient::share_shares_available(self, height)
    }
}

impl<T> ShareClient for T where T: ClientT {}

fn is_ods_square(row: u16, column: u16, square_width: u16) -> bool {
    let ods_width = square_width / 2;
    row < ods_width && column < ods_width
}

/// Server trait for Share RPC endpoints.
pub trait ShareServer: rpc::ShareServer {}

impl<T> ShareServer for T where T: rpc::ShareServer {}

pub use rpc::ShareServer as ShareRpcServer;