use crate::{commit::CommitProof, encoding::encoding_error};
use async_trait::async_trait;
use binary_stream::futures::{
BinaryReader, BinaryWriter, Decodable, Encodable,
};
use futures::io::{AsyncRead, AsyncSeek, AsyncWrite};
use rs_merkle::{algorithms::Sha256, MerkleProof};
use std::io::Result;
#[async_trait]
impl Encodable for CommitProof {
async fn encode<W: AsyncWrite + AsyncSeek + Unpin + Send>(
&self,
writer: &mut BinaryWriter<W>,
) -> Result<()> {
writer.write_bytes(self.root).await?;
let proof_bytes = self.proof.to_bytes();
writer.write_u32(proof_bytes.len() as u32).await?;
writer.write_bytes(&proof_bytes).await?;
writer.write_u32(self.length as u32).await?;
writer.write_u32(self.indices.start as u32).await?;
writer.write_u32(self.indices.end as u32).await?;
Ok(())
}
}
#[async_trait]
impl Decodable for CommitProof {
async fn decode<R: AsyncRead + AsyncSeek + Unpin + Send>(
&mut self,
reader: &mut BinaryReader<R>,
) -> Result<()> {
let root_hash: [u8; 32] = reader
.read_bytes(32)
.await?
.as_slice()
.try_into()
.map_err(encoding_error)?;
self.root = root_hash;
let length = reader.read_u32().await?;
let proof_bytes = reader.read_bytes(length as usize).await?;
let proof = MerkleProof::<Sha256>::from_bytes(&proof_bytes)
.map_err(encoding_error)?;
self.proof = proof;
self.length = reader.read_u32().await? as usize;
let start = reader.read_u32().await?;
let end = reader.read_u32().await?;
self.indices = start as usize..end as usize;
Ok(())
}
}