mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-28 03:58:04 +00:00
2604474880
* Implement a non generic version of child delta trie * Use delta_trie_root in state_machine * Expand InMemory backend to support multi-storage * Create Consolidate trait * Fix all crate compile and remove unused OverlayedChanges::drain * Implement child storage root support and overlay changes * Add child storage reader * Add child storage writer * Implement child storage cleaning * Fix light backend compile * Add all required ext functions for wasm executor * Add ext def to io * Add all io functions * Fix nostd compile * Add simple test * Remove unnecessary vec copy in child_storage_root_transaction * Use values_mut/for_each to make it shorter * Use extend to shorter a for loop * Move record_all_keys to trie so it's easier to generic them later * space -> tab * Remove to_owned in debug format * Clean out all to_owned * Break debug_trace to multiple lines * Remove 0.. * UserError copy/paste typo * Replace Vec::from_raw_parts by slice::from_raw_parts * Use iter::empty() * Wrap some long lines * Wrap a missing line * Remove unnecessary map https://github.com/paritytech/substrate/pull/856#discussion_r226222663 * Call ext_free after from_raw_parts * Fix tests in other crates
477 lines
16 KiB
Rust
477 lines
16 KiB
Rust
// Copyright 2017-2018 Parity Technologies (UK) Ltd.
|
|
// This file is part of Substrate.
|
|
|
|
// Substrate is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Substrate is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Substrate. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! The overlayed changes to state.
|
|
|
|
#[cfg(test)] use std::iter::FromIterator;
|
|
use std::collections::{HashMap, HashSet};
|
|
use codec::Decode;
|
|
use changes_trie::{NO_EXTRINSIC_INDEX, Configuration as ChangesTrieConfig};
|
|
use primitives::storage::well_known_keys::EXTRINSIC_INDEX;
|
|
|
|
/// The overlayed changes to state to be queried on top of the backend.
|
|
///
|
|
/// A transaction shares all prospective changes within an inner overlay
|
|
/// that can be cleared.
|
|
#[derive(Debug, Default, Clone)]
|
|
pub struct OverlayedChanges {
|
|
/// Changes that are not yet committed.
|
|
pub(crate) prospective: OverlayedChangeSet,
|
|
/// Committed changes.
|
|
pub(crate) committed: OverlayedChangeSet,
|
|
/// Changes trie configuration. None by default, but could be installed by the
|
|
/// runtime if it supports change tries.
|
|
pub(crate) changes_trie_config: Option<ChangesTrieConfig>,
|
|
}
|
|
|
|
/// The storage value, used inside OverlayedChanges.
|
|
#[derive(Debug, Default, Clone)]
|
|
#[cfg_attr(test, derive(PartialEq))]
|
|
pub struct OverlayedValue {
|
|
/// Current value. None if value has been deleted.
|
|
pub value: Option<Vec<u8>>,
|
|
/// The set of extinsic indices where the values has been changed.
|
|
/// Is filled only if runtime ahs announced changes trie support.
|
|
pub extrinsics: Option<HashSet<u32>>,
|
|
}
|
|
|
|
/// Prospective or committed overlayed change set.
|
|
#[derive(Debug, Default, Clone)]
|
|
#[cfg_attr(test, derive(PartialEq))]
|
|
pub struct OverlayedChangeSet {
|
|
/// Top level storage changes.
|
|
pub top: HashMap<Vec<u8>, OverlayedValue>,
|
|
/// Child storage changes.
|
|
pub children: HashMap<Vec<u8>, (Option<HashSet<u32>>, HashMap<Vec<u8>, Option<Vec<u8>>>)>,
|
|
}
|
|
|
|
#[cfg(test)]
|
|
impl FromIterator<(Vec<u8>, OverlayedValue)> for OverlayedChangeSet {
|
|
fn from_iter<T: IntoIterator<Item = (Vec<u8>, OverlayedValue)>>(iter: T) -> Self {
|
|
Self {
|
|
top: iter.into_iter().collect(),
|
|
children: Default::default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl OverlayedChangeSet {
|
|
/// Whether the change set is empty.
|
|
pub fn is_empty(&self) -> bool {
|
|
self.top.is_empty() && self.children.is_empty()
|
|
}
|
|
|
|
/// Clear the change set.
|
|
pub fn clear(&mut self) {
|
|
self.top.clear();
|
|
self.children.clear();
|
|
}
|
|
}
|
|
|
|
impl OverlayedChanges {
|
|
/// Sets the changes trie configuration.
|
|
///
|
|
/// Returns false if configuration has been set already and we now trying
|
|
/// to install different configuration. This isn't supported now.
|
|
pub(crate) fn set_changes_trie_config(&mut self, config: ChangesTrieConfig) -> bool {
|
|
if let Some(ref old_config) = self.changes_trie_config {
|
|
// we do not support changes trie configuration' change now
|
|
if *old_config != config {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
self.changes_trie_config = Some(config);
|
|
true
|
|
}
|
|
|
|
/// Returns a double-Option: None if the key is unknown (i.e. and the query should be refered
|
|
/// to the backend); Some(None) if the key has been deleted. Some(Some(...)) for a key whose
|
|
/// value has been set.
|
|
pub fn storage(&self, key: &[u8]) -> Option<Option<&[u8]>> {
|
|
self.prospective.top.get(key)
|
|
.or_else(|| self.committed.top.get(key))
|
|
.map(|x| x.value.as_ref().map(AsRef::as_ref))
|
|
}
|
|
|
|
/// Returns a double-Option: None if the key is unknown (i.e. and the query should be refered
|
|
/// to the backend); Some(None) if the key has been deleted. Some(Some(...)) for a key whose
|
|
/// value has been set.
|
|
pub fn child_storage(&self, storage_key: &[u8], key: &[u8]) -> Option<Option<&[u8]>> {
|
|
if let Some(map) = self.prospective.children.get(storage_key) {
|
|
if let Some(val) = map.1.get(key) {
|
|
return Some(val.as_ref().map(AsRef::as_ref));
|
|
}
|
|
}
|
|
|
|
if let Some(map) = self.committed.children.get(storage_key) {
|
|
if let Some(val) = map.1.get(key) {
|
|
return Some(val.as_ref().map(AsRef::as_ref));
|
|
}
|
|
}
|
|
|
|
None
|
|
}
|
|
|
|
/// Inserts the given key-value pair into the prospective change set.
|
|
///
|
|
/// `None` can be used to delete a value specified by the given key.
|
|
pub(crate) fn set_storage(&mut self, key: Vec<u8>, val: Option<Vec<u8>>) {
|
|
let extrinsic_index = self.extrinsic_index();
|
|
let entry = self.prospective.top.entry(key).or_default();
|
|
entry.value = val;
|
|
|
|
if let Some(extrinsic) = extrinsic_index {
|
|
entry.extrinsics.get_or_insert_with(Default::default)
|
|
.insert(extrinsic);
|
|
}
|
|
}
|
|
|
|
/// Inserts the given key-value pair into the prospective child change set.
|
|
///
|
|
/// `None` can be used to delete a value specified by the given key.
|
|
pub(crate) fn set_child_storage(&mut self, storage_key: Vec<u8>, key: Vec<u8>, val: Option<Vec<u8>>) {
|
|
let extrinsic_index = self.extrinsic_index();
|
|
let map_entry = self.prospective.children.entry(storage_key).or_default();
|
|
map_entry.1.insert(key, val);
|
|
|
|
if let Some(extrinsic) = extrinsic_index {
|
|
map_entry.0.get_or_insert_with(Default::default)
|
|
.insert(extrinsic);
|
|
}
|
|
}
|
|
|
|
/// Sync the child storage root.
|
|
pub(crate) fn sync_child_storage_root(&mut self, storage_key: &[u8], root: Option<Vec<u8>>) {
|
|
let entry = self.prospective.top.entry(storage_key.to_vec()).or_default();
|
|
entry.value = root;
|
|
|
|
if let Some((Some(extrinsics), _)) = self.prospective.children.get(storage_key) {
|
|
for extrinsic in extrinsics {
|
|
entry.extrinsics.get_or_insert_with(Default::default)
|
|
.insert(*extrinsic);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Clear child storage of given storage key.
|
|
///
|
|
/// NOTE that this doesn't take place immediately but written into the prospective
|
|
/// change set, and still can be reverted by [`discard_prospective`].
|
|
///
|
|
/// [`discard_prospective`]: #method.discard_prospective
|
|
pub(crate) fn clear_child_storage(&mut self, storage_key: &[u8]) {
|
|
let extrinsic_index = self.extrinsic_index();
|
|
let map_entry = self.prospective.children.entry(storage_key.to_vec()).or_default();
|
|
|
|
if let Some(extrinsic) = extrinsic_index {
|
|
map_entry.0.get_or_insert_with(Default::default)
|
|
.insert(extrinsic);
|
|
}
|
|
|
|
map_entry.1.values_mut().for_each(|e| *e = None);
|
|
|
|
if let Some((_, committed_map)) = self.committed.children.get(storage_key) {
|
|
for (key, _) in committed_map.iter() {
|
|
map_entry.1.insert(key.clone(), None);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Removes all key-value pairs which keys share the given prefix.
|
|
///
|
|
/// NOTE that this doesn't take place immediately but written into the prospective
|
|
/// change set, and still can be reverted by [`discard_prospective`].
|
|
///
|
|
/// [`discard_prospective`]: #method.discard_prospective
|
|
pub(crate) fn clear_prefix(&mut self, prefix: &[u8]) {
|
|
let extrinsic_index = self.extrinsic_index();
|
|
|
|
// Iterate over all prospective and mark all keys that share
|
|
// the given prefix as removed (None).
|
|
for (key, entry) in self.prospective.top.iter_mut() {
|
|
if key.starts_with(prefix) {
|
|
entry.value = None;
|
|
|
|
if let Some(extrinsic) = extrinsic_index {
|
|
entry.extrinsics.get_or_insert_with(Default::default)
|
|
.insert(extrinsic);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Then do the same with keys from commited changes.
|
|
// NOTE that we are making changes in the prospective change set.
|
|
for key in self.committed.top.keys() {
|
|
if key.starts_with(prefix) {
|
|
let entry = self.prospective.top.entry(key.clone()).or_default();
|
|
entry.value = None;
|
|
|
|
if let Some(extrinsic) = extrinsic_index {
|
|
entry.extrinsics.get_or_insert_with(Default::default)
|
|
.insert(extrinsic);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Discard prospective changes to state.
|
|
pub fn discard_prospective(&mut self) {
|
|
self.prospective.clear();
|
|
}
|
|
|
|
/// Commit prospective changes to state.
|
|
pub fn commit_prospective(&mut self) {
|
|
if self.committed.is_empty() {
|
|
::std::mem::swap(&mut self.prospective, &mut self.committed);
|
|
} else {
|
|
for (key, val) in self.prospective.top.drain() {
|
|
let entry = self.committed.top.entry(key).or_default();
|
|
entry.value = val.value;
|
|
|
|
if let Some(prospective_extrinsics) = val.extrinsics {
|
|
entry.extrinsics.get_or_insert_with(Default::default)
|
|
.extend(prospective_extrinsics);
|
|
}
|
|
}
|
|
for (storage_key, map) in self.prospective.children.drain() {
|
|
let entry = self.committed.children.entry(storage_key).or_default();
|
|
entry.1.extend(map.1.iter().map(|(k, v)| (k.clone(), v.clone())));
|
|
|
|
if let Some(prospective_extrinsics) = map.0 {
|
|
entry.0.get_or_insert_with(Default::default)
|
|
.extend(prospective_extrinsics);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Consume `OverlayedChanges` and take committed set.
|
|
///
|
|
/// Panics:
|
|
/// Will panic if there are any uncommitted prospective changes.
|
|
pub fn into_committed(self) -> impl Iterator<Item=(Vec<u8>, Option<Vec<u8>>)> {
|
|
assert!(self.prospective.is_empty());
|
|
self.committed.top.into_iter().map(|(k, v)| (k, v.value))
|
|
}
|
|
|
|
/// Inserts storage entry responsible for current extrinsic index.
|
|
#[cfg(test)]
|
|
pub(crate) fn set_extrinsic_index(&mut self, extrinsic_index: u32) {
|
|
use codec::Encode;
|
|
self.prospective.top.insert(EXTRINSIC_INDEX.to_vec(), OverlayedValue {
|
|
value: Some(extrinsic_index.encode()),
|
|
extrinsics: None,
|
|
});
|
|
}
|
|
|
|
/// Returns current extrinsic index to use in changes trie construction.
|
|
/// None is returned if it is not set or changes trie config is not set.
|
|
/// Persistent value (from the backend) can be ignored because runtime must
|
|
/// set this index before first and unset after last extrinsic is executied.
|
|
/// Changes that are made outside of extrinsics, are marked with
|
|
/// `NO_EXTRINSIC_INDEX` index.
|
|
fn extrinsic_index(&self) -> Option<u32> {
|
|
match self.changes_trie_config.is_some() {
|
|
true => Some(
|
|
self.storage(EXTRINSIC_INDEX)
|
|
.and_then(|idx| idx.and_then(|idx| Decode::decode(&mut &*idx)))
|
|
.unwrap_or(NO_EXTRINSIC_INDEX)),
|
|
false => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
impl From<Option<Vec<u8>>> for OverlayedValue {
|
|
fn from(value: Option<Vec<u8>>) -> OverlayedValue {
|
|
OverlayedValue { value, ..Default::default() }
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use primitives::{Blake2Hasher, H256};
|
|
use primitives::storage::well_known_keys::EXTRINSIC_INDEX;
|
|
use backend::InMemory;
|
|
use changes_trie::InMemoryStorage as InMemoryChangesTrieStorage;
|
|
use ext::Ext;
|
|
use {Externalities};
|
|
use super::*;
|
|
|
|
fn strip_extrinsic_index(map: &HashMap<Vec<u8>, OverlayedValue>) -> HashMap<Vec<u8>, OverlayedValue> {
|
|
let mut clone = map.clone();
|
|
clone.remove(&EXTRINSIC_INDEX.to_vec());
|
|
clone
|
|
}
|
|
|
|
#[test]
|
|
fn overlayed_storage_works() {
|
|
let mut overlayed = OverlayedChanges::default();
|
|
|
|
let key = vec![42, 69, 169, 142];
|
|
|
|
assert!(overlayed.storage(&key).is_none());
|
|
|
|
overlayed.set_storage(key.clone(), Some(vec![1, 2, 3]));
|
|
assert_eq!(overlayed.storage(&key).unwrap(), Some(&[1, 2, 3][..]));
|
|
|
|
overlayed.commit_prospective();
|
|
assert_eq!(overlayed.storage(&key).unwrap(), Some(&[1, 2, 3][..]));
|
|
|
|
overlayed.set_storage(key.clone(), Some(vec![]));
|
|
assert_eq!(overlayed.storage(&key).unwrap(), Some(&[][..]));
|
|
|
|
overlayed.set_storage(key.clone(), None);
|
|
assert!(overlayed.storage(&key).unwrap().is_none());
|
|
|
|
overlayed.discard_prospective();
|
|
assert_eq!(overlayed.storage(&key).unwrap(), Some(&[1, 2, 3][..]));
|
|
|
|
overlayed.set_storage(key.clone(), None);
|
|
overlayed.commit_prospective();
|
|
assert!(overlayed.storage(&key).unwrap().is_none());
|
|
}
|
|
|
|
#[test]
|
|
fn overlayed_storage_root_works() {
|
|
let initial: HashMap<_, _> = vec![
|
|
(b"doe".to_vec(), b"reindeer".to_vec()),
|
|
(b"dog".to_vec(), b"puppyXXX".to_vec()),
|
|
(b"dogglesworth".to_vec(), b"catXXX".to_vec()),
|
|
(b"doug".to_vec(), b"notadog".to_vec()),
|
|
].into_iter().collect();
|
|
let backend = InMemory::<Blake2Hasher>::from(initial);
|
|
let mut overlay = OverlayedChanges {
|
|
committed: vec![
|
|
(b"dog".to_vec(), Some(b"puppy".to_vec()).into()),
|
|
(b"dogglesworth".to_vec(), Some(b"catYYY".to_vec()).into()),
|
|
(b"doug".to_vec(), Some(vec![]).into()),
|
|
].into_iter().collect(),
|
|
prospective: vec![
|
|
(b"dogglesworth".to_vec(), Some(b"cat".to_vec()).into()),
|
|
(b"doug".to_vec(), None.into()),
|
|
].into_iter().collect(),
|
|
..Default::default()
|
|
};
|
|
|
|
let changes_trie_storage = InMemoryChangesTrieStorage::new();
|
|
let mut ext = Ext::new(&mut overlay, &backend, Some(&changes_trie_storage));
|
|
const ROOT: [u8; 32] = hex!("0b41e488cccbd67d1f1089592c2c235f5c5399b053f7fe9152dd4b5f279914cd");
|
|
assert_eq!(ext.storage_root(), H256(ROOT));
|
|
}
|
|
|
|
#[test]
|
|
fn changes_trie_configuration_is_saved() {
|
|
let mut overlay = OverlayedChanges::default();
|
|
assert!(overlay.changes_trie_config.is_none());
|
|
assert_eq!(overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 4, digest_levels: 1,
|
|
}), true);
|
|
assert!(overlay.changes_trie_config.is_some());
|
|
}
|
|
|
|
#[test]
|
|
fn changes_trie_configuration_is_saved_twice() {
|
|
let mut overlay = OverlayedChanges::default();
|
|
assert!(overlay.changes_trie_config.is_none());
|
|
assert_eq!(overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 4, digest_levels: 1,
|
|
}), true);
|
|
overlay.set_extrinsic_index(0);
|
|
overlay.set_storage(vec![1], Some(vec![2]));
|
|
assert_eq!(overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 4, digest_levels: 1,
|
|
}), true);
|
|
assert_eq!(
|
|
strip_extrinsic_index(&overlay.prospective.top),
|
|
vec![
|
|
(vec![1], OverlayedValue { value: Some(vec![2]), extrinsics: Some(vec![0].into_iter().collect()) }),
|
|
].into_iter().collect(),
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn panics_when_trying_to_save_different_changes_trie_configuration() {
|
|
let mut overlay = OverlayedChanges::default();
|
|
assert_eq!(overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 4, digest_levels: 1,
|
|
}), true);
|
|
assert_eq!(overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 2, digest_levels: 1,
|
|
}), false);
|
|
}
|
|
|
|
#[test]
|
|
fn extrinsic_changes_are_collected() {
|
|
let mut overlay = OverlayedChanges::default();
|
|
let _ = overlay.set_changes_trie_config(ChangesTrieConfig {
|
|
digest_interval: 4, digest_levels: 1,
|
|
});
|
|
|
|
overlay.set_storage(vec![100], Some(vec![101]));
|
|
|
|
overlay.set_extrinsic_index(0);
|
|
overlay.set_storage(vec![1], Some(vec![2]));
|
|
|
|
overlay.set_extrinsic_index(1);
|
|
overlay.set_storage(vec![3], Some(vec![4]));
|
|
|
|
overlay.set_extrinsic_index(2);
|
|
overlay.set_storage(vec![1], Some(vec![6]));
|
|
|
|
assert_eq!(strip_extrinsic_index(&overlay.prospective.top),
|
|
vec![
|
|
(vec![1], OverlayedValue { value: Some(vec![6]), extrinsics: Some(vec![0, 2].into_iter().collect()) }),
|
|
(vec![3], OverlayedValue { value: Some(vec![4]), extrinsics: Some(vec![1].into_iter().collect()) }),
|
|
(vec![100], OverlayedValue { value: Some(vec![101]), extrinsics: Some(vec![NO_EXTRINSIC_INDEX].into_iter().collect()) }),
|
|
].into_iter().collect());
|
|
|
|
overlay.commit_prospective();
|
|
|
|
overlay.set_extrinsic_index(3);
|
|
overlay.set_storage(vec![3], Some(vec![7]));
|
|
|
|
overlay.set_extrinsic_index(4);
|
|
overlay.set_storage(vec![1], Some(vec![8]));
|
|
|
|
assert_eq!(strip_extrinsic_index(&overlay.committed.top),
|
|
vec![
|
|
(vec![1], OverlayedValue { value: Some(vec![6]), extrinsics: Some(vec![0, 2].into_iter().collect()) }),
|
|
(vec![3], OverlayedValue { value: Some(vec![4]), extrinsics: Some(vec![1].into_iter().collect()) }),
|
|
(vec![100], OverlayedValue { value: Some(vec![101]), extrinsics: Some(vec![NO_EXTRINSIC_INDEX].into_iter().collect()) }),
|
|
].into_iter().collect());
|
|
|
|
assert_eq!(strip_extrinsic_index(&overlay.prospective.top),
|
|
vec![
|
|
(vec![1], OverlayedValue { value: Some(vec![8]), extrinsics: Some(vec![4].into_iter().collect()) }),
|
|
(vec![3], OverlayedValue { value: Some(vec![7]), extrinsics: Some(vec![3].into_iter().collect()) }),
|
|
].into_iter().collect());
|
|
|
|
overlay.commit_prospective();
|
|
|
|
assert_eq!(strip_extrinsic_index(&overlay.committed.top),
|
|
vec![
|
|
(vec![1], OverlayedValue { value: Some(vec![8]), extrinsics: Some(vec![0, 2, 4].into_iter().collect()) }),
|
|
(vec![3], OverlayedValue { value: Some(vec![7]), extrinsics: Some(vec![1, 3].into_iter().collect()) }),
|
|
(vec![100], OverlayedValue { value: Some(vec![101]), extrinsics: Some(vec![NO_EXTRINSIC_INDEX].into_iter().collect()) }),
|
|
].into_iter().collect());
|
|
|
|
assert_eq!(overlay.prospective,
|
|
Default::default());
|
|
}
|
|
}
|