use smallvec::{smallvec, SmallVec};
use crate::{i64_key_from_u64_key, u64_key_from_i64_key, RangeI64, RangeU64};
type Level = u64;
#[allow(dead_code)]
mod small_and_slow {
#[allow(clippy::wildcard_imports)]
use super::*;
pub const LEVEL_STEP: u64 = 3;
pub const BOTTOM_LEVEL: Level = 1;
pub const NUM_CHILDREN_IN_DENSE: u64 = 16;
}
mod large_and_fast {
#[allow(clippy::wildcard_imports)]
use super::*;
pub const LEVEL_STEP: u64 = 4;
pub const BOTTOM_LEVEL: Level = 0;
pub const NUM_CHILDREN_IN_DENSE: u64 = 16;
}
use large_and_fast::{BOTTOM_LEVEL, LEVEL_STEP, NUM_CHILDREN_IN_DENSE};
const ROOT_LEVEL: Level = 64 - LEVEL_STEP;
static_assertions::const_assert_eq!(ROOT_LEVEL + LEVEL_STEP, 64);
static_assertions::const_assert_eq!((ROOT_LEVEL - BOTTOM_LEVEL) % LEVEL_STEP, 0);
const NUM_NODE_STEPS: u64 = (ROOT_LEVEL - BOTTOM_LEVEL) / LEVEL_STEP;
#[allow(dead_code)] const NUM_STEPS_IN_DENSE_LEAF: u64 = 64 - NUM_NODE_STEPS * LEVEL_STEP;
static_assertions::const_assert_eq!(1 << NUM_STEPS_IN_DENSE_LEAF, NUM_CHILDREN_IN_DENSE);
const ADDR_MASK: u64 = (1 << LEVEL_STEP) - 1;
const NUM_CHILDREN_IN_NODE: u64 = 1 << LEVEL_STEP;
const MAX_SPARSE_LEAF_LEN: usize = 32;
fn child_level_and_size(level: Level) -> (Level, u64) {
let child_level = level - LEVEL_STEP;
let child_size = if child_level == 0 {
NUM_CHILDREN_IN_DENSE
} else {
1 << level
};
(child_level, child_size)
}
fn range_u64_from_range_bounds(range: impl std::ops::RangeBounds<i64>) -> RangeU64 {
let min = match range.start_bound() {
std::ops::Bound::Included(min) => *min,
std::ops::Bound::Excluded(min) => min.saturating_add(1),
std::ops::Bound::Unbounded => i64::MIN,
};
let max = match range.end_bound() {
std::ops::Bound::Included(min) => *min,
std::ops::Bound::Excluded(min) => min.saturating_sub(1),
std::ops::Bound::Unbounded => i64::MAX,
};
RangeU64 {
min: u64_key_from_i64_key(min),
max: u64_key_from_i64_key(max),
}
}
#[derive(Clone, Debug)]
pub struct Int64Histogram {
root: Node,
}
impl Default for Int64Histogram {
fn default() -> Self {
Self {
root: Node::SparseLeaf(SparseLeaf::default()),
}
}
}
impl Int64Histogram {
pub fn increment(&mut self, key: i64, inc: u32) {
if inc != 0 {
self.root
.increment(ROOT_LEVEL, u64_key_from_i64_key(key), inc);
}
}
pub fn decrement(&mut self, key: i64, dec: u32) -> u32 {
if dec == 0 {
0
} else {
self.root
.decrement(ROOT_LEVEL, u64_key_from_i64_key(key), dec)
}
}
pub fn remove(&mut self, range: impl std::ops::RangeBounds<i64>) -> u64 {
let range = range_u64_from_range_bounds(range);
self.root.remove(0, ROOT_LEVEL, range)
}
pub fn is_empty(&self) -> bool {
self.total_count() == 0
}
pub fn total_count(&self) -> u64 {
self.root.total_count()
}
pub fn min_key(&self) -> Option<i64> {
self.root.min_key(0, ROOT_LEVEL).map(i64_key_from_u64_key)
}
pub fn max_key(&self) -> Option<i64> {
self.root.max_key(0, ROOT_LEVEL).map(i64_key_from_u64_key)
}
pub fn range_count(&self, range: impl std::ops::RangeBounds<i64>) -> u64 {
let range = range_u64_from_range_bounds(range);
if range.min <= range.max {
self.root.range_count(0, ROOT_LEVEL, range)
} else {
0
}
}
pub fn range(&self, range: impl std::ops::RangeBounds<i64>, cutoff_size: u64) -> Iter<'_> {
let range = range_u64_from_range_bounds(range);
Iter {
iter: TreeIterator {
range,
cutoff_size,
stack: smallvec![NodeIterator {
level: ROOT_LEVEL,
abs_addr: 0,
node: &self.root,
index: 0,
}],
},
}
}
}
pub struct Iter<'a> {
iter: TreeIterator<'a>,
}
impl Iterator for Iter<'_> {
type Item = (RangeI64, u64);
#[inline]
fn next(&mut self) -> Option<Self::Item> {
self.iter.next().map(|(range, count)| {
(
RangeI64 {
min: i64_key_from_u64_key(range.min),
max: i64_key_from_u64_key(range.max),
},
count,
)
})
}
}
#[allow(clippy::enum_variant_names)]
#[derive(Clone, Debug)]
enum Node {
BranchNode(BranchNode),
SparseLeaf(SparseLeaf),
DenseLeaf(DenseLeaf),
}
#[derive(Clone, Debug, Default)]
struct BranchNode {
total_count: u64,
children: [Option<Box<Node>>; NUM_CHILDREN_IN_NODE as usize],
}
#[derive(Clone, Debug, Default)]
struct SparseLeaf {
addrs: SmallVec<[u64; 3]>,
counts: SmallVec<[u32; 3]>,
}
#[derive(Clone, Copy, Debug, Default)]
struct DenseLeaf {
counts: [u32; NUM_CHILDREN_IN_DENSE as usize],
}
impl Node {
fn for_level(level: Level) -> Self {
if level == BOTTOM_LEVEL {
Self::DenseLeaf(DenseLeaf::default())
} else {
Self::SparseLeaf(SparseLeaf::default())
}
}
fn increment(&mut self, level: Level, addr: u64, inc: u32) {
match self {
Self::BranchNode(node) => {
node.increment(level, addr, inc);
}
Self::SparseLeaf(sparse) => {
*self = std::mem::take(sparse).increment(level, addr, inc);
}
Self::DenseLeaf(dense) => {
dense.increment(addr, inc);
}
}
}
#[must_use]
fn decrement(&mut self, level: Level, addr: u64, dec: u32) -> u32 {
match self {
Self::BranchNode(node) => {
let count_loss = node.decrement(level, addr, dec);
if node.is_empty() {
*self = Self::SparseLeaf(SparseLeaf::default());
}
count_loss
}
Self::SparseLeaf(sparse) => sparse.decrement(addr, dec),
Self::DenseLeaf(dense) => dense.decrement(addr, dec),
}
}
fn remove(&mut self, my_addr: u64, my_level: Level, range: RangeU64) -> u64 {
match self {
Self::BranchNode(node) => {
let count_loss = node.remove(my_addr, my_level, range);
if node.is_empty() {
*self = Self::SparseLeaf(SparseLeaf::default());
}
count_loss
}
Self::SparseLeaf(sparse) => sparse.remove(range),
Self::DenseLeaf(dense) => dense.remove(my_addr, range),
}
}
fn is_empty(&self) -> bool {
match self {
Self::BranchNode(node) => node.is_empty(),
Self::SparseLeaf(sparse) => sparse.is_empty(),
Self::DenseLeaf(dense) => dense.is_empty(),
}
}
fn total_count(&self) -> u64 {
match self {
Self::BranchNode(node) => node.total_count(),
Self::SparseLeaf(sparse) => sparse.total_count(),
Self::DenseLeaf(dense) => dense.total_count(),
}
}
fn min_key(&self, my_addr: u64, my_level: Level) -> Option<u64> {
match self {
Self::BranchNode(node) => node.min_key(my_addr, my_level),
Self::SparseLeaf(sparse) => sparse.min_key(),
Self::DenseLeaf(dense) => dense.min_key(my_addr),
}
}
fn max_key(&self, my_addr: u64, my_level: Level) -> Option<u64> {
match self {
Self::BranchNode(node) => node.max_key(my_addr, my_level),
Self::SparseLeaf(sparse) => sparse.max_key(),
Self::DenseLeaf(dense) => dense.max_key(my_addr),
}
}
fn range_count(&self, my_addr: u64, my_level: Level, range: RangeU64) -> u64 {
match self {
Self::BranchNode(node) => node.range_count(my_addr, my_level, range),
Self::SparseLeaf(sparse) => sparse.range_count(range),
Self::DenseLeaf(dense) => dense.range_count(my_addr, range),
}
}
}
impl BranchNode {
fn increment(&mut self, level: Level, addr: u64, inc: u32) {
debug_assert!(level != BOTTOM_LEVEL);
let child_level = level - LEVEL_STEP;
let top_addr = (addr >> level) & ADDR_MASK;
self.children[top_addr as usize]
.get_or_insert_with(|| Box::new(Node::for_level(child_level)))
.increment(child_level, addr, inc);
self.total_count += inc as u64;
}
#[must_use]
fn decrement(&mut self, level: Level, addr: u64, dec: u32) -> u32 {
debug_assert!(level != BOTTOM_LEVEL);
let child_level = level - LEVEL_STEP;
let top_addr = (addr >> level) & ADDR_MASK;
if let Some(child) = &mut self.children[top_addr as usize] {
let count_loss = child.decrement(child_level, addr, dec);
if child.is_empty() {
self.children[top_addr as usize] = None;
}
self.total_count -= count_loss as u64;
count_loss
} else {
0
}
}
#[must_use]
fn remove(&mut self, my_addr: u64, my_level: Level, range: RangeU64) -> u64 {
debug_assert!(range.min <= range.max);
debug_assert!(my_level != BOTTOM_LEVEL);
let mut count_loss = 0;
let (child_level, child_size) = child_level_and_size(my_level);
for ci in 0..NUM_CHILDREN_IN_NODE {
let child_addr = my_addr + ci * child_size;
let child_range = RangeU64::new(child_addr, child_addr + (child_size - 1));
if range.intersects(child_range) {
if let Some(child) = &mut self.children[ci as usize] {
if range.contains_all_of(child_range) {
count_loss += child.total_count();
self.children[ci as usize] = None;
} else {
count_loss += child.remove(child_addr, child_level, range);
if child.is_empty() {
self.children[ci as usize] = None;
}
}
}
}
}
self.total_count -= count_loss;
count_loss
}
fn is_empty(&self) -> bool {
self.total_count == 0
}
fn total_count(&self) -> u64 {
self.total_count
}
fn min_key(&self, my_addr: u64, my_level: Level) -> Option<u64> {
debug_assert!(my_level != BOTTOM_LEVEL);
let (child_level, child_size) = child_level_and_size(my_level);
for ci in 0..NUM_CHILDREN_IN_NODE {
let child_addr = my_addr + ci * child_size;
if let Some(child) = &self.children[ci as usize] {
if let Some(min_key) = child.min_key(child_addr, child_level) {
return Some(min_key);
}
}
}
None
}
fn max_key(&self, my_addr: u64, my_level: Level) -> Option<u64> {
debug_assert!(my_level != BOTTOM_LEVEL);
let (child_level, child_size) = child_level_and_size(my_level);
for ci in (0..NUM_CHILDREN_IN_NODE).rev() {
let child_addr = my_addr + ci * child_size;
if let Some(child) = &self.children[ci as usize] {
if let Some(max_key) = child.max_key(child_addr, child_level) {
return Some(max_key);
}
}
}
None
}
fn range_count(&self, my_addr: u64, my_level: Level, range: RangeU64) -> u64 {
debug_assert!(range.min <= range.max);
debug_assert!(my_level != BOTTOM_LEVEL);
let (child_level, child_size) = child_level_and_size(my_level);
let mut total_count = 0;
for ci in 0..NUM_CHILDREN_IN_NODE {
let child_addr = my_addr + ci * child_size;
let child_range = RangeU64::new(child_addr, child_addr + (child_size - 1));
if range.intersects(child_range) {
if let Some(child) = &self.children[ci as usize] {
if range.contains_all_of(child_range) {
total_count += child.total_count();
} else {
total_count += child.range_count(child_addr, child_level, range);
}
}
}
}
total_count
}
}
impl SparseLeaf {
#[must_use]
fn increment(mut self, level: Level, abs_addr: u64, inc: u32) -> Node {
let index = self.addrs.partition_point(|&addr| addr < abs_addr);
if let (Some(addr), Some(count)) = (self.addrs.get_mut(index), self.counts.get_mut(index)) {
if *addr == abs_addr {
*count += inc;
return Node::SparseLeaf(self);
}
}
if self.addrs.len() < MAX_SPARSE_LEAF_LEN {
self.addrs.insert(index, abs_addr);
self.counts.insert(index, inc);
Node::SparseLeaf(self)
} else {
let mut node = self.into_branch_node(level);
node.increment(level, abs_addr, inc);
Node::BranchNode(node)
}
}
#[must_use]
fn into_branch_node(self, level: Level) -> BranchNode {
debug_assert!(level != BOTTOM_LEVEL);
let mut node = BranchNode::default();
for (key, count) in self.addrs.iter().zip(&self.counts) {
node.increment(level, *key, *count);
}
node
}
#[must_use]
fn decrement(&mut self, abs_addr: u64, dec: u32) -> u32 {
debug_assert_eq!(self.addrs.len(), self.counts.len());
let index = self.addrs.partition_point(|&addr| addr < abs_addr);
if let (Some(addr), Some(count)) = (self.addrs.get_mut(index), self.counts.get_mut(index)) {
if *addr == abs_addr {
return if dec < *count {
*count -= dec;
dec
} else {
let count_loss = *count;
self.addrs.remove(index);
self.counts.remove(index);
debug_assert_eq!(self.addrs.len(), self.counts.len());
count_loss
};
}
}
0 }
#[must_use]
fn remove(&mut self, range: RangeU64) -> u64 {
debug_assert_eq!(self.addrs.len(), self.counts.len());
let mut count_loss = 0;
for (key, count) in self.addrs.iter().zip(&mut self.counts) {
if range.contains(*key) {
count_loss += *count as u64;
*count = 0;
}
}
self.addrs.retain(|addr| !range.contains(*addr));
self.counts.retain(|count| *count > 0);
debug_assert_eq!(self.addrs.len(), self.counts.len());
count_loss
}
fn is_empty(&self) -> bool {
self.addrs.is_empty() }
fn total_count(&self) -> u64 {
self.counts.iter().map(|&c| c as u64).sum()
}
fn min_key(&self) -> Option<u64> {
self.addrs.first().copied()
}
fn max_key(&self) -> Option<u64> {
self.addrs.last().copied()
}
fn range_count(&self, range: RangeU64) -> u64 {
let mut total = 0;
for (key, count) in self.addrs.iter().zip(&self.counts) {
if range.contains(*key) {
total += *count as u64;
}
}
total
}
}
impl DenseLeaf {
fn increment(&mut self, abs_addr: u64, inc: u32) {
self.counts[(abs_addr & (NUM_CHILDREN_IN_DENSE - 1)) as usize] += inc;
}
#[must_use]
fn decrement(&mut self, abs_addr: u64, dec: u32) -> u32 {
let bucket_index = (abs_addr & (NUM_CHILDREN_IN_DENSE - 1)) as usize;
let bucket = &mut self.counts[bucket_index];
if dec < *bucket {
*bucket -= dec;
dec
} else {
let count_loss = *bucket;
*bucket = 0;
count_loss
}
}
#[must_use]
fn remove(&mut self, my_addr: u64, range: RangeU64) -> u64 {
debug_assert!(range.min <= range.max);
let mut count_loss = 0;
for (i, count) in self.counts.iter_mut().enumerate() {
if range.contains(my_addr + i as u64) {
count_loss += *count as u64;
*count = 0;
}
}
count_loss
}
fn is_empty(&self) -> bool {
self.total_count() == 0
}
fn total_count(&self) -> u64 {
self.counts.iter().map(|&c| c as u64).sum()
}
fn min_key(&self, my_addr: u64) -> Option<u64> {
for (i, count) in self.counts.iter().enumerate() {
if *count > 0 {
return Some(my_addr + i as u64);
}
}
None
}
fn max_key(&self, my_addr: u64) -> Option<u64> {
for (i, count) in self.counts.iter().enumerate().rev() {
if *count > 0 {
return Some(my_addr + i as u64);
}
}
None
}
fn range_count(&self, my_addr: u64, range: RangeU64) -> u64 {
debug_assert!(range.min <= range.max);
let mut total_count = 0;
for (i, count) in self.counts.iter().enumerate() {
if range.contains(my_addr + i as u64) {
total_count += *count as u64;
}
}
total_count
}
}
struct TreeIterator<'a> {
range: RangeU64,
cutoff_size: u64,
stack: SmallVec<[NodeIterator<'a>; (NUM_NODE_STEPS + 1) as usize]>,
}
struct NodeIterator<'a> {
level: Level,
abs_addr: u64,
node: &'a Node,
index: usize,
}
impl Iterator for TreeIterator<'_> {
type Item = (RangeU64, u64);
fn next(&mut self) -> Option<Self::Item> {
'outer: while let Some(it) = self.stack.last_mut() {
match it.node {
Node::BranchNode(node) => {
let (child_level, child_size) = child_level_and_size(it.level);
while it.index < NUM_CHILDREN_IN_NODE as _ {
let child_addr = it.abs_addr + child_size * it.index as u64;
let child_range = RangeU64 {
min: child_addr,
max: child_addr + (child_size - 1),
};
if self.range.intersects(child_range) {
if let Some(Some(child)) = node.children.get(it.index) {
it.index += 1;
if child_size <= self.cutoff_size
&& self.range.contains_all_of(child_range)
{
if let (Some(min_key), Some(max_key)) = (
child.min_key(child_addr, child_level),
child.max_key(child_addr, child_level),
) {
return Some((
RangeU64::new(min_key, max_key),
child.total_count(),
));
} else {
unreachable!(
"A `BranchNode` can only have non-empty children"
);
}
}
self.stack.push(NodeIterator {
level: child_level,
abs_addr: child_addr,
node: child,
index: 0,
});
continue 'outer;
}
}
it.index += 1;
}
}
Node::SparseLeaf(sparse) => {
while let (Some(abs_addr), Some(count)) =
(sparse.addrs.get(it.index), sparse.counts.get(it.index))
{
it.index += 1;
if self.range.contains(*abs_addr) {
return Some((RangeU64::single(*abs_addr), *count as u64));
}
}
}
Node::DenseLeaf(dense) => {
while let Some(count) = dense.counts.get(it.index) {
let abs_addr = it.abs_addr + it.index as u64;
it.index += 1;
if 0 < *count && self.range.contains(abs_addr) {
return Some((RangeU64::single(abs_addr), *count as u64));
}
}
}
}
self.stack.pop();
}
None
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_dense() {
let mut set = Int64Histogram::default();
debug_assert_eq!(set.min_key(), None);
debug_assert_eq!(set.max_key(), None);
let mut expected_ranges = vec![];
for i in 0..100 {
debug_assert_eq!(set.total_count(), i);
debug_assert_eq!(set.range_count(-10000..10000), i);
let key = i as i64;
set.increment(key, 1);
expected_ranges.push((RangeI64::single(key), 1));
debug_assert_eq!(set.min_key(), Some(0));
debug_assert_eq!(set.max_key(), Some(key));
}
assert_eq!(set.range(.., 1).collect::<Vec<_>>(), expected_ranges);
assert_eq!(set.range(..10, 1).count(), 10);
assert_eq!(set.decrement(5, 1), 1);
assert_eq!(set.range(..10, 1).count(), 9);
assert_eq!(set.decrement(5, 1), 0);
assert_eq!(set.range(..10, 1).count(), 9);
}
#[test]
fn test_sparse() {
let inc = 2;
let spacing = 1_000_000;
let mut set = Int64Histogram::default();
let mut expected_ranges = vec![];
for i in 0..100 {
debug_assert_eq!(set.total_count(), inc * i);
debug_assert_eq!(set.range_count(-10000..10000 * spacing), inc * i);
let key = i as i64 * spacing;
set.increment(key, inc as u32);
expected_ranges.push((RangeI64::single(key), inc));
debug_assert_eq!(set.min_key(), Some(0));
debug_assert_eq!(set.max_key(), Some(key));
}
assert_eq!(set.range(.., 1).collect::<Vec<_>>(), expected_ranges);
assert_eq!(set.range(..10 * spacing, 1).count(), 10);
}
#[test]
fn test_two_dense_ranges() {
let mut set = Int64Histogram::default();
for i in 0..100 {
set.increment(i, 1);
set.increment(10_000 + i, 1);
set.increment(20_000 + i, 1);
debug_assert_eq!(set.min_key(), Some(0));
debug_assert_eq!(set.max_key(), Some(20_000 + i));
}
assert_eq!(set.range(..15_000, 1000).count(), 2);
assert_eq!(set.total_count(), 300);
assert_eq!(set.remove(..10_020), 120);
assert_eq!(set.total_count(), 180);
}
#[test]
fn test_two_sparse_ranges() {
let mut set = Int64Histogram::default();
let mut should_contain = vec![];
let mut should_not_contain = vec![];
for i in 0..100 {
let a = -1_000_000_000 + i * 1_000;
let b = (i - 50) * 1_000;
let c = 1_000_000_000 + i * 1_000;
set.increment(a, 1);
set.increment(b, 1);
set.increment(c, 1);
should_contain.push(a);
should_contain.push(b);
should_not_contain.push(c);
}
let ranges = set.range(..1_000_000_000, 1_000_000).collect::<Vec<_>>();
assert!(ranges.len() < 10, "We shouldn't get too many ranges");
let ranges_contains = |value| ranges.iter().any(|(range, _count)| range.contains(value));
for value in should_contain {
assert!(ranges_contains(value));
}
for value in should_not_contain {
assert!(!ranges_contains(value));
}
assert_eq!(set.total_count(), 300);
assert_eq!(set.remove(..0), 150);
assert_eq!(set.total_count(), 150);
}
fn glue_adjacent_ranges(ranges: &[(RangeI64, u64)], cutoff_size: u64) -> Vec<(RangeI64, u64)> {
if ranges.is_empty() {
return vec![];
}
let mut it = ranges.iter();
let mut result = vec![*it.next().unwrap()];
for &(new_range, new_count) in it {
let (last_range, last_count) = result.last_mut().unwrap();
if new_range.min.abs_diff(last_range.max) < cutoff_size {
*last_count += new_count;
last_range.max = new_range.max;
} else {
result.push((new_range, new_count));
}
}
result
}
#[test]
fn test_ranges_are_tight() {
let mut set = Int64Histogram::default();
for i in 1..=99 {
set.increment(10_000_000 + i * 1000, 1);
set.increment(500_000_000 + i * 1000, 1);
set.increment(9_000_000_000 + i * 1000, 1);
}
let cutoff_size = 100_000;
let ranges = set.range(.., cutoff_size).collect::<Vec<_>>();
assert!(ranges.len() <= 10, "We shouldn't get too many ranges");
let ranges = glue_adjacent_ranges(&ranges, cutoff_size);
assert_eq!(
ranges,
vec![
(RangeI64::new(10_001_000, 10_099_000), 99),
(RangeI64::new(500_001_000, 500_099_000), 99),
(RangeI64::new(9_000_001_000, 9_000_099_000), 99),
]
);
}
#[test]
fn test_removal() {
let mut set = Int64Histogram::default();
set.increment(i64::MAX, 1);
set.increment(i64::MAX - 1, 2);
set.increment(i64::MAX - 2, 3);
set.increment(i64::MIN + 2, 3);
set.increment(i64::MIN + 1, 2);
set.increment(i64::MIN, 1);
debug_assert_eq!(set.min_key(), Some(i64::MIN));
debug_assert_eq!(set.max_key(), Some(i64::MAX));
debug_assert_eq!(set.range_count((i64::MAX - 1)..=i64::MAX), 3);
debug_assert_eq!(
set.range(0.., 1).collect::<Vec<_>>(),
vec![
(RangeI64::single(i64::MAX - 2), 3),
(RangeI64::single(i64::MAX - 1), 2),
(RangeI64::single(i64::MAX), 1),
]
);
set.remove(i64::MAX..=i64::MAX);
debug_assert_eq!(set.min_key(), Some(i64::MIN));
debug_assert_eq!(set.max_key(), Some(i64::MAX - 1));
debug_assert_eq!(
set.range(.., 1).collect::<Vec<_>>(),
vec![
(RangeI64::single(i64::MIN), 1),
(RangeI64::single(i64::MIN + 1), 2),
(RangeI64::single(i64::MIN + 2), 3),
(RangeI64::single(i64::MAX - 2), 3),
(RangeI64::single(i64::MAX - 1), 2),
]
);
set.remove(i64::MIN..=(i64::MAX - 2));
debug_assert_eq!(set.min_key(), Some(i64::MAX - 1));
debug_assert_eq!(set.max_key(), Some(i64::MAX - 1));
debug_assert_eq!(
set.range(.., 1).collect::<Vec<_>>(),
vec![(RangeI64::single(i64::MAX - 1), 2),]
);
}
#[test]
fn test_decrement() {
let mut set = Int64Histogram::default();
for i in 0..100 {
set.increment(i, 2);
}
assert_eq!((set.min_key(), set.max_key()), (Some(0), Some(99)));
assert_eq!(set.range(.., 1).count(), 100);
for i in 0..100 {
assert_eq!(set.decrement(i, 1), 1);
}
assert_eq!((set.min_key(), set.max_key()), (Some(0), Some(99)));
assert_eq!(set.range(.., 1).count(), 100);
for i in 0..50 {
assert_eq!(set.decrement(i, 1), 1);
}
assert_eq!((set.min_key(), set.max_key()), (Some(50), Some(99)));
assert_eq!(set.range(.., 1).count(), 50);
for i in 0..50 {
assert_eq!(
set.decrement(i, 1),
0,
"Should already have been decremented"
);
}
assert_eq!((set.min_key(), set.max_key()), (Some(50), Some(99)));
assert_eq!(set.range(.., 1).count(), 50);
for i in 50..99 {
assert_eq!(set.decrement(i, 1), 1);
}
assert_eq!((set.min_key(), set.max_key()), (Some(99), Some(99)));
assert_eq!(set.range(.., 1).count(), 1);
assert_eq!(set.decrement(99, 1), 1);
assert_eq!((set.min_key(), set.max_key()), (None, None));
assert_eq!(set.range(.., 1).count(), 0);
}
}