surrealpatch/lib/src/sql/part.rs

305 lines
6.6 KiB
Rust
Raw Normal View History

use crate::sql::comment::shouldbespace;
2023-05-10 02:08:09 +00:00
use crate::sql::common::{closebracket, openbracket};
use crate::sql::ending::ident as ending;
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
use crate::sql::fmt::Fmt;
use crate::sql::graph::{self, Graph};
use crate::sql::ident::{self, Ident};
2023-07-15 07:08:26 +00:00
use crate::sql::idiom::{self, Idiom};
use crate::sql::number::{number, Number};
2023-07-15 07:08:26 +00:00
use crate::sql::param::{self};
use crate::sql::strand::{self, no_nul_bytes};
use crate::sql::value::{self, Value};
use nom::branch::alt;
use nom::bytes::complete::tag;
use nom::bytes::complete::tag_no_case;
use nom::character::complete::char;
2023-07-15 07:08:26 +00:00
use nom::combinator::{map, not, peek};
use serde::{Deserialize, Serialize};
use std::fmt;
use std::str;
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
pub enum Part {
All,
Flatten,
Last,
First,
Field(Ident),
Index(Number),
Where(Value),
Graph(Graph),
Value(Value),
2023-07-15 07:08:26 +00:00
Start(Value),
Method(#[serde(with = "no_nul_bytes")] String, Vec<Value>),
}
impl From<i32> for Part {
fn from(v: i32) -> Self {
Self::Index(v.into())
}
}
impl From<isize> for Part {
fn from(v: isize) -> Self {
Self::Index(v.into())
}
}
impl From<usize> for Part {
fn from(v: usize) -> Self {
Self::Index(v.into())
}
}
impl From<String> for Part {
fn from(v: String) -> Self {
Self::Field(v.into())
}
}
impl From<Number> for Part {
fn from(v: Number) -> Self {
Self::Index(v)
}
}
impl From<Ident> for Part {
fn from(v: Ident) -> Self {
Self::Field(v)
}
}
impl From<Graph> for Part {
fn from(v: Graph) -> Self {
Self::Graph(v)
}
}
impl From<&str> for Part {
fn from(v: &str) -> Self {
match v.parse::<isize>() {
Ok(v) => Self::from(v),
_ => Self::from(v.to_owned()),
}
}
}
impl Part {
/// Check if we require a writeable transaction
pub(crate) fn writeable(&self) -> bool {
match self {
2023-07-15 07:08:26 +00:00
Part::Start(v) => v.writeable(),
Part::Where(v) => v.writeable(),
Part::Value(v) => v.writeable(),
Part::Method(_, v) => v.iter().any(Value::writeable),
_ => false,
}
}
2022-10-19 09:55:19 +00:00
/// Returns a yield if an alias is specified
pub(crate) fn alias(&self) -> Option<&Idiom> {
match self {
Part::Graph(v) => v.alias.as_ref(),
_ => None,
}
}
}
impl fmt::Display for Part {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Part::All => f.write_str("[*]"),
Part::Last => f.write_str("[$]"),
Part::First => f.write_str("[0]"),
2023-07-15 07:08:26 +00:00
Part::Start(v) => write!(f, "{v}"),
2023-02-03 11:47:07 +00:00
Part::Field(v) => write!(f, ".{v}"),
Part::Flatten => f.write_str(""),
2023-02-03 11:47:07 +00:00
Part::Index(v) => write!(f, "[{v}]"),
Part::Where(v) => write!(f, "[WHERE {v}]"),
Part::Graph(v) => write!(f, "{v}"),
2023-07-15 07:08:26 +00:00
Part::Value(v) => write!(f, "[{v}]"),
Part::Method(v, a) => write!(f, ".{v}({})", Fmt::comma_separated(a)),
}
}
}
// ------------------------------
pub trait Next<'a> {
fn next(&'a self) -> &[Part];
}
impl<'a> Next<'a> for &'a [Part] {
fn next(&'a self) -> &'a [Part] {
match self.len() {
0 => &[],
_ => &self[1..],
}
}
}
// ------------------------------
pub fn part(i: &str) -> IResult<&str, Part> {
alt((all, flatten, last, index, field, value, graph, filter))(i)
}
pub fn first(i: &str) -> IResult<&str, Part> {
let (i, _) = peek(not(number))(i)?;
let (i, v) = ident::ident(i)?;
let (i, _) = ending(i)?;
Ok((i, Part::Field(v)))
}
pub fn all(i: &str) -> IResult<&str, Part> {
let (i, _) = alt((
|i| {
let (i, _) = char('.')(i)?;
let (i, _) = char('*')(i)?;
Ok((i, ()))
},
|i| {
2023-05-10 02:08:09 +00:00
let (i, _) = openbracket(i)?;
let (i, _) = char('*')(i)?;
2023-05-10 02:08:09 +00:00
let (i, _) = closebracket(i)?;
Ok((i, ()))
},
))(i)?;
Ok((i, Part::All))
}
pub fn last(i: &str) -> IResult<&str, Part> {
2023-05-10 02:08:09 +00:00
let (i, _) = openbracket(i)?;
let (i, _) = char('$')(i)?;
2023-05-10 02:08:09 +00:00
let (i, _) = closebracket(i)?;
Ok((i, Part::Last))
}
pub fn index(i: &str) -> IResult<&str, Part> {
2023-05-10 02:08:09 +00:00
let (i, _) = openbracket(i)?;
let (i, v) = number(i)?;
2023-05-10 02:08:09 +00:00
let (i, _) = closebracket(i)?;
Ok((i, Part::Index(v)))
}
pub fn flatten(i: &str) -> IResult<&str, Part> {
let (i, _) = alt((tag(""), tag("...")))(i)?;
let (i, _) = ending(i)?;
Ok((i, Part::Flatten))
}
pub fn field(i: &str) -> IResult<&str, Part> {
let (i, _) = char('.')(i)?;
let (i, v) = ident::ident(i)?;
let (i, _) = ending(i)?;
Ok((i, Part::Field(v)))
}
pub fn filter(i: &str) -> IResult<&str, Part> {
2023-05-10 02:08:09 +00:00
let (i, _) = openbracket(i)?;
let (i, _) = alt((tag_no_case("WHERE"), tag("?")))(i)?;
let (i, _) = shouldbespace(i)?;
let (i, v) = value::value(i)?;
2023-05-10 02:08:09 +00:00
let (i, _) = closebracket(i)?;
Ok((i, Part::Where(v)))
}
pub fn value(i: &str) -> IResult<&str, Part> {
2023-07-15 07:08:26 +00:00
let (i, _) = openbracket(i)?;
let (i, v) = alt((
map(strand::strand, Value::Strand),
map(param::param, Value::Param),
map(idiom::basic, Value::Idiom),
))(i)?;
let (i, _) = closebracket(i)?;
Ok((i, Part::Value(v)))
}
2023-07-15 07:08:26 +00:00
pub fn start(i: &str) -> IResult<&str, Part> {
let (i, v) = value::start(i)?;
Ok((i, Part::Start(v)))
}
pub fn graph(i: &str) -> IResult<&str, Part> {
let (i, v) = graph::graph(i)?;
Ok((i, Part::Graph(v)))
}
#[cfg(test)]
mod tests {
use super::*;
use crate::sql::expression::Expression;
use crate::sql::test::Parse;
#[test]
fn part_all() {
let sql = "[*]";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("[*]", format!("{}", out));
assert_eq!(out, Part::All);
}
#[test]
fn part_last() {
let sql = "[$]";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("[$]", format!("{}", out));
assert_eq!(out, Part::Last);
}
#[test]
fn part_flatten() {
let sql = "...";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("", format!("{}", out));
assert_eq!(out, Part::Flatten);
}
#[test]
fn part_flatten_ellipsis() {
let sql = "";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("", format!("{}", out));
assert_eq!(out, Part::Flatten);
}
#[test]
fn part_number() {
let sql = "[0]";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("[0]", format!("{}", out));
assert_eq!(out, Part::Index(Number::from(0)));
}
#[test]
fn part_expression_question() {
let sql = "[? test = true]";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("[WHERE test = true]", format!("{}", out));
assert_eq!(out, Part::Where(Value::from(Expression::parse("test = true"))));
}
#[test]
fn part_expression_condition() {
let sql = "[WHERE test = true]";
let res = part(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("[WHERE test = true]", format!("{}", out));
assert_eq!(out, Part::Where(Value::from(Expression::parse("test = true"))));
}
}