use crate::KeyRange;
use crate::comparator::UserComparator;
#[cfg(not(feature = "std"))]
use alloc::vec::Vec;
use core::ops::{Bound, RangeBounds};
pub trait Ranged {
fn key_range(&self) -> &KeyRange;
}
#[expect(dead_code, reason = "planned for cascading index optimization")]
pub struct Indexed<T: Ranged> {
inner: T,
}
impl<T: Ranged> Ranged for Indexed<T> {
fn key_range(&self) -> &KeyRange {
self.inner.key_range()
}
}
impl<T: Ranged> core::ops::Deref for Indexed<T> {
type Target = T;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct Run<T: Ranged>(Vec<T>);
impl<T: Ranged> core::ops::Deref for Run<T> {
type Target = [T];
fn deref(&self) -> &Self::Target {
&self.0
}
}
fn trim_slice<T, F>(s: &[T], pred: F) -> &[T]
where
F: Fn(&T) -> bool,
{
let start = s.iter().position(&pred).unwrap_or(s.len());
let end = s.iter().rposition(&pred).map_or(start, |i| i + 1);
#[expect(
clippy::expect_used,
reason = "start..end are derived from position/rposition on the same slice"
)]
s.get(start..end).expect("should be in range")
}
impl<T: Ranged> Run<T> {
pub fn new(items: Vec<T>) -> Option<Self> {
if items.is_empty() {
None
} else {
Some(Self(items))
}
}
pub fn inner_mut(&mut self) -> &mut Vec<T> {
&mut self.0
}
pub fn push_lexicographic(&mut self, item: T) {
self.0.push(item);
self.0
.sort_by(|a, b| a.key_range().min().cmp(b.key_range().min()));
}
pub fn push_cmp(&mut self, item: T, cmp: &dyn UserComparator) {
self.0.push(item);
self.sort_by_cmp(cmp);
}
pub fn sort_by_cmp(&mut self, cmp: &dyn UserComparator) {
self.0
.sort_by(|a, b| cmp.compare(a.key_range().min(), b.key_range().min()));
}
pub fn extend(&mut self, items: Vec<T>) {
self.0.extend(items);
}
pub fn retain<F>(&mut self, f: F)
where
F: FnMut(&T) -> bool,
{
self.0.retain(f);
}
pub fn remove(&mut self, idx: usize) -> T {
self.0.remove(idx)
}
pub fn get_for_key(&self, key: &[u8]) -> Option<&T> {
let idx = self.partition_point(|x| x.key_range().max() < &key);
self.0.get(idx).filter(|x| x.key_range().min() <= &key)
}
pub fn get_for_key_cmp(
&self,
key: &[u8],
cmp: &dyn crate::comparator::UserComparator,
) -> Option<&T> {
let idx = self.partition_point(|x| {
cmp.compare(x.key_range().max(), key) == core::cmp::Ordering::Less
});
self.0
.get(idx)
.filter(|x| cmp.compare(x.key_range().min(), key) != core::cmp::Ordering::Greater)
}
pub fn aggregate_key_range(&self) -> KeyRange {
#[expect(clippy::expect_used, reason = "by definition, runs are never empty")]
let lo = self.first().expect("run should never be empty");
#[expect(clippy::expect_used, reason = "by definition, runs are never empty")]
let hi = self.last().expect("run should never be empty");
KeyRange::new((lo.key_range().min().clone(), hi.key_range().max().clone()))
}
pub fn get_overlapping<'a>(&'a self, key_range: &'a KeyRange) -> &'a [T] {
let range = key_range.min()..=key_range.max();
let Some((lo, hi)) = self.range_overlap_indexes::<crate::Slice, _>(&range) else {
return &[];
};
self.get(lo..=hi).unwrap_or_default()
}
pub fn get_overlapping_cmp<'a>(
&'a self,
key_range: &'a KeyRange,
cmp: &dyn UserComparator,
) -> &'a [T] {
let range = key_range.min()..=key_range.max();
let Some((lo, hi)) = self.range_overlap_indexes_cmp::<crate::Slice, _>(&range, cmp) else {
return &[];
};
self.get(lo..=hi).unwrap_or_default()
}
pub fn get_contained<'a>(&'a self, key_range: &KeyRange) -> &'a [T] {
let range = key_range.min()..=key_range.max();
let Some((lo, hi)) = self.range_overlap_indexes::<crate::Slice, _>(&range) else {
return &[];
};
self.get(lo..=hi)
.map(|slice| trim_slice(slice, |x| key_range.contains_range(x.key_range())))
.unwrap_or_default()
}
pub fn get_contained_cmp<'a>(
&'a self,
key_range: &KeyRange,
cmp: &dyn UserComparator,
) -> &'a [T] {
let range = key_range.min()..=key_range.max();
let Some((lo, hi)) = self.range_overlap_indexes_cmp::<crate::Slice, _>(&range, cmp) else {
return &[];
};
self.get(lo..=hi)
.map(|slice| trim_slice(slice, |x| key_range.contains_range_cmp(x.key_range(), cmp)))
.unwrap_or_default()
}
pub fn range_overlap_indexes<K: AsRef<[u8]>, R: RangeBounds<K>>(
&self,
key_range: &R,
) -> Option<(usize, usize)> {
let level = &self.0;
let lo = match key_range.start_bound() {
Bound::Unbounded => 0,
Bound::Included(start_key) => {
level.partition_point(|x| x.key_range().max() < start_key)
}
Bound::Excluded(start_key) => {
level.partition_point(|x| x.key_range().max() <= start_key)
}
};
if lo >= level.len() {
return None;
}
#[expect(clippy::indexing_slicing)]
let truncated_level = &level[lo..];
let hi = match key_range.end_bound() {
Bound::Unbounded => level.len() - 1,
Bound::Included(end_key) => {
let idx = lo + truncated_level.partition_point(|x| x.key_range().min() <= end_key);
if idx == 0 {
return None;
}
idx.saturating_sub(1) }
Bound::Excluded(end_key) => {
let idx = lo + truncated_level.partition_point(|x| x.key_range().min() < end_key);
if idx == 0 {
return None;
}
idx.saturating_sub(1) }
};
if lo > hi {
return None;
}
Some((lo, hi))
}
pub fn range_overlap_indexes_cmp<K: AsRef<[u8]>, R: RangeBounds<K>>(
&self,
key_range: &R,
cmp: &dyn UserComparator,
) -> Option<(usize, usize)> {
use core::cmp::Ordering;
let level = &self.0;
let lo = match key_range.start_bound() {
Bound::Unbounded => 0,
Bound::Included(start_key) => level.partition_point(|x| {
cmp.compare(x.key_range().max(), start_key.as_ref()) == Ordering::Less
}),
Bound::Excluded(start_key) => level.partition_point(|x| {
cmp.compare(x.key_range().max(), start_key.as_ref()) != Ordering::Greater
}),
};
if lo >= level.len() {
return None;
}
#[expect(clippy::indexing_slicing)]
let truncated_level = &level[lo..];
let hi = match key_range.end_bound() {
Bound::Unbounded => level.len() - 1,
Bound::Included(end_key) => {
let idx = lo
+ truncated_level.partition_point(|x| {
cmp.compare(x.key_range().min(), end_key.as_ref()) != Ordering::Greater
});
if idx == 0 {
return None;
}
idx.saturating_sub(1)
}
Bound::Excluded(end_key) => {
let idx = lo
+ truncated_level.partition_point(|x| {
cmp.compare(x.key_range().min(), end_key.as_ref()) == Ordering::Less
});
if idx == 0 {
return None;
}
idx.saturating_sub(1)
}
};
if lo > hi {
return None;
}
Some((lo, hi))
}
}
#[cfg(test)]
#[expect(clippy::unwrap_used, clippy::indexing_slicing, reason = "test code")]
mod tests;