surrealpatch/src/sql/order.rs
2021-12-14 10:24:54 +00:00

262 lines
5.4 KiB
Rust

use crate::sql::comment::shouldbespace;
use crate::sql::common::commas;
use crate::sql::idiom::{idiom, Idiom};
use nom::branch::alt;
use nom::bytes::complete::tag_no_case;
use nom::combinator::{map, opt};
use nom::multi::separated_list1;
use nom::sequence::tuple;
use nom::IResult;
use serde::{Deserialize, Serialize};
use std::fmt;
#[derive(Clone, Debug, Default, Eq, PartialEq, Serialize, Deserialize)]
pub struct Orders(Vec<Order>);
impl fmt::Display for Orders {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"ORDER BY {}",
self.0.iter().map(|ref v| format!("{}", v)).collect::<Vec<_>>().join(", ")
)
}
}
#[derive(Clone, Debug, Default, Eq, PartialEq, Serialize, Deserialize)]
pub struct Order {
pub order: Idiom,
pub random: bool,
pub collate: bool,
pub numeric: bool,
pub direction: bool,
}
impl fmt::Display for Order {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}", self.order)?;
if self.random {
write!(f, "RAND()")?;
}
if self.collate {
write!(f, " COLLATE")?;
}
if self.numeric {
write!(f, " NUMERIC")?;
}
match self.direction {
false => write!(f, " DESC")?,
true => (),
};
Ok(())
}
}
pub fn order(i: &str) -> IResult<&str, Orders> {
let (i, _) = tag_no_case("ORDER")(i)?;
let (i, _) = opt(tuple((shouldbespace, tag_no_case("BY"))))(i)?;
let (i, _) = shouldbespace(i)?;
let (i, v) = alt((order_rand, separated_list1(commas, order_raw)))(i)?;
Ok((i, Orders(v)))
}
fn order_rand(i: &str) -> IResult<&str, Vec<Order>> {
let (i, _) = tag_no_case("RAND()")(i)?;
Ok((
i,
vec![Order {
order: Default::default(),
random: true,
collate: false,
numeric: false,
direction: true,
}],
))
}
fn order_raw(i: &str) -> IResult<&str, Order> {
let (i, v) = idiom(i)?;
let (i, c) = opt(tuple((shouldbespace, tag_no_case("COLLATE"))))(i)?;
let (i, n) = opt(tuple((shouldbespace, tag_no_case("NUMERIC"))))(i)?;
let (i, d) = opt(alt((
map(tuple((shouldbespace, tag_no_case("ASC"))), |_| true),
map(tuple((shouldbespace, tag_no_case("DESC"))), |_| false),
)))(i)?;
Ok((
i,
Order {
order: v,
random: false,
collate: c.is_some(),
numeric: n.is_some(),
direction: d.unwrap_or(true),
},
))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn order_statement() {
let sql = "ORDER field";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: false,
numeric: false,
direction: true,
}])
);
assert_eq!("ORDER BY field", format!("{}", out));
}
#[test]
fn order_statement_by() {
let sql = "ORDER BY field";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: false,
numeric: false,
direction: true,
}])
);
assert_eq!("ORDER BY field", format!("{}", out));
}
#[test]
fn order_statement_random() {
let sql = "ORDER RAND()";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Default::default(),
random: true,
collate: false,
numeric: false,
direction: true,
}])
);
assert_eq!("ORDER BY RAND()", format!("{}", out));
}
#[test]
fn order_statement_multiple() {
let sql = "ORDER field, other.field";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![
Order {
order: Idiom::from("field"),
random: false,
collate: false,
numeric: false,
direction: true,
},
Order {
order: Idiom::from("other.field"),
random: false,
collate: false,
numeric: false,
direction: true,
},
])
);
assert_eq!("ORDER BY field, other.field", format!("{}", out));
}
#[test]
fn order_statement_collate() {
let sql = "ORDER field COLLATE";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: true,
numeric: false,
direction: true,
}])
);
assert_eq!("ORDER BY field COLLATE", format!("{}", out));
}
#[test]
fn order_statement_numeric() {
let sql = "ORDER field NUMERIC";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: false,
numeric: true,
direction: true,
}])
);
assert_eq!("ORDER BY field NUMERIC", format!("{}", out));
}
#[test]
fn order_statement_direction() {
let sql = "ORDER field DESC";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: false,
numeric: false,
direction: false,
}])
);
assert_eq!("ORDER BY field DESC", format!("{}", out));
}
#[test]
fn order_statement_all() {
let sql = "ORDER field COLLATE NUMERIC DESC";
let res = order(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(
out,
Orders(vec![Order {
order: Idiom::from("field"),
random: false,
collate: true,
numeric: true,
direction: false,
}])
);
assert_eq!("ORDER BY field COLLATE NUMERIC DESC", format!("{}", out));
}
}