surrealpatch/lib/src/sql/split.rs

106 lines
2.5 KiB
Rust
Raw Normal View History

2020-06-29 15:36:01 +00:00
use crate::sql::comment::shouldbespace;
use crate::sql::common::commas;
2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
use crate::sql::fmt::Fmt;
use crate::sql::idiom::{basic, Idiom};
2020-06-29 15:36:01 +00:00
use nom::bytes::complete::tag_no_case;
use nom::combinator::{cut, opt};
2021-03-29 15:43:37 +00:00
use nom::multi::separated_list1;
use nom::sequence::terminated;
use revision::revisioned;
2020-06-29 15:36:01 +00:00
use serde::{Deserialize, Serialize};
use std::fmt::{self, Display, Formatter};
2022-03-25 18:43:22 +00:00
use std::ops::Deref;
2020-06-29 15:36:01 +00:00
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
#[revisioned(revision = 1)]
pub struct Splits(pub Vec<Split>);
2020-06-29 15:36:01 +00:00
2022-03-25 18:43:22 +00:00
impl Deref for Splits {
type Target = Vec<Split>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl IntoIterator for Splits {
type Item = Split;
type IntoIter = std::vec::IntoIter<Self::Item>;
fn into_iter(self) -> Self::IntoIter {
self.0.into_iter()
}
}
2020-06-29 15:36:01 +00:00
impl fmt::Display for Splits {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "SPLIT ON {}", Fmt::comma_separated(&self.0))
2020-06-29 15:36:01 +00:00
}
}
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
#[revisioned(revision = 1)]
pub struct Split(pub Idiom);
impl Deref for Split {
type Target = Idiom;
fn deref(&self) -> &Self::Target {
&self.0
}
2020-06-29 15:36:01 +00:00
}
impl Display for Split {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
Display::fmt(&self.0, f)
2020-06-29 15:36:01 +00:00
}
}
pub fn split(i: &str) -> IResult<&str, Splits> {
let (i, _) = tag_no_case("SPLIT")(i)?;
let (i, _) = shouldbespace(i)?;
2023-09-08 11:28:36 +00:00
let (i, _) = opt(terminated(tag_no_case("ON"), shouldbespace))(i)?;
let (i, v) = cut(separated_list1(commas, split_raw))(i)?;
Ok((i, Splits(v)))
2020-06-29 15:36:01 +00:00
}
fn split_raw(i: &str) -> IResult<&str, Split> {
let (i, v) = basic(i)?;
Ok((i, Split(v)))
2020-06-29 15:36:01 +00:00
}
#[cfg(test)]
mod tests {
use super::*;
use crate::sql::test::Parse;
2020-06-29 15:36:01 +00:00
#[test]
fn split_statement() {
let sql = "SPLIT field";
let res = split(sql);
let out = res.unwrap().1;
assert_eq!(out, Splits(vec![Split(Idiom::parse("field"))]),);
2020-06-29 15:36:01 +00:00
assert_eq!("SPLIT ON field", format!("{}", out));
}
#[test]
fn split_statement_on() {
let sql = "SPLIT ON field";
let res = split(sql);
let out = res.unwrap().1;
assert_eq!(out, Splits(vec![Split(Idiom::parse("field"))]),);
2020-06-29 15:36:01 +00:00
assert_eq!("SPLIT ON field", format!("{}", out));
}
#[test]
fn split_statement_multiple() {
let sql = "SPLIT field, other.field";
let res = split(sql);
let out = res.unwrap().1;
assert_eq!(
out,
Splits(vec![Split(Idiom::parse("field")), Split(Idiom::parse("other.field")),])
2020-06-29 15:36:01 +00:00
);
assert_eq!("SPLIT ON field, other.field", format!("{}", out));
}
}