use crate::{
declaration::AllocationDeclaration, key::StableKey, slot::AllocationSlotDescriptor,
substrate::StorageSubstrate,
};
use std::sync::Arc;
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct ValidatedAllocations {
inner: Arc<ValidatedState>,
_private: (),
}
#[derive(Clone, Debug, Eq, PartialEq)]
struct ValidatedState {
generation: u64,
declarations: Vec<AllocationDeclaration>,
runtime_fingerprint: Option<String>,
}
impl ValidatedAllocations {
pub(crate) fn new(
generation: u64,
declarations: Vec<AllocationDeclaration>,
runtime_fingerprint: Option<String>,
) -> Self {
Self {
inner: Arc::new(ValidatedState {
generation,
declarations,
runtime_fingerprint,
}),
_private: (),
}
}
pub(crate) fn with_generation(self, generation: u64) -> Self {
let mut state = (*self.inner).clone();
state.generation = generation;
Self {
inner: Arc::new(state),
_private: (),
}
}
#[must_use]
pub fn generation(&self) -> u64 {
self.inner.generation
}
#[must_use]
pub fn declarations(&self) -> &[AllocationDeclaration] {
&self.inner.declarations
}
#[must_use]
pub fn runtime_fingerprint(&self) -> Option<&str> {
self.inner.runtime_fingerprint.as_deref()
}
#[must_use]
pub fn slot_for(&self, key: &StableKey) -> Option<&AllocationSlotDescriptor> {
self.declarations()
.iter()
.find(|declaration| &declaration.stable_key == key)
.map(|declaration| &declaration.slot)
}
}
pub struct AllocationSession<S: StorageSubstrate> {
substrate: S,
validated: ValidatedAllocations,
}
impl<S: StorageSubstrate> AllocationSession<S> {
#[must_use]
pub const fn new(substrate: S, validated: ValidatedAllocations) -> Self {
Self {
substrate,
validated,
}
}
#[must_use]
pub const fn validated(&self) -> &ValidatedAllocations {
&self.validated
}
pub fn open(
&self,
key: &StableKey,
) -> Result<S::MemoryHandle, AllocationSessionError<S::Error>> {
let slot = self
.validated
.slot_for(key)
.ok_or_else(|| AllocationSessionError::UnknownStableKey(key.clone()))?;
self.substrate
.open_slot(slot)
.map_err(AllocationSessionError::Substrate)
}
}
#[derive(Clone, Debug, Eq, thiserror::Error, PartialEq)]
pub enum AllocationSessionError<E> {
#[error("stable key '{0}' was not validated for this allocation session")]
UnknownStableKey(StableKey),
#[error("storage substrate failed to open allocation slot")]
Substrate(E),
}