use crate::sql::comment::shouldbespace; use crate::sql::ending::ident as ending; use crate::sql::error::IResult; use crate::sql::graph::{graph as graph_raw, Graph}; use crate::sql::ident::{ident, Ident}; use crate::sql::idiom::Idiom; use crate::sql::number::{number, Number}; use crate::sql::thing::{thing as thing_raw, Thing}; use crate::sql::value::{value, Value}; use nom::branch::alt; use nom::bytes::complete::tag; use nom::bytes::complete::tag_no_case; use nom::character::complete::char; use serde::{Deserialize, Serialize}; use std::fmt; use std::str; #[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize)] pub enum Part { All, Last, First, Field(Ident), Index(Number), Where(Value), Thing(Thing), Graph(Graph), } impl From for Part { fn from(v: i32) -> Self { Self::Index(v.into()) } } impl From for Part { fn from(v: isize) -> Self { Self::Index(v.into()) } } impl From for Part { fn from(v: usize) -> Self { Self::Index(v.into()) } } impl From for Part { fn from(v: Number) -> Self { Self::Index(v) } } impl From for Part { fn from(v: Ident) -> Self { Self::Field(v) } } impl From for Part { fn from(v: Value) -> Self { Self::Where(v) } } impl From for Part { fn from(v: Thing) -> Self { Self::Thing(v) } } impl From for Part { fn from(v: Graph) -> Self { Self::Graph(v) } } impl From for Part { fn from(v: String) -> Self { Self::Field(Ident(v)) } } impl From<&str> for Part { fn from(v: &str) -> Self { match v.parse::() { Ok(v) => Self::from(v), _ => Self::from(v.to_owned()), } } } impl Part { /// 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]"), Part::Field(v) => write!(f, ".{}", v), Part::Index(v) => write!(f, "[{}]", v), Part::Where(v) => write!(f, "[WHERE {}]", v), Part::Thing(v) => write!(f, "{}", v), Part::Graph(v) => write!(f, "{}", v), } } } // ------------------------------ 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, last, index, field, graph, filter))(i) } pub fn first(i: &str) -> IResult<&str, Part> { let (i, v) = 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| { let (i, _) = char('[')(i)?; let (i, _) = char('*')(i)?; let (i, _) = char(']')(i)?; Ok((i, ())) }, ))(i)?; Ok((i, Part::All)) } pub fn last(i: &str) -> IResult<&str, Part> { let (i, _) = char('[')(i)?; let (i, _) = char('$')(i)?; let (i, _) = char(']')(i)?; Ok((i, Part::Last)) } pub fn index(i: &str) -> IResult<&str, Part> { let (i, _) = char('[')(i)?; let (i, v) = number(i)?; let (i, _) = char(']')(i)?; Ok((i, Part::Index(v))) } pub fn field(i: &str) -> IResult<&str, Part> { let (i, _) = char('.')(i)?; let (i, v) = ident(i)?; let (i, _) = ending(i)?; Ok((i, Part::Field(v))) } pub fn filter(i: &str) -> IResult<&str, Part> { let (i, _) = char('[')(i)?; let (i, _) = alt((tag_no_case("WHERE"), tag("?")))(i)?; let (i, _) = shouldbespace(i)?; let (i, v) = value(i)?; let (i, _) = char(']')(i)?; Ok((i, Part::Where(v))) } pub fn thing(i: &str) -> IResult<&str, Part> { let (i, v) = thing_raw(i)?; Ok((i, Part::Thing(v))) } pub fn graph(i: &str) -> IResult<&str, Part> { let (i, v) = graph_raw(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_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")))); } }