mirror of
https://github.com/logos-blockchain/logos-execution-zone.git
synced 2026-03-25 19:53:07 +00:00
325 lines
8.7 KiB
Rust
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(_, _))
|
|
));
|
|
}
|
|
}
|