205 lines
4.3 KiB
Rust
205 lines
4.3 KiB
Rust
|
use crate::sql::comment::mightbespace;
|
||
|
use crate::sql::comment::shouldbespace;
|
||
|
use crate::sql::idiom::{idiom, Idiom};
|
||
|
use crate::sql::table::{tables, Tables};
|
||
|
use crate::sql::value::{value, Value};
|
||
|
use nom::branch::alt;
|
||
|
use nom::bytes::complete::tag;
|
||
|
use nom::bytes::complete::tag_no_case;
|
||
|
use nom::combinator::opt;
|
||
|
use nom::IResult;
|
||
|
use serde::{Deserialize, Serialize};
|
||
|
use std::fmt;
|
||
|
|
||
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize)]
|
||
|
pub enum Dir {
|
||
|
In,
|
||
|
Out,
|
||
|
Both,
|
||
|
}
|
||
|
|
||
|
impl Default for Dir {
|
||
|
fn default() -> Dir {
|
||
|
Dir::Both
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl fmt::Display for Dir {
|
||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||
|
match self {
|
||
|
Dir::In => write!(f, "<-"),
|
||
|
Dir::Out => write!(f, "->"),
|
||
|
Dir::Both => write!(f, "<->"),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize)]
|
||
|
pub struct Graph {
|
||
|
pub dir: Dir,
|
||
|
pub what: Tables,
|
||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||
|
pub cond: Option<Value>,
|
||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||
|
pub alias: Option<Idiom>,
|
||
|
}
|
||
|
|
||
|
impl fmt::Display for Graph {
|
||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||
|
if self.what.0.len() == 1 && self.cond.is_none() && self.alias.is_none() {
|
||
|
write!(f, "{}{}", self.dir, self.what)
|
||
|
} else {
|
||
|
write!(f, "{}({}", self.dir, self.what)?;
|
||
|
if let Some(ref v) = self.cond {
|
||
|
write!(f, " WHERE {}", v)?
|
||
|
}
|
||
|
if let Some(ref v) = self.alias {
|
||
|
write!(f, " AS {}", v)?
|
||
|
}
|
||
|
write!(f, ")")
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pub fn graph(i: &str) -> IResult<&str, Graph> {
|
||
|
alt((graph_in, graph_out, graph_both))(i)
|
||
|
}
|
||
|
|
||
|
fn graph_in(i: &str) -> IResult<&str, Graph> {
|
||
|
let (i, _) = tag("<-")(i)?;
|
||
|
let (i, (what, cond, alias)) = alt((simple, custom))(i)?;
|
||
|
Ok((
|
||
|
i,
|
||
|
Graph {
|
||
|
dir: Dir::In,
|
||
|
what,
|
||
|
cond,
|
||
|
alias,
|
||
|
},
|
||
|
))
|
||
|
}
|
||
|
|
||
|
fn graph_out(i: &str) -> IResult<&str, Graph> {
|
||
|
let (i, _) = tag("->")(i)?;
|
||
|
let (i, (what, cond, alias)) = alt((simple, custom))(i)?;
|
||
|
Ok((
|
||
|
i,
|
||
|
Graph {
|
||
|
dir: Dir::Out,
|
||
|
what,
|
||
|
cond,
|
||
|
alias,
|
||
|
},
|
||
|
))
|
||
|
}
|
||
|
|
||
|
fn graph_both(i: &str) -> IResult<&str, Graph> {
|
||
|
let (i, _) = tag("<->")(i)?;
|
||
|
let (i, (what, cond, alias)) = alt((simple, custom))(i)?;
|
||
|
Ok((
|
||
|
i,
|
||
|
Graph {
|
||
|
dir: Dir::Both,
|
||
|
what,
|
||
|
cond,
|
||
|
alias,
|
||
|
},
|
||
|
))
|
||
|
}
|
||
|
|
||
|
fn simple(i: &str) -> IResult<&str, (Tables, Option<Value>, Option<Idiom>)> {
|
||
|
let (i, w) = what(i)?;
|
||
|
Ok((i, (w, None, None)))
|
||
|
}
|
||
|
|
||
|
fn custom(i: &str) -> IResult<&str, (Tables, Option<Value>, Option<Idiom>)> {
|
||
|
let (i, _) = tag("(")(i)?;
|
||
|
let (i, _) = mightbespace(i)?;
|
||
|
let (i, w) = what(i)?;
|
||
|
let (i, c) = opt(cond)(i)?;
|
||
|
let (i, a) = opt(alias)(i)?;
|
||
|
let (i, _) = tag(")")(i)?;
|
||
|
Ok((i, (w, c, a)))
|
||
|
}
|
||
|
|
||
|
fn what(i: &str) -> IResult<&str, Tables> {
|
||
|
let (i, v) = tables(i)?;
|
||
|
Ok((i, v))
|
||
|
}
|
||
|
|
||
|
fn cond(i: &str) -> IResult<&str, Value> {
|
||
|
let (i, _) = shouldbespace(i)?;
|
||
|
let (i, _) = tag_no_case("WHERE")(i)?;
|
||
|
let (i, _) = shouldbespace(i)?;
|
||
|
let (i, v) = value(i)?;
|
||
|
Ok((i, v))
|
||
|
}
|
||
|
|
||
|
fn alias(i: &str) -> IResult<&str, Idiom> {
|
||
|
let (i, _) = shouldbespace(i)?;
|
||
|
let (i, _) = tag_no_case("AS")(i)?;
|
||
|
let (i, _) = shouldbespace(i)?;
|
||
|
let (i, v) = idiom(i)?;
|
||
|
Ok((i, v))
|
||
|
}
|
||
|
|
||
|
#[cfg(test)]
|
||
|
mod tests {
|
||
|
|
||
|
use super::*;
|
||
|
|
||
|
#[test]
|
||
|
fn graph_in() {
|
||
|
let sql = "<-likes";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("<-likes", format!("{}", out));
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn graph_out() {
|
||
|
let sql = "->likes";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("->likes", format!("{}", out));
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn graph_both() {
|
||
|
let sql = "<->likes";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("<->likes", format!("{}", out));
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn graph_multiple() {
|
||
|
let sql = "->(likes, follows)";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("->(likes, follows)", format!("{}", out));
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn graph_aliases() {
|
||
|
let sql = "->(likes, follows AS connections)";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("->(likes, follows AS connections)", format!("{}", out));
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn graph_conditions() {
|
||
|
let sql = "->(likes, follows WHERE influencer = true AS connections)";
|
||
|
let res = graph(sql);
|
||
|
assert!(res.is_ok());
|
||
|
let out = res.unwrap().1;
|
||
|
assert_eq!("->(likes, follows WHERE influencer = true AS connections)", format!("{}", out));
|
||
|
}
|
||
|
}
|