Struct KeyValueStoreView

Source
pub struct KeyValueStoreView<C> { /* private fields */ }
Expand description

A view that represents the functions of KeyValueStore.

Comment on the data set: In order to work, the view needs to store the updates and deleted prefixes. The updates and deleted prefixes have to be coherent. This means:

  • If an index is deleted by one in deleted prefixes then it should not be present in the updates at all.
  • DeletePrefix::key_prefix should not dominate anyone. That is if we have [0,2] then we should not have [0,2,3] since it would be dominated by the preceding.

With that we have:

  • in order to test if an index is deleted by a prefix we compute the highest deleted prefix dp such that dp <= index. If dp is indeed a prefix then we conclude that index is deleted, otherwise not. The no domination is essential here.

Implementations§

Source§

impl<C: Context> KeyValueStoreView<C>

Source

pub fn total_size(&self) -> SizeData

Getting the total sizes that will be used for keys and values when stored

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0, 1, 2, 3, 4]).await.unwrap();
let total_size = view.total_size();
assert_eq!(total_size, SizeData { key: 2, value: 5 });
Source

pub async fn for_each_index_while<F>(&self, f: F) -> Result<(), ViewError>
where F: FnMut(&[u8]) -> Result<bool, ViewError> + Send,

Applies the function f over all indices. If the function f returns false, then the loop ends prematurely.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
view.insert(vec![0, 3], vec![0]).await.unwrap();
let mut count = 0;
view.for_each_index_while(|_key| {
    count += 1;
    Ok(count < 2)
})
.await
.unwrap();
assert_eq!(count, 2);
Source

pub async fn for_each_index<F>(&self, f: F) -> Result<(), ViewError>
where F: FnMut(&[u8]) -> Result<(), ViewError> + Send,

Applies the function f over all indices.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
view.insert(vec![0, 3], vec![0]).await.unwrap();
let mut count = 0;
view.for_each_index(|_key| {
    count += 1;
    Ok(())
})
.await
.unwrap();
assert_eq!(count, 3);
Source

pub async fn for_each_index_value_while<F>(&self, f: F) -> Result<(), ViewError>
where F: FnMut(&[u8], &[u8]) -> Result<bool, ViewError> + Send,

Applies the function f over all index/value pairs. If the function f returns false then the loop ends prematurely.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
let mut values = Vec::new();
view.for_each_index_value_while(|_key, value| {
    values.push(value.to_vec());
    Ok(values.len() < 1)
})
.await
.unwrap();
assert_eq!(values, vec![vec![0]]);
Source

pub async fn for_each_index_value<F>(&self, f: F) -> Result<(), ViewError>
where F: FnMut(&[u8], &[u8]) -> Result<(), ViewError> + Send,

Applies the function f over all index/value pairs.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
let mut part_keys = Vec::new();
view.for_each_index_while(|key| {
    part_keys.push(key.to_vec());
    Ok(part_keys.len() < 1)
})
.await
.unwrap();
assert_eq!(part_keys, vec![vec![0, 1]]);
Source

pub async fn indices(&self) -> Result<Vec<Vec<u8>>, ViewError>

Returns the list of indices in lexicographic order.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
let indices = view.indices().await.unwrap();
assert_eq!(indices, vec![vec![0, 1], vec![0, 2]]);
Source

pub async fn index_values(&self) -> Result<Vec<(Vec<u8>, Vec<u8>)>, ViewError>

Returns the list of indices and values in lexicographic order.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
let key_values = view.indices().await.unwrap();
assert_eq!(key_values, vec![vec![0, 1], vec![0, 2]]);
Source

pub async fn count(&self) -> Result<usize, ViewError>

Returns the number of entries.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![0]).await.unwrap();
view.insert(vec![0, 2], vec![0]).await.unwrap();
let count = view.count().await.unwrap();
assert_eq!(count, 2);
Source

pub async fn get(&self, index: &[u8]) -> Result<Option<Vec<u8>>, ViewError>

Obtains the value at the given index, if any.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![42]).await.unwrap();
assert_eq!(view.get(&[0, 1]).await.unwrap(), Some(vec![42]));
assert_eq!(view.get(&[0, 2]).await.unwrap(), None);
Source

pub async fn contains_key(&self, index: &[u8]) -> Result<bool, ViewError>

Tests whether the store contains a specific index.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![42]).await.unwrap();
assert!(view.contains_key(&[0, 1]).await.unwrap());
assert!(!view.contains_key(&[0, 2]).await.unwrap());
Source

pub async fn contains_keys( &self, indices: &[Vec<u8>], ) -> Result<Vec<bool>, ViewError>

Tests whether the view contains a range of indices

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![42]).await.unwrap();
let keys = vec![vec![0, 1], vec![0, 2]];
let results = view.contains_keys(&keys).await.unwrap();
assert_eq!(results, vec![true, false]);
Source

pub async fn multi_get( &self, indices: &[Vec<u8>], ) -> Result<Vec<Option<Vec<u8>>>, ViewError>

Obtains the values of a range of indices

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![42]).await.unwrap();
assert_eq!(
    view.multi_get(&[vec![0, 1], vec![0, 2]]).await.unwrap(),
    vec![Some(vec![42]), None]
);
Source

pub async fn write_batch(&mut self, batch: Batch) -> Result<(), ViewError>

Applies the given batch of crate::common::WriteOperation.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
view.insert(vec![3, 4], vec![42]).await.unwrap();
let mut batch = Batch::new();
batch.delete_key_prefix(vec![0]);
view.write_batch(batch).await.unwrap();
let key_values = view.find_key_values_by_prefix(&[0]).await.unwrap();
assert_eq!(key_values, vec![]);
Source

pub async fn insert( &mut self, index: Vec<u8>, value: Vec<u8>, ) -> Result<(), ViewError>

Sets or inserts a value.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
assert_eq!(view.get(&[0, 1]).await.unwrap(), Some(vec![34]));
Source

pub async fn remove(&mut self, index: Vec<u8>) -> Result<(), ViewError>

Removes a value. If absent then the action has no effect.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
view.remove(vec![0, 1]).await.unwrap();
assert_eq!(view.get(&[0, 1]).await.unwrap(), None);
Source

pub async fn remove_by_prefix( &mut self, key_prefix: Vec<u8>, ) -> Result<(), ViewError>

Deletes a key prefix.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
view.remove_by_prefix(vec![0]).await.unwrap();
assert_eq!(view.get(&[0, 1]).await.unwrap(), None);
Source

pub async fn find_keys_by_prefix( &self, key_prefix: &[u8], ) -> Result<Vec<Vec<u8>>, ViewError>

Iterates over all the keys matching the given prefix. The prefix is not included in the returned keys.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
view.insert(vec![3, 4], vec![42]).await.unwrap();
let keys = view.find_keys_by_prefix(&[0]).await.unwrap();
assert_eq!(keys, vec![vec![1]]);
Source

pub async fn find_key_values_by_prefix( &self, key_prefix: &[u8], ) -> Result<Vec<(Vec<u8>, Vec<u8>)>, ViewError>

Iterates over all the key-value pairs, for keys matching the given prefix. The prefix is not included in the returned keys.

let mut view = KeyValueStoreView::load(context).await.unwrap();
view.insert(vec![0, 1], vec![34]).await.unwrap();
view.insert(vec![3, 4], vec![42]).await.unwrap();
let key_values = view.find_key_values_by_prefix(&[0]).await.unwrap();
assert_eq!(key_values, vec![(vec![1], vec![34])]);

Trait Implementations§

Source§

impl<C> Allocative for KeyValueStoreView<C>

Source§

fn visit<'allocative_a, 'allocative_b: 'allocative_a>( &self, visitor: &'allocative_a mut Visitor<'allocative_b>, )

Source§

impl<C: Context> ClonableView for KeyValueStoreView<C>

Source§

fn clone_unchecked(&mut self) -> Result<Self, ViewError>

Creates a clone of this view, sharing the underlying storage context but prone to data races which can corrupt the view state.
Source§

impl<C: Debug> Debug for KeyValueStoreView<C>

Source§

fn fmt(&self, f: &mut Formatter<'_>) -> Result

Formats the value using the given formatter. Read more
Source§

impl<C: Context> HashableView for KeyValueStoreView<C>

Source§

type Hasher = CoreWrapper<Sha3_256Core>

How to compute hashes.
Source§

async fn hash_mut( &mut self, ) -> Result<<Self::Hasher as Hasher>::Output, ViewError>

Same as hash but guaranteed to be wait-free.
Source§

async fn hash(&self) -> Result<<Self::Hasher as Hasher>::Output, ViewError>

Computes the hash of the values. Read more
Source§

impl<C: Context, C2: Context> ReplaceContext<C2> for KeyValueStoreView<C>

Source§

type Target = KeyValueStoreView<C2>

The type returned after replacing the context.
Source§

async fn with_context( &mut self, ctx: impl FnOnce(&Self::Context) -> C2 + Clone, ) -> Self::Target

Returns a view with a replaced context.
Source§

impl<C: Context> View for KeyValueStoreView<C>

Source§

const NUM_INIT_KEYS: usize = 2usize

The number of keys used for the initialization
Source§

type Context = C

The type of context stored in this view
Source§

fn context(&self) -> C

Obtains a mutable reference to the internal context.
Source§

fn pre_load(context: &C) -> Result<Vec<Vec<u8>>, ViewError>

Creates the keys needed for loading the view
Source§

fn post_load(context: C, values: &[Option<Vec<u8>>]) -> Result<Self, ViewError>

Loads a view from the values
Source§

fn rollback(&mut self)

Discards all pending changes. After that flush should have no effect to storage.
Source§

async fn has_pending_changes(&self) -> bool

Returns true if flushing this view would result in changes to the persistent storage.
Source§

fn pre_save(&self, batch: &mut Batch) -> Result<bool, ViewError>

Computes the batch of operations to persist changes to storage without modifying the view. Crash-resistant storage implementations accumulate the desired changes in the batch variable. The returned boolean indicates whether the operation removes the view or not.
Source§

fn post_save(&mut self)

Updates the view state after the batch has been executed in the database. This should be called after pre_save and after the batch has been successfully written to storage. This leaves the view in a clean state with no pending changes. Read more
Source§

fn clear(&mut self)

Clears the view. That can be seen as resetting to default. If the clear is followed by a flush then all the relevant data is removed on the storage.
Source§

fn load( context: Self::Context, ) -> impl Future<Output = Result<Self, ViewError>> + Send + Sync

Loads a view
Source§

fn new(context: Self::Context) -> Result<Self, ViewError>

Builds a trivial view that is already deleted

Auto Trait Implementations§

§

impl<C> !Freeze for KeyValueStoreView<C>

§

impl<C> RefUnwindSafe for KeyValueStoreView<C>
where C: RefUnwindSafe,

§

impl<C> Send for KeyValueStoreView<C>
where C: Send,

§

impl<C> Sync for KeyValueStoreView<C>
where C: Sync,

§

impl<C> Unpin for KeyValueStoreView<C>
where C: Unpin,

§

impl<C> UnwindSafe for KeyValueStoreView<C>
where C: UnwindSafe,

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> ArchivePointee for T

Source§

type ArchivedMetadata = ()

The archived version of the pointer metadata for this type.
Source§

fn pointer_metadata( _: &<T as ArchivePointee>::ArchivedMetadata, ) -> <T as Pointee>::Metadata

Converts some archived metadata to the pointer metadata for itself.
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<Choices> CoproductSubsetter<CNil, HNil> for Choices

Source§

type Remainder = Choices

Source§

fn subset( self, ) -> Result<CNil, <Choices as CoproductSubsetter<CNil, HNil>>::Remainder>

Extract a subset of the possible types in a coproduct (or get the remaining possibilities) Read more
Source§

impl<F, W, T, D> Deserialize<With<T, W>, D> for F
where W: DeserializeWith<F, T, D>, D: Fallible + ?Sized, F: ?Sized,

Source§

fn deserialize( &self, deserializer: &mut D, ) -> Result<With<T, W>, <D as Fallible>::Error>

Deserializes using the given deserializer
Source§

impl<To, From> DynInto<To> for From
where From: Into<To>,

Source§

fn into_box(self: Box<From>) -> To

Converts a boxed object into the target type.
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T> Instrument for T

Source§

fn instrument(self, span: Span) -> Instrumented<Self>

Instruments this type with the provided Span, returning an Instrumented wrapper. Read more
Source§

fn in_current_span(self) -> Instrumented<Self>

Instruments this type with the current Span, returning an Instrumented wrapper. Read more
Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<T> IntoEither for T

Source§

fn into_either(self, into_left: bool) -> Either<Self, Self>

Converts self into a Left variant of Either<Self, Self> if into_left is true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
where F: FnOnce(&Self) -> bool,

Converts self into a Left variant of Either<Self, Self> if into_left(&self) returns true. Converts self into a Right variant of Either<Self, Self> otherwise. Read more
Source§

impl<Unshared, Shared> IntoShared<Shared> for Unshared
where Shared: FromUnshared<Unshared>,

Source§

fn into_shared(self) -> Shared

Creates a shared type from an unshared type.
Source§

impl<T> LayoutRaw for T

Source§

fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError>

Gets the layout of the type.
Source§

impl<T, U, I> LiftInto<U, I> for T
where U: LiftFrom<T, I>,

Source§

fn lift_into(self) -> U

Performs the indexed conversion.
Source§

impl<T> MockResults for T

Source§

type Results = T

The mock native type of the results for the MockInstance.
Source§

impl<T> Pointable for T

Source§

const ALIGN: usize

The alignment of pointer.
Source§

type Init = T

The type for initializers.
Source§

unsafe fn init(init: <T as Pointable>::Init) -> usize

Initializes a with the given initializer. Read more
Source§

unsafe fn deref<'a>(ptr: usize) -> &'a T

Dereferences the given pointer. Read more
Source§

unsafe fn deref_mut<'a>(ptr: usize) -> &'a mut T

Mutably dereferences the given pointer. Read more
Source§

unsafe fn drop(ptr: usize)

Drops the object pointed to by the given pointer. Read more
Source§

impl<T> Pointee for T

Source§

type Metadata = ()

The type for metadata in pointers and references to Self.
Source§

impl<M, I> RuntimeMemory<&mut I> for M
where M: RuntimeMemory<I>,

Source§

fn read<'instance>( &self, instance: &'instance &mut I, location: GuestPointer, length: u32, ) -> Result<Cow<'instance, [u8]>, RuntimeError>

Reads length bytes from memory from the provided location.

Source§

fn write( &mut self, instance: &mut &mut I, location: GuestPointer, bytes: &[u8], ) -> Result<(), RuntimeError>

Writes the bytes to memory at the provided location.

Source§

impl<T> Same for T

Source§

type Output = T

Should always be Self
Source§

impl<Source> Sculptor<HNil, HNil> for Source

Source§

type Remainder = Source

Source§

fn sculpt(self) -> (HNil, <Source as Sculptor<HNil, HNil>>::Remainder)

Consumes the current HList and returns an HList with the requested shape. Read more
Source§

impl<AnyTail> Split<HNil> for AnyTail

Source§

type Remainder = AnyTail

The tail of remaining elements after splitting up the list.
Source§

fn split(self) -> (HNil, <AnyTail as Split<HNil>>::Remainder)

Splits the current heterogeneous list in two.
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.
Source§

impl<T> Upcastable for T
where T: Any + Send + Sync + 'static,

Source§

fn upcast_any_ref(&self) -> &(dyn Any + 'static)

upcast ref
Source§

fn upcast_any_mut(&mut self) -> &mut (dyn Any + 'static)

upcast mut ref
Source§

fn upcast_any_box(self: Box<T>) -> Box<dyn Any>

upcast boxed dyn
Source§

impl<V, T> VZip<V> for T
where V: MultiLane<T>,

Source§

fn vzip(self) -> V

Source§

impl<T> WithSubscriber for T

Source§

fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
where S: Into<Dispatch>,

Attaches the provided Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

fn with_current_subscriber(self) -> WithDispatch<Self>

Attaches the current default Subscriber to this type, returning a WithDispatch wrapper. Read more
Source§

impl<_INNER> AutoTraits for _INNER
where _INNER: Send + Sync + 'static,

Source§

impl<T> ErasedDestructor for T
where T: 'static,