#![allow(dead_code)]
use std::{
cmp,
collections::{btree_map, BTreeMap},
fmt::{Debug, Error as FmtError, Formatter},
iter::{FromIterator, FusedIterator},
ops::{Bound, Range},
};
#[derive(Clone)]
pub struct RangeMap<K, V> {
btm: BTreeMap<K, Entry<K, V>>,
}
#[derive(Clone)]
struct Entry<K, V> {
end: K,
value: V,
}
impl<K, V> Default for RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
fn default() -> Self {
Self::new()
}
}
impl<K, V> RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
#[inline]
pub fn new() -> Self {
RangeMap {
btm: BTreeMap::new(),
}
}
#[inline]
pub fn get(&self, key: &K) -> Option<&V> {
self.get_key_value(key).map(|(_range, value)| value)
}
#[inline]
pub fn get_key_value(&self, key: &K) -> Option<(Range<K>, &V)> {
self.btm
.range((Bound::Unbounded, Bound::Included(key)))
.next_back()
.filter(|(_start, Entry { end, value: _ })| {
end > key
})
.map(|(start, Entry { end, value })| (start.clone()..end.clone(), value))
}
#[inline]
pub fn contains_key(&self, key: &K) -> bool {
self.get(key).is_some()
}
#[inline]
pub fn contains_any(&self, range: &Range<K>) -> bool {
self.range(range).next().is_some()
}
#[inline]
pub fn iter(&self) -> Iter<'_, K, V> {
Iter {
inner: self.btm.iter(),
}
}
#[inline]
pub fn iter_mut(&mut self) -> IterMut<'_, K, V> {
IterMut {
inner: self.btm.iter_mut(),
}
}
pub fn insert(&mut self, mut range: Range<K>, value: V) {
assert!(range.start < range.end);
let new_value = value;
let mut candidates = self
.btm
.range((Bound::Unbounded, Bound::Included(&range.start)))
.rev()
.take(2)
.filter(|(start, Entry { end, value: _ })| {
(*start..end).touches(&(&range.start..&range.end))
});
if let Some(mut candidate) = candidates.next() {
if let Some(another_candidate) = candidates.next() {
candidate = another_candidate;
}
let (stored_start, stored_entry) = (candidate.0.clone(), candidate.1.clone());
self.adjust_touching_ranges_for_insert(
stored_start,
stored_entry,
&mut range,
&new_value,
);
}
while let Some((start, entry)) = self
.btm
.range((Bound::Included(&range.start), Bound::Included(&range.end)))
.next()
{
#[allow(clippy::suspicious_operation_groupings)]
if start == &range.end && entry.value != new_value {
break;
}
let stored_start = start.clone();
let stored_entry = entry.clone();
self.adjust_touching_ranges_for_insert(
stored_start,
stored_entry,
&mut range,
&new_value,
);
}
self.btm.insert(
range.start,
Entry {
end: range.end,
value: new_value,
},
);
}
pub fn remove(&mut self, range: Range<K>) {
assert!(range.start < range.end);
if let Some((stored_start, stored_entry)) = self
.btm
.range((Bound::Unbounded, Bound::Included(&range.start)))
.next_back()
.filter(|(start, Entry { end, value: _ })| {
(*start..end).overlaps(&(&range.start..&range.end))
})
.map(|(stored_start, stored_entry)| (stored_start.clone(), stored_entry.clone()))
{
self.adjust_overlapping_ranges_for_remove(stored_start, stored_entry, &range);
}
while let Some((stored_start, stored_entry)) = self
.btm
.range((Bound::Excluded(&range.start), Bound::Excluded(&range.end)))
.next()
.map(|(stored_start, stored_entry)| (stored_start.clone(), stored_entry.clone()))
{
self.adjust_overlapping_ranges_for_remove(stored_start, stored_entry, &range);
}
}
fn adjust_touching_ranges_for_insert(
&mut self,
stored_start: K,
stored_entry: Entry<K, V>,
new_range: &mut Range<K>,
new_value: &V,
) {
if stored_entry.value == *new_value {
new_range.start = cmp::min(&new_range.start, &stored_start).clone();
new_range.end = cmp::max(&new_range.end, &stored_entry.end).clone();
self.btm.remove(&stored_start);
} else {
if new_range.overlaps(&(stored_start.clone()..stored_entry.end.clone())) {
self.btm.remove(&stored_start);
if stored_start < new_range.start {
self.btm.insert(
stored_start,
Entry {
end: new_range.start.clone(),
value: stored_entry.value.clone(),
},
);
}
if stored_entry.end > new_range.end {
self.btm.insert(new_range.end.clone(), stored_entry);
}
} else {
}
}
}
fn adjust_overlapping_ranges_for_remove(
&mut self,
stored_start: K,
stored_entry: Entry<K, V>,
range_to_remove: &Range<K>,
) {
self.btm.remove(&stored_start);
if stored_start < range_to_remove.start {
self.btm.insert(
stored_start,
Entry {
end: range_to_remove.start.clone(),
value: stored_entry.value.clone(),
},
);
}
if stored_entry.end > range_to_remove.end {
self.btm.insert(range_to_remove.end.clone(), stored_entry);
}
}
pub fn split_at(&mut self, key: &K) {
let bounds = (Bound::Unbounded, Bound::Excluded(key.clone()));
if let Some((_start, entry)) = self
.btm
.range_mut(bounds)
.next_back()
.filter(|(_start, Entry { end, value: _ })| end > key)
{
let second_half_entry = entry.clone();
entry.end = key.clone();
self.btm.insert(key.clone(), second_half_entry);
}
}
pub fn range(&self, range: &Range<K>) -> RangeIter<'_, K, V> {
let start = self
.get_key_value(&range.start)
.map_or(range.start.clone(), |(k, _v)| k.start);
let end = range.end.clone();
RangeIter {
inner: self
.btm
.range((Bound::Included(start), Bound::Excluded(end))),
}
}
pub fn range_mut(&mut self, range: &Range<K>) -> RangeMutIter<'_, K, V> {
let start = self
.get_key_value(&range.start)
.map_or(range.start.clone(), |(k, _v)| k.start);
let end = range.end.clone();
RangeMutIter {
inner: self
.btm
.range_mut((Bound::Included(start), Bound::Excluded(end))),
}
}
}
pub struct Iter<'a, K, V> {
inner: btree_map::Iter<'a, K, Entry<K, V>>,
}
impl<'a, K, V> Iterator for Iter<'a, K, V>
where
K: 'a + Clone,
V: 'a,
{
type Item = (Range<K>, &'a V);
fn next(&mut self) -> Option<Self::Item> {
self.inner
.next()
.map(|(start, Entry { end, value })| (start.clone()..end.clone(), value))
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<K, V> FusedIterator for Iter<'_, K, V> where K: Ord + Clone {}
impl<K, V> ExactSizeIterator for Iter<'_, K, V> where K: Ord + Clone {}
pub struct IterMut<'a, K, V> {
inner: btree_map::IterMut<'a, K, Entry<K, V>>,
}
impl<'a, K, V> Iterator for IterMut<'a, K, V>
where
K: 'a + Clone,
V: 'a,
{
type Item = (Range<K>, &'a mut V);
fn next(&mut self) -> Option<Self::Item> {
self.inner
.next()
.map(|(start, Entry { end, value })| (start.clone()..end.clone(), value))
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<K, V> FusedIterator for IterMut<'_, K, V> where K: Ord + Clone {}
impl<K, V> ExactSizeIterator for IterMut<'_, K, V> where K: Ord + Clone {}
pub struct IntoIter<K, V> {
inner: btree_map::IntoIter<K, Entry<K, V>>,
}
impl<K, V> IntoIterator for RangeMap<K, V> {
type Item = (Range<K>, V);
type IntoIter = IntoIter<K, V>;
fn into_iter(self) -> Self::IntoIter {
IntoIter {
inner: self.btm.into_iter(),
}
}
}
impl<K, V> Iterator for IntoIter<K, V> {
type Item = (Range<K>, V);
fn next(&mut self) -> Option<Self::Item> {
self.inner
.next()
.map(|(start, Entry { end, value })| (start..end, value))
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<K, V> FusedIterator for IntoIter<K, V> where K: Ord + Clone {}
impl<K, V> ExactSizeIterator for IntoIter<K, V> where K: Ord + Clone {}
impl<K: Debug, V: Debug> Debug for RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), FmtError> {
f.debug_map().entries(self.iter()).finish()
}
}
impl<K, V> FromIterator<(Range<K>, V)> for RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
fn from_iter<T: IntoIterator<Item = (Range<K>, V)>>(iter: T) -> Self {
let mut range_map = RangeMap::new();
range_map.extend(iter);
range_map
}
}
impl<K, V> Extend<(Range<K>, V)> for RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
fn extend<T: IntoIterator<Item = (Range<K>, V)>>(&mut self, iter: T) {
iter.into_iter().for_each(move |(k, v)| {
self.insert(k, v);
})
}
}
pub struct RangeIter<'a, K, V> {
inner: btree_map::Range<'a, K, Entry<K, V>>,
}
impl<K, V> FusedIterator for RangeIter<'_, K, V> where K: Ord + Clone {}
impl<'a, K, V> Iterator for RangeIter<'a, K, V>
where
K: 'a + Clone,
V: 'a,
{
type Item = (Range<K>, &'a V);
fn next(&mut self) -> Option<Self::Item> {
self.inner
.next()
.map(|(start, Entry { end, value })| (start.clone()..end.clone(), value))
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
pub struct RangeMutIter<'a, K, V> {
inner: btree_map::RangeMut<'a, K, Entry<K, V>>,
}
impl<K, V> FusedIterator for RangeMutIter<'_, K, V> where K: Ord + Clone {}
impl<'a, K, V> Iterator for RangeMutIter<'a, K, V>
where
K: 'a + Clone,
V: 'a,
{
type Item = (Range<K>, &'a mut V);
fn next(&mut self) -> Option<Self::Item> {
self.inner
.next()
.map(|(start, Entry { end, value })| (start.clone()..end.clone(), value))
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
pub trait RangeExt<T> {
fn overlaps(&self, other: &Self) -> bool;
fn touches(&self, other: &Self) -> bool;
}
impl<T> RangeExt<T> for Range<T>
where
T: Ord,
{
fn overlaps(&self, other: &Self) -> bool {
cmp::max(&self.start, &other.start) < cmp::min(&self.end, &other.end)
}
fn touches(&self, other: &Self) -> bool {
cmp::max(&self.start, &other.start) <= cmp::min(&self.end, &other.end)
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::{format, vec, vec::Vec};
trait RangeMapExt<K, V> {
fn to_vec(&self) -> Vec<(Range<K>, V)>;
}
impl<K, V> RangeMapExt<K, V> for RangeMap<K, V>
where
K: Ord + Clone,
V: Eq + Clone,
{
fn to_vec(&self) -> Vec<(Range<K>, V)> {
self.iter().map(|(kr, v)| (kr, v.clone())).collect()
}
}
#[test]
fn empty_map_is_empty() {
let range_map: RangeMap<u32, bool> = RangeMap::new();
assert_eq!(range_map.to_vec(), vec![]);
}
#[test]
fn insert_into_empty_map() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(0..50, false);
assert_eq!(range_map.to_vec(), vec![(0..50, false)]);
}
#[test]
fn new_same_value_immediately_following_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..3, false);
range_map.insert(3..5, false);
assert_eq!(range_map.to_vec(), vec![(1..5, false)]);
}
#[test]
fn new_different_value_immediately_following_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..3, false);
range_map.insert(3..5, true);
assert_eq!(range_map.to_vec(), vec![(1..3, false), (3..5, true)]);
}
#[test]
fn new_same_value_overlapping_end_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..4, false);
range_map.insert(3..5, false);
assert_eq!(range_map.to_vec(), vec![(1..5, false)]);
}
#[test]
fn new_different_value_overlapping_end_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..4, false);
range_map.insert(3..5, true);
assert_eq!(range_map.to_vec(), vec![(1..3, false), (3..5, true)]);
}
#[test]
fn new_same_value_immediately_preceding_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(3..5, false);
range_map.insert(1..3, false);
assert_eq!(range_map.to_vec(), vec![(1..5, false)]);
}
#[test]
fn new_different_value_immediately_preceding_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(3..5, true);
range_map.insert(1..3, false);
assert_eq!(range_map.to_vec(), vec![(1..3, false), (3..5, true)]);
}
#[test]
fn new_same_value_wholly_inside_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..5, false);
range_map.insert(2..4, false);
assert_eq!(range_map.to_vec(), vec![(1..5, false)]);
}
#[test]
fn new_different_value_wholly_inside_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..5, true);
range_map.insert(2..4, false);
assert_eq!(
range_map.to_vec(),
vec![(1..2, true), (2..4, false), (4..5, true)]
);
}
#[test]
fn replace_at_end_of_existing_range_should_coalesce() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..3, false);
range_map.insert(3..5, true);
range_map.insert(3..5, false);
assert_eq!(range_map.to_vec(), vec![(1..5, false)]);
}
#[test]
fn get() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(0..50, false);
assert_eq!(range_map.get(&49), Some(&false));
assert_eq!(range_map.get(&50), None);
}
#[test]
fn get_key_value() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(0..50, false);
assert_eq!(range_map.get_key_value(&49), Some((0..50, &false)));
assert_eq!(range_map.get_key_value(&50), None);
}
#[test]
fn remove_from_empty_map() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.remove(0..50);
assert_eq!(range_map.to_vec(), vec![]);
}
#[test]
fn remove_non_covered_range_before_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(0..25);
assert_eq!(range_map.to_vec(), vec![(25..75, false)]);
}
#[test]
fn remove_non_covered_range_after_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(75..100);
assert_eq!(range_map.to_vec(), vec![(25..75, false)]);
}
#[test]
fn remove_overlapping_start_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(0..30);
assert_eq!(range_map.to_vec(), vec![(30..75, false)]);
}
#[test]
fn remove_middle_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(30..70);
assert_eq!(range_map.to_vec(), vec![(25..30, false), (70..75, false)]);
}
#[test]
fn remove_overlapping_end_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(70..100);
assert_eq!(range_map.to_vec(), vec![(25..70, false)]);
}
#[test]
fn remove_exactly_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(25..75);
assert_eq!(range_map.to_vec(), vec![]);
}
#[test]
fn remove_superset_of_stored() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(25..75, false);
range_map.remove(0..100);
assert_eq!(range_map.to_vec(), vec![]);
}
#[test]
fn map_debug_repr_looks_right() {
let mut map: RangeMap<u32, ()> = RangeMap::new();
assert_eq!(format!("{:?}", map), "{}");
map.insert(2..5, ());
assert_eq!(format!("{:?}", map), "{2..5: ()}");
map.insert(6..7, ());
map.insert(8..9, ());
assert_eq!(format!("{:?}", map), "{2..5: (), 6..7: (), 8..9: ()}");
}
#[test]
fn into_iter_matches_iter() {
let mut range_map: RangeMap<u32, bool> = RangeMap::new();
range_map.insert(1..3, false);
range_map.insert(3..5, true);
let cloned = range_map.to_vec();
let consumed = range_map.into_iter().collect::<Vec<_>>();
assert_eq!(cloned, vec![(1..3, false), (3..5, true)]);
assert_eq!(cloned, consumed);
}
}