surrealpatch/lib/src/sql/thing.rs

250 lines
4.9 KiB
Rust
Raw Normal View History

use crate::ctx::Context;
use crate::dbs::Options;
use crate::err::Error;
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
use crate::sql::escape::escape_rid;
use crate::sql::id::{id, Id};
2020-06-29 15:36:01 +00:00
use crate::sql::ident::ident_raw;
use crate::sql::strand::Strand;
use crate::sql::value::Value;
2022-04-09 09:09:01 +00:00
use derive::Store;
use nom::branch::alt;
use nom::bytes::complete::tag;
use nom::character::complete::char;
use nom::combinator::map;
2022-10-16 20:35:50 +00:00
use nom::sequence::delimited;
2020-06-29 15:36:01 +00:00
use serde::{Deserialize, Serialize};
use std::fmt;
use std::str::FromStr;
2020-06-29 15:36:01 +00:00
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Thing";
2023-04-29 15:58:22 +00:00
#[derive(Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize, Store, Hash)]
#[serde(rename = "$surrealdb::private::sql::Thing")]
2020-06-29 15:36:01 +00:00
pub struct Thing {
pub tb: String,
pub id: Id,
}
impl From<(&str, Id)> for Thing {
fn from((tb, id): (&str, Id)) -> Self {
Self {
tb: tb.to_owned(),
id,
}
}
}
impl From<(String, Id)> for Thing {
fn from((tb, id): (String, Id)) -> Self {
Self {
tb,
id,
}
}
2020-06-29 15:36:01 +00:00
}
2022-03-07 18:11:44 +00:00
impl From<(String, String)> for Thing {
fn from((tb, id): (String, String)) -> Self {
Self::from((tb, Id::from(id)))
}
}
impl From<(&str, &str)> for Thing {
fn from((tb, id): (&str, &str)) -> Self {
Self::from((tb.to_owned(), Id::from(id)))
}
}
impl FromStr for Thing {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::try_from(s)
}
}
impl TryFrom<String> for Thing {
type Error = ();
fn try_from(v: String) -> Result<Self, Self::Error> {
Self::try_from(v.as_str())
}
}
impl TryFrom<Strand> for Thing {
type Error = ();
fn try_from(v: Strand) -> Result<Self, Self::Error> {
Self::try_from(v.as_str())
}
}
impl TryFrom<&str> for Thing {
type Error = ();
fn try_from(v: &str) -> Result<Self, Self::Error> {
match thing_raw(v) {
Ok((_, v)) => Ok(v),
_ => Err(()),
}
}
}
impl Thing {
2022-10-19 09:55:19 +00:00
/// Convert the Thing to a raw String
pub fn to_raw(&self) -> String {
self.to_string()
}
}
2020-06-29 15:36:01 +00:00
impl fmt::Display for Thing {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}:{}", escape_rid(&self.tb), self.id)
2021-03-29 15:43:37 +00:00
}
}
impl Thing {
/// Process this type returning a computed simple Value
pub(crate) async fn compute(&self, ctx: &Context<'_>, opt: &Options) -> Result<Value, Error> {
Ok(Value::Thing(Thing {
tb: self.tb.clone(),
id: self.id.compute(ctx, opt).await?,
}))
}
}
2020-06-29 15:36:01 +00:00
pub fn thing(i: &str) -> IResult<&str, Thing> {
2022-10-16 20:35:50 +00:00
alt((thing_raw, thing_single, thing_double))(i)
}
2022-10-16 20:35:50 +00:00
fn thing_single(i: &str) -> IResult<&str, Thing> {
delimited(char('\''), thing_raw, char('\''))(i)
}
2022-10-16 20:35:50 +00:00
fn thing_double(i: &str) -> IResult<&str, Thing> {
delimited(char('\"'), thing_raw, char('\"'))(i)
}
fn thing_raw(i: &str) -> IResult<&str, Thing> {
2020-06-29 15:36:01 +00:00
let (i, t) = ident_raw(i)?;
let (i, _) = char(':')(i)?;
let (i, v) = alt((
map(tag("rand()"), |_| Id::rand()),
map(tag("ulid()"), |_| Id::ulid()),
map(tag("uuid()"), |_| Id::uuid()),
id,
))(i)?;
2020-06-29 15:36:01 +00:00
Ok((
i,
Thing {
tb: t,
id: v,
2020-06-29 15:36:01 +00:00
},
))
}
#[cfg(test)]
mod tests {
use super::*;
use crate::sql::array::Array;
use crate::sql::object::Object;
use crate::sql::value::Value;
2020-06-29 15:36:01 +00:00
#[test]
fn thing_normal() {
let sql = "test:id";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("test:id", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::from("id"),
2020-06-29 15:36:01 +00:00
}
);
}
#[test]
fn thing_integer() {
let sql = "test:001";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("test:1", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::from(1),
}
);
}
2020-06-29 15:36:01 +00:00
#[test]
fn thing_quoted_backtick() {
let sql = "`test`:`id`";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("test:id", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::from("id"),
2020-06-29 15:36:01 +00:00
}
);
}
#[test]
fn thing_quoted_brackets() {
let sql = "⟨test⟩:⟨id⟩";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("test:id", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::from("id"),
2020-06-29 15:36:01 +00:00
}
);
}
#[test]
fn thing_object() {
let sql = "test:{ location: 'GBR', year: 2022 }";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
2022-10-19 14:48:50 +00:00
assert_eq!("test:{ location: 'GBR', year: 2022 }", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::Object(Object::from(map! {
"location".to_string() => Value::from("GBR"),
"year".to_string() => Value::from(2022),
})),
}
);
}
#[test]
fn thing_array() {
let sql = "test:['GBR', 2022]";
let res = thing(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
2022-10-19 14:48:50 +00:00
assert_eq!("test:['GBR', 2022]", format!("{}", out));
assert_eq!(
out,
Thing {
tb: String::from("test"),
id: Id::Array(Array::from(vec![Value::from("GBR"), Value::from(2022)])),
}
);
}
2020-06-29 15:36:01 +00:00
}