risingwave_common/range.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
// Copyright 2024 RisingWave Labs
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
use std::ops::{Add, Bound, RangeBounds, Sub};
mod private {
pub trait ZeroOne {
fn zero() -> Self;
fn one() -> Self;
}
macro_rules! impl_one {
($($t:ty),*) => {
$(
impl ZeroOne for $t {
fn zero() -> Self {
0 as $t
}
fn one() -> Self {
1 as $t
}
}
)*
};
}
macro_rules! for_all_num_type {
($macro:ident) => {
$macro! { u8, u16, u32, u64, usize, i8, i16, i32, i64, isize, f32, f64 }
};
}
for_all_num_type! { impl_one }
}
use private::ZeroOne;
pub trait Idx = PartialOrd<Self>
+ Add<Output = Self>
+ Sub<Output = Self>
+ Clone
+ Copy
+ Send
+ Sync
+ 'static
+ ZeroOne;
pub trait RangeBoundsExt<T: Idx>: RangeBounds<T> {
fn start(&self) -> Option<T> {
match self.start_bound() {
Bound::Included(v) => Some(*v),
Bound::Excluded(v) => Some(*v + ZeroOne::one()),
Bound::Unbounded => None,
}
}
fn end(&self) -> Option<T> {
match self.end_bound() {
Bound::Included(v) => Some(*v + ZeroOne::one()),
Bound::Excluded(v) => Some(*v),
Bound::Unbounded => None,
}
}
fn len(&self) -> Option<T> {
let start = self.start()?;
let end = self.end()?;
Some(end - start)
}
fn is_empty(&self) -> bool {
match self.len() {
Some(len) => len == ZeroOne::zero(),
None => false,
}
}
fn is_full(&self) -> bool {
self.start_bound() == Bound::Unbounded && self.end_bound() == Bound::Unbounded
}
fn map<F, R>(&self, f: F) -> (Bound<R>, Bound<R>)
where
F: Fn(&T) -> R,
{
(self.start_bound().map(&f), self.end_bound().map(&f))
}
}
impl<T: Idx, RB: RangeBounds<T>> RangeBoundsExt<T> for RB {}