mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-11 04:51:09 +00:00
bd652793db
* Move transaction pool to primitives * move backend, errors into primitives * remove unused client depencies * Move rpc-api into primitives * Move peerset back to client * Move rpc/api back to client, move palette/support/rpc into utils * move support-rpc into subfolder * move system-rpc into utils * move transaction-pool and -graph back into client * fix broken imports * Clean up test primitives * Make support test utils independent of frame * remove unnecessary node dependencies from service * Reactivate dependency script: - only enforce the now achieved status quo will remain - allow for primitives to depend on /client for now without failing - more discriptive error message so people understand, what it wants - minor fix to differentiative between ../client and /client (which may be a subfolder) - don't allow this to fail anylonger. * fix doc comment * 'Should not' rather than 'must not'. * Revert unwanted dependency changes * fix faulty import * fixup derive_more version * fix wrong import path
162 lines
5.7 KiB
Rust
162 lines
5.7 KiB
Rust
// Copyright 2017-2019 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/>.
|
|
|
|
//! List-cache storage entries.
|
|
|
|
use sp_blockchain::Result as ClientResult;
|
|
use sr_primitives::traits::{Block as BlockT, NumberFor};
|
|
use codec::{Encode, Decode};
|
|
|
|
use crate::cache::{CacheItemT, ComplexBlockId};
|
|
use crate::cache::list_storage::{Storage};
|
|
|
|
/// Single list-based cache entry.
|
|
#[derive(Debug)]
|
|
#[cfg_attr(test, derive(PartialEq))]
|
|
pub struct Entry<Block: BlockT, T> {
|
|
/// first block, when this value became actual.
|
|
pub valid_from: ComplexBlockId<Block>,
|
|
/// Value stored at this entry.
|
|
pub value: T,
|
|
}
|
|
|
|
/// Internal representation of the single list-based cache entry. The entry points to the
|
|
/// previous entry in the cache, allowing us to traverse back in time in list-style.
|
|
#[derive(Debug, Encode, Decode)]
|
|
#[cfg_attr(test, derive(Clone, PartialEq))]
|
|
pub struct StorageEntry<Block: BlockT, T: CacheItemT> {
|
|
/// None if valid from the beginning.
|
|
pub prev_valid_from: Option<ComplexBlockId<Block>>,
|
|
/// Value stored at this entry.
|
|
pub value: T,
|
|
}
|
|
|
|
impl<Block: BlockT, T: CacheItemT> Entry<Block, T> {
|
|
/// Returns Some if the entry should be updated with the new value.
|
|
pub fn try_update(&self, value: Option<T>) -> Option<StorageEntry<Block, T>> {
|
|
match value {
|
|
Some(value) => match self.value == value {
|
|
true => None,
|
|
false => Some(StorageEntry {
|
|
prev_valid_from: Some(self.valid_from.clone()),
|
|
value,
|
|
}),
|
|
},
|
|
None => None,
|
|
}
|
|
}
|
|
|
|
/// Wrapper that calls search_before to get range where the given block fits.
|
|
pub fn search_best_range_before<S: Storage<Block, T>>(
|
|
&self,
|
|
storage: &S,
|
|
block: NumberFor<Block>,
|
|
) -> ClientResult<Option<(ComplexBlockId<Block>, Option<ComplexBlockId<Block>>)>> {
|
|
Ok(self.search_best_before(storage, block)?
|
|
.map(|(entry, next)| (entry.valid_from, next)))
|
|
}
|
|
|
|
/// Searches the list, ending with THIS entry for the best entry preceeding (or at)
|
|
/// given block number.
|
|
/// If the entry is found, result is the entry and the block id of next entry (if exists).
|
|
/// NOTE that this function does not check that the passed block is actually linked to
|
|
/// the blocks it found.
|
|
pub fn search_best_before<S: Storage<Block, T>>(
|
|
&self,
|
|
storage: &S,
|
|
block: NumberFor<Block>,
|
|
) -> ClientResult<Option<(Entry<Block, T>, Option<ComplexBlockId<Block>>)>> {
|
|
// we're looking for the best value
|
|
let mut next = None;
|
|
let mut current = self.valid_from.clone();
|
|
if block >= self.valid_from.number {
|
|
let value = self.value.clone();
|
|
return Ok(Some((Entry { valid_from: current, value }, next)));
|
|
}
|
|
|
|
// else - travel back in time
|
|
loop {
|
|
let entry = storage.require_entry(¤t)?;
|
|
if block >= current.number {
|
|
return Ok(Some((Entry { valid_from: current, value: entry.value }, next)));
|
|
}
|
|
|
|
next = Some(current);
|
|
current = match entry.prev_valid_from {
|
|
Some(prev_valid_from) => prev_valid_from,
|
|
None => return Ok(None),
|
|
};
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<Block: BlockT, T: CacheItemT> StorageEntry<Block, T> {
|
|
/// Converts storage entry into an entry, valid from given block.
|
|
pub fn into_entry(self, valid_from: ComplexBlockId<Block>) -> Entry<Block, T> {
|
|
Entry {
|
|
valid_from,
|
|
value: self.value,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use crate::cache::list_cache::tests::test_id;
|
|
use crate::cache::list_storage::tests::{DummyStorage, FaultyStorage};
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn entry_try_update_works() {
|
|
// when trying to update with None value
|
|
assert_eq!(Entry::<_, u64> { valid_from: test_id(1), value: 42 }.try_update(None), None);
|
|
// when trying to update with the same Some value
|
|
assert_eq!(Entry { valid_from: test_id(1), value: 1 }.try_update(Some(1)), None);
|
|
// when trying to update with different Some value
|
|
assert_eq!(Entry { valid_from: test_id(1), value: 1 }.try_update(Some(2)),
|
|
Some(StorageEntry { prev_valid_from: Some(test_id(1)), value: 2 }));
|
|
}
|
|
|
|
#[test]
|
|
fn entry_search_best_before_fails() {
|
|
// when storage returns error
|
|
assert!(Entry::<_, u64> { valid_from: test_id(100), value: 42 }
|
|
.search_best_before(&FaultyStorage, 50).is_err());
|
|
}
|
|
|
|
#[test]
|
|
fn entry_search_best_before_works() {
|
|
// when block is better than our best block
|
|
assert_eq!(Entry::<_, u64> { valid_from: test_id(100), value: 100 }
|
|
.search_best_before(&DummyStorage::new(), 150).unwrap(),
|
|
Some((Entry::<_, u64> { valid_from: test_id(100), value: 100 }, None)));
|
|
// when block is found between two entries
|
|
assert_eq!(Entry::<_, u64> { valid_from: test_id(100), value: 100 }
|
|
.search_best_before(&DummyStorage::new()
|
|
.with_entry(test_id(100), StorageEntry { prev_valid_from: Some(test_id(50)), value: 100 })
|
|
.with_entry(test_id(50), StorageEntry { prev_valid_from: Some(test_id(30)), value: 50 }),
|
|
75).unwrap(),
|
|
Some((Entry::<_, u64> { valid_from: test_id(50), value: 50 }, Some(test_id(100)))));
|
|
// when block is not found
|
|
assert_eq!(Entry::<_, u64> { valid_from: test_id(100), value: 100 }
|
|
.search_best_before(&DummyStorage::new()
|
|
.with_entry(test_id(100), StorageEntry { prev_valid_from: Some(test_id(50)), value: 100 })
|
|
.with_entry(test_id(50), StorageEntry { prev_valid_from: None, value: 50 }),
|
|
30).unwrap(),
|
|
None);
|
|
}
|
|
}
|