use std::ops::Neg;
use crate::interval_arithmetic::Interval;
use arrow::compute::SortOptions;
use arrow::datatypes::DataType;
#[derive(PartialEq, Debug, Clone, Copy, Default)]
pub enum SortProperties {
Ordered(SortOptions),
#[default]
Unordered,
Singleton,
}
impl SortProperties {
pub fn add(&self, rhs: &Self) -> Self {
match (self, rhs) {
(Self::Singleton, _) => *rhs,
(_, Self::Singleton) => *self,
(Self::Ordered(lhs), Self::Ordered(rhs))
if lhs.descending == rhs.descending =>
{
Self::Ordered(SortOptions {
descending: lhs.descending,
nulls_first: lhs.nulls_first || rhs.nulls_first,
})
}
_ => Self::Unordered,
}
}
pub fn sub(&self, rhs: &Self) -> Self {
match (self, rhs) {
(Self::Singleton, Self::Singleton) => Self::Singleton,
(Self::Singleton, Self::Ordered(rhs)) => Self::Ordered(SortOptions {
descending: !rhs.descending,
nulls_first: rhs.nulls_first,
}),
(_, Self::Singleton) => *self,
(Self::Ordered(lhs), Self::Ordered(rhs))
if lhs.descending != rhs.descending =>
{
Self::Ordered(SortOptions {
descending: lhs.descending,
nulls_first: lhs.nulls_first || rhs.nulls_first,
})
}
_ => Self::Unordered,
}
}
pub fn gt_or_gteq(&self, rhs: &Self) -> Self {
match (self, rhs) {
(Self::Singleton, Self::Ordered(rhs)) => Self::Ordered(SortOptions {
descending: !rhs.descending,
nulls_first: rhs.nulls_first,
}),
(_, Self::Singleton) => *self,
(Self::Ordered(lhs), Self::Ordered(rhs))
if lhs.descending != rhs.descending =>
{
*self
}
_ => Self::Unordered,
}
}
pub fn and_or(&self, rhs: &Self) -> Self {
match (self, rhs) {
(Self::Ordered(lhs), Self::Ordered(rhs))
if lhs.descending == rhs.descending =>
{
Self::Ordered(SortOptions {
descending: lhs.descending,
nulls_first: lhs.nulls_first || rhs.nulls_first,
})
}
(Self::Ordered(opt), Self::Singleton)
| (Self::Singleton, Self::Ordered(opt)) => Self::Ordered(SortOptions {
descending: opt.descending,
nulls_first: opt.nulls_first,
}),
(Self::Singleton, Self::Singleton) => Self::Singleton,
_ => Self::Unordered,
}
}
}
impl Neg for SortProperties {
type Output = Self;
fn neg(mut self) -> Self::Output {
if let SortProperties::Ordered(SortOptions { descending, .. }) = &mut self {
*descending = !*descending;
}
self
}
}
#[derive(Debug, Clone)]
pub struct ExprProperties {
pub sort_properties: SortProperties,
pub range: Interval,
}
impl ExprProperties {
pub fn new_unknown() -> Self {
Self {
sort_properties: SortProperties::default(),
range: Interval::make_unbounded(&DataType::Null).unwrap(),
}
}
pub fn with_order(mut self, order: SortProperties) -> Self {
self.sort_properties = order;
self
}
pub fn with_range(mut self, range: Interval) -> Self {
self.range = range;
self
}
}