logos-execution-zone/sc_core/src/private_state.rs
Rostyslav Tyshko 01e3c0fe54 fmt
2025-05-02 17:28:14 -04:00

325 lines
8.7 KiB
Rust

use std::collections::BTreeMap;
use serde::{de::Error, Deserialize, Serialize};
pub const PRIVATE_BLOB_SIZE: usize = 32;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct PrivateDataBlob(pub [u8; PRIVATE_BLOB_SIZE]);
pub type PrivateSCState = BTreeMap<usize, PrivateDataBlob>;
#[derive(thiserror::Error, Debug)]
pub enum PrivateStateError {
#[error("Trying to read from slot too big: Read slot {0}, max_slot {1}")]
ReadSizeMismatch(usize, usize),
#[error("Can not write empty bytes into state")]
EmptyWrite,
}
impl From<[u8; PRIVATE_BLOB_SIZE]> for PrivateDataBlob {
fn from(value: [u8; PRIVATE_BLOB_SIZE]) -> Self {
Self(value)
}
}
impl Serialize for PrivateDataBlob {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
let data_vec = self.0.to_vec();
data_vec.serialize(serializer)
}
}
impl AsRef<[u8]> for PrivateDataBlob {
fn as_ref(&self) -> &[u8] {
self.0.as_ref()
}
}
impl<'de> Deserialize<'de> for PrivateDataBlob {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let data_vec = Vec::<u8>::deserialize(deserializer)?;
let chunk: [u8; PRIVATE_BLOB_SIZE] = data_vec
.try_into()
.map_err(|data| {
anyhow::anyhow!("failed to fit vec {data:?} to {:?}", PRIVATE_BLOB_SIZE)
})
.map_err(D::Error::custom)?;
Ok(Self(chunk))
}
}
impl PrivateDataBlob {
pub fn to_vec(&self) -> Vec<u8> {
self.0.to_vec()
}
}
///Produce `DataBlob` from vector of size <= `PRIVATE_BLOB_SIZE`
///
///Extends to `PRIVATE_BLOB_SIZE`, if necessary.
///
///Panics, if size > `PRIVATE_BLOB_SIZE`
pub fn produce_blob_from_fit_vec(data: Vec<u8>) -> PrivateDataBlob {
let data_len = data.len();
assert!(data_len <= PRIVATE_BLOB_SIZE);
let mut blob: PrivateDataBlob = [0; PRIVATE_BLOB_SIZE].into();
for (idx, item) in data.into_iter().enumerate() {
blob.0[idx] = item
}
blob
}
///Produce `DataBlob` from slice of size <= `PRIVATE_BLOB_SIZE`
///
///Extends to `PRIVATE_BLOB_SIZE`, if necessary.
///
///Panics, if size > `PRIVATE_BLOB_SIZE`
pub fn produce_blob_from_fit_slice(data: &[u8]) -> PrivateDataBlob {
let data_len = data.len();
assert!(data_len <= PRIVATE_BLOB_SIZE);
let mut blob: PrivateDataBlob = [0; PRIVATE_BLOB_SIZE].into();
for (idx, item) in data.into_iter().enumerate() {
blob.0[idx] = *item
}
blob
}
pub fn calculate_offset_slot(offset: usize) -> usize {
offset / PRIVATE_BLOB_SIZE
}
pub fn max_slot(state: &PrivateSCState) -> usize {
*state.keys().max().unwrap_or(&0)
}
///Read at least `num` bytes from the start of a state
pub fn read_num_bytes_start(
state: &PrivateSCState,
num: usize,
) -> Result<Vec<PrivateDataBlob>, PrivateStateError> {
let mut resp = vec![];
let max_offset_slot = calculate_offset_slot(num);
let max_slot_state = max_slot(state);
if max_offset_slot > max_slot_state {
return Err(PrivateStateError::ReadSizeMismatch(
max_offset_slot,
max_slot_state,
));
}
for i in 0..max_offset_slot {
resp.push(*state.get(&i).unwrap());
}
Ok(resp)
}
///Read at least `num` bytes from the `offset` slot
pub fn read_num_bytes_offset(
state: &PrivateSCState,
num: usize,
offset: usize,
) -> Result<Vec<PrivateDataBlob>, PrivateStateError> {
let mut resp = vec![];
let max_offset_slot = offset + calculate_offset_slot(num);
let max_slot_state = max_slot(state);
if max_offset_slot > max_slot_state {
return Err(PrivateStateError::ReadSizeMismatch(
max_offset_slot,
max_slot_state,
));
}
for i in offset..max_offset_slot {
resp.push(*state.get(&i).unwrap());
}
Ok(resp)
}
///Write at least `bytes.len()` bytes at the end of the state
///
/// Returns new last slot
pub fn write_num_bytes_append(
state: &mut PrivateSCState,
bytes: Vec<u8>,
) -> Result<usize, PrivateStateError> {
if bytes.is_empty() {
return Err(PrivateStateError::EmptyWrite);
}
let mut max_slot_state = max_slot(state) + 1;
let mut curr = 0;
while (curr + PRIVATE_BLOB_SIZE) < bytes.len() {
let data_blob = produce_blob_from_fit_slice(&bytes[curr..(curr + PRIVATE_BLOB_SIZE)]);
state.insert(max_slot_state, data_blob);
curr += PRIVATE_BLOB_SIZE;
max_slot_state += 1;
}
let data_blob = produce_blob_from_fit_slice(&bytes[curr..(bytes.len())]);
state.insert(max_slot_state, data_blob);
Ok(max_slot_state)
}
/// Rewrite at least `bytes.len()` bytes starting from the offset slot
///
/// Returns last (re)written slot
pub fn write_num_bytes_rewrite(
state: &mut PrivateSCState,
bytes: Vec<u8>,
offset: usize,
) -> Result<usize, PrivateStateError> {
if bytes.is_empty() {
return Err(PrivateStateError::EmptyWrite);
}
let mut curr_slot = offset;
let mut curr = 0;
while (curr + PRIVATE_BLOB_SIZE) < bytes.len() {
let data_blob = produce_blob_from_fit_slice(&bytes[curr..(curr + PRIVATE_BLOB_SIZE)]);
state.insert(curr_slot, data_blob);
curr += PRIVATE_BLOB_SIZE;
curr_slot += 1;
}
let data_blob = produce_blob_from_fit_slice(&bytes[curr..(bytes.len())]);
state.insert(curr_slot, data_blob);
Ok(curr_slot)
}
#[cfg(test)]
mod tests {
use super::*;
use serde_json;
#[test]
fn test_blob_from_fit_vec_and_slice() {
let data = vec![1u8, 2, 3];
let blob_from_vec = produce_blob_from_fit_vec(data.clone());
let blob_from_slice = produce_blob_from_fit_slice(&data);
assert_eq!(blob_from_vec, blob_from_slice);
assert_eq!(blob_from_vec.0[0..3], [1, 2, 3]);
assert_eq!(blob_from_vec.0[3..], [0u8; PRIVATE_BLOB_SIZE - 3]);
}
#[test]
#[should_panic]
fn test_blob_from_fit_vec_panic() {
let data = vec![1u8; PRIVATE_BLOB_SIZE + 1];
let _ = produce_blob_from_fit_vec(data);
}
#[test]
#[should_panic]
fn test_blob_from_fit_slice_panic() {
let data = vec![1u8; PRIVATE_BLOB_SIZE + 1];
let _ = produce_blob_from_fit_slice(&data);
}
#[test]
fn test_serialization_roundtrip() {
let data = [42u8; PRIVATE_BLOB_SIZE];
let blob = PrivateDataBlob::from(data);
let serialized = serde_json::to_string(&blob).unwrap();
let deserialized: PrivateDataBlob = serde_json::from_str(&serialized).unwrap();
assert_eq!(blob, deserialized);
}
#[test]
fn test_calculate_offset_slot() {
assert_eq!(calculate_offset_slot(0), 0);
assert_eq!(calculate_offset_slot(PRIVATE_BLOB_SIZE), 1);
assert_eq!(calculate_offset_slot(PRIVATE_BLOB_SIZE * 2 - 1), 1);
}
#[test]
fn test_max_slot_empty_and_nonempty() {
let empty: PrivateSCState = BTreeMap::new();
assert_eq!(max_slot(&empty), 0);
let mut state = BTreeMap::new();
state.insert(3, produce_blob_from_fit_vec(vec![1, 2, 3]));
state.insert(5, produce_blob_from_fit_vec(vec![4, 5, 6]));
assert_eq!(max_slot(&state), 5);
}
#[test]
fn test_write_append_and_read() {
let mut state = PrivateSCState::new();
let data = vec![1u8; PRIVATE_BLOB_SIZE * 2 + 10];
let last_slot = write_num_bytes_append(&mut state, data.clone()).unwrap();
assert_eq!(last_slot, 3);
let read = read_num_bytes_start(&state, last_slot).unwrap();
assert_eq!(read.len(), 0);
}
#[test]
fn test_write_rewrite() {
let mut state = PrivateSCState::new();
let data1 = vec![1u8; PRIVATE_BLOB_SIZE * 2];
write_num_bytes_append(&mut state, data1).unwrap();
let rewrite_data = vec![2u8; PRIVATE_BLOB_SIZE];
let last_slot = write_num_bytes_rewrite(&mut state, rewrite_data.clone(), 1).unwrap();
assert_eq!(last_slot, 1);
assert_eq!(state.get(&1).unwrap().0[0], 2);
}
#[test]
fn test_empty_write_error() {
let mut state = PrivateSCState::new();
let append_result = write_num_bytes_append(&mut state, vec![]);
assert!(matches!(append_result, Err(PrivateStateError::EmptyWrite)));
let rewrite_result = write_num_bytes_rewrite(&mut state, vec![], 0);
assert!(matches!(rewrite_result, Err(PrivateStateError::EmptyWrite)));
}
#[test]
fn test_read_size_mismatch_error() {
let state = PrivateSCState::new();
let result = read_num_bytes_start(&state, PRIVATE_BLOB_SIZE * 2);
assert!(matches!(
result,
Err(PrivateStateError::ReadSizeMismatch(_, _))
));
}
}