use super::Id;
use crate::cnf::GENERATION_ALLOCATION_LIMIT;
use crate::ctx::Context;
use crate::dbs::Options;
use crate::doc::CursorDoc;
use crate::err::Error;
use crate::sql::{Number, Subquery, Value};
use crate::syn;
use reblessive::tree::Stk;
use revision::revisioned;
use serde::{Deserialize, Serialize};
use std::cmp::Ordering;
use std::fmt;
use std::ops::Bound;
use std::str::FromStr;
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Range";
#[revisioned(revision = 1)]
#[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize, Hash)]
#[serde(rename = "$surrealdb::private::sql::Range")]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
#[non_exhaustive]
pub struct Range {
pub beg: Bound<Value>,
pub end: Bound<Value>,
}
impl FromStr for Range {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::try_from(s)
}
}
impl TryFrom<&str> for Range {
type Error = ();
fn try_from(v: &str) -> Result<Self, Self::Error> {
match syn::range(v) {
Ok(v) => Ok(v),
_ => Err(()),
}
}
}
impl From<(Bound<Id>, Bound<Id>)> for Range {
fn from(v: (Bound<Id>, Bound<Id>)) -> Self {
fn convert(v: Bound<Id>) -> Bound<Value> {
match v {
Bound::Included(v) => Bound::Included(v.into()),
Bound::Excluded(v) => Bound::Excluded(v.into()),
Bound::Unbounded => Bound::Unbounded,
}
}
Self {
beg: convert(v.0),
end: convert(v.1),
}
}
}
impl TryInto<std::ops::Range<i64>> for Range {
type Error = Error;
fn try_into(self) -> Result<std::ops::Range<i64>, Self::Error> {
let beg = match self.beg {
Bound::Unbounded => i64::MIN,
Bound::Included(beg) => to_i64(beg)?,
Bound::Excluded(beg) => to_i64(beg)? + 1,
};
let end = match self.end {
Bound::Unbounded => i64::MAX,
Bound::Included(end) => to_i64(end)? + 1,
Bound::Excluded(end) => to_i64(end)?,
};
if (beg + *GENERATION_ALLOCATION_LIMIT as i64) < end {
Err(Error::RangeTooBig {
max: *GENERATION_ALLOCATION_LIMIT,
})
} else {
Ok(beg..end)
}
}
}
impl Range {
pub fn new(beg: Bound<Value>, end: Bound<Value>) -> Self {
Self {
beg,
end,
}
}
pub(crate) async fn compute(
&self,
stk: &mut Stk,
ctx: &Context,
opt: &Options,
doc: Option<&CursorDoc>,
) -> Result<Value, Error> {
Ok(Value::Range(Box::new(Range {
beg: match &self.beg {
Bound::Included(v) => Bound::Included(v.compute(stk, ctx, opt, doc).await?),
Bound::Excluded(v) => Bound::Excluded(v.compute(stk, ctx, opt, doc).await?),
Bound::Unbounded => Bound::Unbounded,
},
end: match &self.end {
Bound::Included(v) => Bound::Included(v.compute(stk, ctx, opt, doc).await?),
Bound::Excluded(v) => Bound::Excluded(v.compute(stk, ctx, opt, doc).await?),
Bound::Unbounded => Bound::Unbounded,
},
})))
}
pub fn validate_computed(&self) -> Result<(), Error> {
match &self.beg {
Bound::Included(ref v) | Bound::Excluded(ref v) => v.validate_computed()?,
Bound::Unbounded => {}
}
match &self.end {
Bound::Included(ref v) | Bound::Excluded(ref v) => v.validate_computed()?,
Bound::Unbounded => {}
}
Ok(())
}
}
impl PartialOrd for Range {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl Ord for Range {
fn cmp(&self, other: &Self) -> Ordering {
match &self.beg {
Bound::Unbounded => match &other.beg {
Bound::Unbounded => Ordering::Equal,
_ => Ordering::Less,
},
Bound::Included(v) => match &other.beg {
Bound::Unbounded => Ordering::Greater,
Bound::Included(w) => match v.cmp(w) {
Ordering::Equal => match &self.end {
Bound::Unbounded => match &other.end {
Bound::Unbounded => Ordering::Equal,
_ => Ordering::Greater,
},
Bound::Included(v) => match &other.end {
Bound::Unbounded => Ordering::Less,
Bound::Included(w) => v.cmp(w),
_ => Ordering::Greater,
},
Bound::Excluded(v) => match &other.end {
Bound::Excluded(w) => v.cmp(w),
_ => Ordering::Less,
},
},
ordering => ordering,
},
_ => Ordering::Less,
},
Bound::Excluded(v) => match &other.beg {
Bound::Excluded(w) => match v.cmp(w) {
Ordering::Equal => match &self.end {
Bound::Unbounded => match &other.end {
Bound::Unbounded => Ordering::Equal,
_ => Ordering::Greater,
},
Bound::Included(v) => match &other.end {
Bound::Unbounded => Ordering::Less,
Bound::Included(w) => v.cmp(w),
_ => Ordering::Greater,
},
Bound::Excluded(v) => match &other.end {
Bound::Excluded(w) => v.cmp(w),
_ => Ordering::Less,
},
},
ordering => ordering,
},
_ => Ordering::Greater,
},
}
}
}
impl fmt::Display for Range {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fn bound_value(v: &Value) -> Value {
if v.can_be_range_bound() {
v.to_owned()
} else {
Value::Subquery(Box::new(Subquery::Value(v.to_owned())))
}
}
match &self.beg {
Bound::Unbounded => write!(f, ""),
Bound::Included(v) => write!(f, "{}", bound_value(v)),
Bound::Excluded(v) => write!(f, "{}>", bound_value(v)),
}?;
match &self.end {
Bound::Unbounded => write!(f, ".."),
Bound::Excluded(v) => write!(f, "..{}", bound_value(v)),
Bound::Included(v) => write!(f, "..={}", bound_value(v)),
}?;
Ok(())
}
}
fn to_i64(v: Value) -> Result<i64, Error> {
match v {
Value::Number(Number::Int(v)) => Ok(v),
v => Err(Error::InvalidRangeValue {
expected: "int".to_string(),
found: v.kindof().to_string(),
}),
}
}
#[revisioned(revision = 1)]
#[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize, Hash)]
#[serde(rename = "$surrealdb::private::sql::Range")]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
#[non_exhaustive]
pub struct OldRange {
pub tb: String,
pub beg: Bound<Id>,
pub end: Bound<Id>,
}