Struct linera_views::views::key_value_store_view::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 prefixdp
such thatdp <= index
. Ifdp
is indeed a prefix then we conclude thatindex
is deleted, otherwise not. The no domination is essential here.
Implementations§
source§impl<'a, C> KeyValueStoreView<C>
impl<'a, C> KeyValueStoreView<C>
sourcepub fn total_size(&self) -> SizeData
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();
let total_size = view.total_size();
assert_eq!(total_size, SizeData::default());
sourcepub async fn for_each_index_while<F>(&self, f: F) -> Result<(), ViewError>
pub async fn for_each_index_while<F>(&self, f: F) -> Result<(), ViewError>
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);
sourcepub async fn for_each_index<F>(&self, f: F) -> Result<(), ViewError>
pub async fn for_each_index<F>(&self, f: F) -> Result<(), ViewError>
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);
sourcepub async fn for_each_index_value_while<F>(&self, f: F) -> Result<(), ViewError>
pub async fn for_each_index_value_while<F>(&self, f: F) -> Result<(), ViewError>
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]]);
sourcepub async fn for_each_index_value<F>(&self, f: F) -> Result<(), ViewError>
pub async fn for_each_index_value<F>(&self, f: F) -> Result<(), ViewError>
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]]);
sourcepub async fn indices(&self) -> Result<Vec<Vec<u8>>, ViewError>
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]]);
sourcepub async fn index_values(&self) -> Result<Vec<(Vec<u8>, Vec<u8>)>, ViewError>
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]]);
sourcepub async fn count(&self) -> Result<usize, ViewError>
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);
sourcepub async fn get(&self, index: &[u8]) -> Result<Option<Vec<u8>>, ViewError>
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);
sourcepub async fn contains_key(&self, index: &[u8]) -> Result<bool, ViewError>
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());
sourcepub async fn contains_keys(
&self,
indices: Vec<Vec<u8>>,
) -> Result<Vec<bool>, ViewError>
pub async fn contains_keys( &self, indices: Vec<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]);
sourcepub async fn multi_get(
&self,
indices: Vec<Vec<u8>>,
) -> Result<Vec<Option<Vec<u8>>>, ViewError>
pub async fn multi_get( &self, indices: Vec<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![vec![0, 1], vec![0, 2]]).await.unwrap(),
vec![Some(vec![42]), None]
);
sourcepub async fn write_batch(&mut self, batch: Batch) -> Result<(), ViewError>
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![]);
sourcepub async fn insert(
&mut self,
index: Vec<u8>,
value: Vec<u8>,
) -> Result<(), ViewError>
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]));
sourcepub async fn remove(&mut self, index: Vec<u8>) -> Result<(), ViewError>
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);
sourcepub async fn remove_by_prefix(
&mut self,
key_prefix: Vec<u8>,
) -> Result<(), ViewError>
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);
sourcepub async fn find_keys_by_prefix(
&self,
key_prefix: &[u8],
) -> Result<Vec<Vec<u8>>, ViewError>
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]]);
sourcepub async fn find_key_values_by_prefix(
&self,
key_prefix: &[u8],
) -> Result<Vec<(Vec<u8>, Vec<u8>)>, ViewError>
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> ClonableView<C> for KeyValueStoreView<C>
impl<C> ClonableView<C> for KeyValueStoreView<C>
source§fn clone_unchecked(&mut self) -> Result<Self, ViewError>
fn clone_unchecked(&mut self) -> Result<Self, ViewError>
source§impl<C: Debug> Debug for KeyValueStoreView<C>
impl<C: Debug> Debug for KeyValueStoreView<C>
source§impl<C> HashableView<C> for KeyValueStoreView<C>
impl<C> HashableView<C> for KeyValueStoreView<C>
§type Hasher = CoreWrapper<Sha3_256Core>
type Hasher = CoreWrapper<Sha3_256Core>
source§impl<C> View<C> for KeyValueStoreView<C>
impl<C> View<C> for KeyValueStoreView<C>
source§const NUM_INIT_KEYS: usize = 2usize
const NUM_INIT_KEYS: usize = 2usize
source§fn pre_load(context: &C) -> Result<Vec<Vec<u8>>, ViewError>
fn pre_load(context: &C) -> Result<Vec<Vec<u8>>, ViewError>
source§fn post_load(context: C, values: &[Option<Vec<u8>>]) -> Result<Self, ViewError>
fn post_load(context: C, values: &[Option<Vec<u8>>]) -> Result<Self, ViewError>
source§fn load<'async_trait>(
context: C,
) -> Pin<Box<dyn Future<Output = Result<Self, ViewError>> + Send + 'async_trait>>where
Self: 'async_trait,
fn load<'async_trait>(
context: C,
) -> Pin<Box<dyn Future<Output = Result<Self, ViewError>> + Send + 'async_trait>>where
Self: 'async_trait,
source§fn rollback(&mut self)
fn rollback(&mut self)
flush
should have no effect to storage.source§fn has_pending_changes<'life0, 'async_trait>(
&'life0 self,
) -> Pin<Box<dyn Future<Output = bool> + Send + 'async_trait>>where
Self: 'async_trait,
'life0: 'async_trait,
fn has_pending_changes<'life0, 'async_trait>(
&'life0 self,
) -> Pin<Box<dyn Future<Output = bool> + Send + 'async_trait>>where
Self: 'async_trait,
'life0: 'async_trait,
true
if flushing this view would result in changes to the persistent storage.source§fn flush(&mut self, batch: &mut Batch) -> Result<bool, ViewError>
fn flush(&mut self, batch: &mut Batch) -> Result<bool, ViewError>
batch
variable first. If the view is dropped without calling flush
, staged
changes are simply lost.
The returned boolean indicates whether the operation removes the view or not.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> ArchivePointee for T
impl<T> ArchivePointee for T
§type ArchivedMetadata = ()
type ArchivedMetadata = ()
source§fn pointer_metadata(
_: &<T as ArchivePointee>::ArchivedMetadata,
) -> <T as Pointee>::Metadata
fn pointer_metadata( _: &<T as ArchivePointee>::ArchivedMetadata, ) -> <T as Pointee>::Metadata
source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
source§impl<Choices> CoproductSubsetter<CNil, HNil> for Choices
impl<Choices> CoproductSubsetter<CNil, HNil> for Choices
source§impl<F, W, T, D> Deserialize<With<T, W>, D> for F
impl<F, W, T, D> Deserialize<With<T, W>, D> for F
source§impl<T> Instrument for T
impl<T> Instrument for T
source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
source§impl<T> IntoEither for T
impl<T> IntoEither for T
source§fn into_either(self, into_left: bool) -> Either<Self, Self>
fn into_either(self, into_left: bool) -> Either<Self, Self>
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 moresource§fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
fn into_either_with<F>(self, into_left: F) -> Either<Self, Self>
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 moresource§impl<T> LayoutRaw for T
impl<T> LayoutRaw for T
source§fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError>
fn layout_raw(_: <T as Pointee>::Metadata) -> Result<Layout, LayoutError>
source§impl<T> MockResults for T
impl<T> MockResults for T
§type Results = T
type Results = T
MockInstance
.source§impl<T> Pointable for T
impl<T> Pointable for T
source§impl<M, I> RuntimeMemory<&mut I> for Mwhere
M: RuntimeMemory<I>,
impl<M, I> RuntimeMemory<&mut I> for Mwhere
M: RuntimeMemory<I>,
source§fn read<'instance>(
&self,
instance: &'instance &mut I,
location: GuestPointer,
length: u32,
) -> Result<Cow<'instance, [u8]>, RuntimeError>
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>
fn write( &mut self, instance: &mut &mut I, location: GuestPointer, bytes: &[u8], ) -> Result<(), RuntimeError>
Writes the bytes
to memory at the provided location
.