2023-05-10 02:08:09 +00:00
|
|
|
use crate::sql::common::{closeparentheses, openparentheses};
|
2022-06-09 08:18:08 +00:00
|
|
|
use crate::sql::dir::{dir, Dir};
|
|
|
|
use crate::sql::error::IResult;
|
|
|
|
use crate::sql::table::{table, tables, Tables};
|
|
|
|
use crate::sql::thing::{thing, Thing};
|
|
|
|
use nom::branch::alt;
|
|
|
|
use nom::character::complete::char;
|
|
|
|
use nom::combinator::map;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use std::fmt;
|
|
|
|
|
2023-03-30 10:41:44 +00:00
|
|
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Edges";
|
|
|
|
|
2023-04-29 15:58:22 +00:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
|
|
|
|
#[serde(rename = "$surrealdb::private::sql::Edges")]
|
2022-06-09 08:18:08 +00:00
|
|
|
pub struct Edges {
|
|
|
|
pub dir: Dir,
|
|
|
|
pub from: Thing,
|
|
|
|
pub what: Tables,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Edges {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match self.what.len() {
|
|
|
|
0 => write!(f, "{}{}?", self.from, self.dir,),
|
|
|
|
1 => write!(f, "{}{}{}", self.from, self.dir, self.what),
|
|
|
|
_ => write!(f, "{}{}({})", self.from, self.dir, self.what),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn edges(i: &str) -> IResult<&str, Edges> {
|
|
|
|
let (i, from) = thing(i)?;
|
|
|
|
let (i, dir) = dir(i)?;
|
|
|
|
let (i, what) = alt((simple, custom))(i)?;
|
|
|
|
Ok((
|
|
|
|
i,
|
|
|
|
Edges {
|
|
|
|
dir,
|
|
|
|
from,
|
|
|
|
what,
|
|
|
|
},
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn simple(i: &str) -> IResult<&str, Tables> {
|
2022-09-27 21:35:30 +00:00
|
|
|
alt((any, one))(i)
|
2022-06-09 08:18:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn custom(i: &str) -> IResult<&str, Tables> {
|
2023-05-10 02:08:09 +00:00
|
|
|
let (i, _) = openparentheses(i)?;
|
2022-06-09 08:18:08 +00:00
|
|
|
let (i, w) = alt((any, tables))(i)?;
|
2023-05-10 02:08:09 +00:00
|
|
|
let (i, _) = closeparentheses(i)?;
|
2022-06-09 08:18:08 +00:00
|
|
|
Ok((i, w))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn one(i: &str) -> IResult<&str, Tables> {
|
|
|
|
let (i, v) = table(i)?;
|
|
|
|
Ok((i, Tables::from(v)))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn any(i: &str) -> IResult<&str, Tables> {
|
|
|
|
map(char('?'), |_| Tables::default())(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn edges_in() {
|
|
|
|
let sql = "person:test<-likes";
|
|
|
|
let res = edges(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("person:test<-likes", format!("{}", out));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn edges_out() {
|
|
|
|
let sql = "person:test->likes";
|
|
|
|
let res = edges(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("person:test->likes", format!("{}", out));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn edges_both() {
|
|
|
|
let sql = "person:test<->likes";
|
|
|
|
let res = edges(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("person:test<->likes", format!("{}", out));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn edges_multiple() {
|
|
|
|
let sql = "person:test->(likes, follows)";
|
|
|
|
let res = edges(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("person:test->(likes, follows)", format!("{}", out));
|
|
|
|
}
|
|
|
|
}
|