mirror of
https://github.com/pezkuwichain/pezkuwi-telemetry.git
synced 2026-04-30 02:38:01 +00:00
166 lines
4.4 KiB
Rust
166 lines
4.4 KiB
Rust
// Source code for the Substrate Telemetry Server.
|
|
// Copyright (C) 2021 Parity Technologies (UK) Ltd.
|
|
//
|
|
// This program 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.
|
|
//
|
|
// This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
/// This stores items in contiguous memory, making a note of free
|
|
/// slots when items are removed again so that they can be reused.
|
|
///
|
|
/// This is particularly efficient when items are often added and
|
|
/// seldom removed.
|
|
///
|
|
/// Items are keyed by an Id, which can be any type you wish, but
|
|
/// must be convertible to/from a `usize`. This promotes using a
|
|
/// custom Id type to talk about items in the map.
|
|
pub struct DenseMap<Id, T> {
|
|
/// List of retired indexes that can be re-used
|
|
retired: Vec<usize>,
|
|
/// All items
|
|
items: Vec<Option<T>>,
|
|
/// Our ID type
|
|
_id_type: std::marker::PhantomData<Id>,
|
|
}
|
|
|
|
impl<Id, T> DenseMap<Id, T>
|
|
where
|
|
Id: From<usize> + Copy,
|
|
usize: From<Id>,
|
|
{
|
|
pub fn new() -> Self {
|
|
DenseMap {
|
|
retired: Vec::new(),
|
|
items: Vec::new(),
|
|
_id_type: std::marker::PhantomData,
|
|
}
|
|
}
|
|
|
|
pub fn add(&mut self, item: T) -> Id {
|
|
self.add_with(|_| item)
|
|
}
|
|
|
|
pub fn as_slice(&self) -> &[Option<T>] {
|
|
&self.items
|
|
}
|
|
|
|
pub fn add_with<F>(&mut self, f: F) -> Id
|
|
where
|
|
F: FnOnce(Id) -> T,
|
|
{
|
|
match self.retired.pop() {
|
|
Some(id) => {
|
|
let id_out = id.into();
|
|
self.items[id] = Some(f(id_out));
|
|
id_out
|
|
}
|
|
None => {
|
|
let id = self.items.len().into();
|
|
self.items.push(Some(f(id)));
|
|
id
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn get(&self, id: Id) -> Option<&T> {
|
|
let id: usize = id.into();
|
|
self.items.get(id).and_then(|item| item.as_ref())
|
|
}
|
|
|
|
pub fn get_mut(&mut self, id: Id) -> Option<&mut T> {
|
|
let id: usize = id.into();
|
|
self.items.get_mut(id).and_then(|item| item.as_mut())
|
|
}
|
|
|
|
pub fn remove(&mut self, id: Id) -> Option<T> {
|
|
let id: usize = id.into();
|
|
let old = self.items.get_mut(id).and_then(|item| item.take());
|
|
|
|
if old.is_some() {
|
|
// something was actually removed, so lets add the id to
|
|
// the list of retired ids!
|
|
self.retired.push(id);
|
|
}
|
|
|
|
old
|
|
}
|
|
|
|
pub fn iter(&self) -> impl Iterator<Item = (Id, &T)> + '_ {
|
|
self.items
|
|
.iter()
|
|
.enumerate()
|
|
.filter_map(|(id, item)| Some((id.into(), item.as_ref()?)))
|
|
}
|
|
|
|
pub fn iter_mut(&mut self) -> impl Iterator<Item = (Id, &mut T)> + '_ {
|
|
self.items
|
|
.iter_mut()
|
|
.enumerate()
|
|
.filter_map(|(id, item)| Some((id.into(), item.as_mut()?)))
|
|
}
|
|
|
|
pub fn into_iter(self) -> impl Iterator<Item = (Id, T)> {
|
|
self.items
|
|
.into_iter()
|
|
.enumerate()
|
|
.filter_map(|(id, item)| Some((id.into(), item?)))
|
|
}
|
|
|
|
pub fn len(&self) -> usize {
|
|
self.items.len() - self.retired.len()
|
|
}
|
|
|
|
pub fn is_empty(&self) -> bool {
|
|
self.len() == 0
|
|
}
|
|
|
|
/// Return the next Id that will be assigned.
|
|
pub fn next_id(&self) -> usize {
|
|
match self.retired.last() {
|
|
Some(id) => *id,
|
|
None => self.items.len(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn len_doesnt_panic_if_lots_of_ids_are_retired() {
|
|
let mut map = DenseMap::<usize, usize>::new();
|
|
|
|
let id1 = map.add(1);
|
|
let id2 = map.add(2);
|
|
let id3 = map.add(3);
|
|
|
|
assert_eq!(map.len(), 3);
|
|
|
|
map.remove(id1);
|
|
map.remove(id2);
|
|
|
|
assert_eq!(map.len(), 1);
|
|
|
|
map.remove(id3);
|
|
|
|
assert_eq!(map.len(), 0);
|
|
|
|
map.remove(id1);
|
|
map.remove(id1);
|
|
map.remove(id1);
|
|
|
|
assert_eq!(map.len(), 0);
|
|
}
|
|
}
|