kaspa-consensus-core 0.15.0

Kaspa consensus core
Documentation
use crate::{BlockHashSet, HashMapCustomHasher};
use kaspa_hashes::{Hash, HASH_SIZE};
use std::sync::Arc;

pub type BlockHashes = Arc<Vec<Hash>>;

/// `blockhash::NONE` is a hash which is used in rare cases as the `None` block hash
pub const NONE: Hash = Hash::from_bytes([0u8; HASH_SIZE]);

/// `blockhash::ORIGIN` is a special hash representing a `virtual genesis` block.
/// It serves as a special local block which all locally-known
/// blocks are in its future.
pub const ORIGIN: Hash = Hash::from_bytes([0xfe; HASH_SIZE]);

pub trait BlockHashExtensions {
    fn is_none(&self) -> bool;
    fn is_origin(&self) -> bool;
}

impl BlockHashExtensions for Hash {
    fn is_none(&self) -> bool {
        self.eq(&NONE)
    }

    fn is_origin(&self) -> bool {
        self.eq(&ORIGIN)
    }
}

/// Generates a unique block hash for each call to this function.
/// To be used for test purposes only.
pub fn new_unique() -> Hash {
    use std::sync::atomic::{AtomicU64, Ordering};
    static COUNTER: AtomicU64 = AtomicU64::new(1);
    let c = COUNTER.fetch_add(1, Ordering::Relaxed);
    Hash::from_u64_word(c)
}

pub trait BlockHashIteratorExtensions: Iterator<Item = Hash> {
    /// Copy of itertools::unique, adapted for block hashes (uses `BlockHashSet` under the hood)
    ///
    /// Returns an iterator adaptor that filters out hashes that have
    /// already been produced once during the iteration.
    ///
    /// Clones of visited elements are stored in a hash set in the
    /// iterator.
    ///
    /// The iterator is stable, returning the non-duplicate items in the order
    /// in which they occur in the adapted iterator. In a set of duplicate
    /// items, the first item encountered is the item retained.
    ///
    /// NOTE: currently usages are expected to contain no duplicates, hence we alloc the expected capacity
    fn block_unique(self) -> BlockUnique<Self>
    where
        Self: Sized,
    {
        let (lower, _) = self.size_hint();
        BlockUnique { iter: self, seen: BlockHashSet::with_capacity(lower) }
    }
}

impl<T: ?Sized> BlockHashIteratorExtensions for T where T: Iterator<Item = Hash> {}

#[derive(Clone)]
pub struct BlockUnique<I: Iterator<Item = Hash>> {
    iter: I,
    seen: BlockHashSet,
}

impl<I> Iterator for BlockUnique<I>
where
    I: Iterator<Item = Hash>,
{
    type Item = I::Item;

    fn next(&mut self) -> Option<Self::Item> {
        self.iter.by_ref().find(|&hash| self.seen.insert(hash))
    }

    #[inline]
    fn size_hint(&self) -> (usize, Option<usize>) {
        let (low, hi) = self.iter.size_hint();
        ((low > 0 && self.seen.is_empty()) as usize, hi)
    }
}

impl<I> DoubleEndedIterator for BlockUnique<I>
where
    I: DoubleEndedIterator<Item = Hash>,
{
    fn next_back(&mut self) -> Option<Self::Item> {
        self.iter.by_ref().rev().find(|&hash| self.seen.insert(hash))
    }
}

impl<I> std::iter::FusedIterator for BlockUnique<I> where I: Iterator<Item = Hash> + std::iter::FusedIterator {}