2020-06-29 15:36:01 +00:00
|
|
|
use crate::sql::comment::shouldbespace;
|
|
|
|
use crate::sql::cond::{cond, Cond};
|
2022-01-16 20:31:50 +00:00
|
|
|
use crate::sql::error::IResult;
|
2020-06-29 15:36:01 +00:00
|
|
|
use crate::sql::field::{fields, Fields};
|
|
|
|
use crate::sql::group::{group, Groups};
|
|
|
|
use crate::sql::table::{tables, Tables};
|
|
|
|
use nom::bytes::complete::tag_no_case;
|
|
|
|
use nom::combinator::opt;
|
|
|
|
use nom::sequence::preceded;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use std::fmt;
|
|
|
|
|
2022-10-27 12:23:24 +00:00
|
|
|
#[derive(Clone, Debug, Default, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
|
2020-06-29 15:36:01 +00:00
|
|
|
pub struct View {
|
|
|
|
pub expr: Fields,
|
|
|
|
pub what: Tables,
|
|
|
|
pub cond: Option<Cond>,
|
|
|
|
pub group: Option<Groups>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for View {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "AS SELECT {} FROM {}", self.expr, self.what)?;
|
|
|
|
if let Some(ref v) = self.cond {
|
2023-02-03 11:47:07 +00:00
|
|
|
write!(f, " {v}")?
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
if let Some(ref v) = self.group {
|
2023-02-03 11:47:07 +00:00
|
|
|
write!(f, " {v}")?
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn view(i: &str) -> IResult<&str, View> {
|
|
|
|
let (i, _) = tag_no_case("AS")(i)?;
|
|
|
|
let (i, _) = shouldbespace(i)?;
|
|
|
|
let (i, _) = opt(tag_no_case("("))(i)?;
|
|
|
|
let (i, _) = tag_no_case("SELECT")(i)?;
|
|
|
|
let (i, _) = shouldbespace(i)?;
|
|
|
|
let (i, expr) = fields(i)?;
|
|
|
|
let (i, _) = shouldbespace(i)?;
|
|
|
|
let (i, _) = tag_no_case("FROM")(i)?;
|
|
|
|
let (i, _) = shouldbespace(i)?;
|
|
|
|
let (i, what) = tables(i)?;
|
|
|
|
let (i, cond) = opt(preceded(shouldbespace, cond))(i)?;
|
|
|
|
let (i, group) = opt(preceded(shouldbespace, group))(i)?;
|
|
|
|
let (i, _) = opt(tag_no_case(")"))(i)?;
|
|
|
|
Ok((
|
|
|
|
i,
|
|
|
|
View {
|
|
|
|
expr,
|
|
|
|
what,
|
|
|
|
cond,
|
|
|
|
group,
|
|
|
|
},
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn view_simple() {
|
|
|
|
let sql = "AS SELECT * FROM test";
|
|
|
|
let res = view(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("AS SELECT * FROM test", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn view_brackets() {
|
|
|
|
let sql = "AS (SELECT * FROM test)";
|
|
|
|
let res = view(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("AS SELECT * FROM test", format!("{}", out))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn view_brackets_where() {
|
2021-03-29 15:43:37 +00:00
|
|
|
let sql = "AS (SELECT temp FROM test WHERE temp IS NOT NONE)";
|
2020-06-29 15:36:01 +00:00
|
|
|
let res = view(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
2021-03-29 15:43:37 +00:00
|
|
|
assert_eq!("AS SELECT temp FROM test WHERE temp != NONE", format!("{}", out))
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn view_brackets_group() {
|
2021-03-29 15:43:37 +00:00
|
|
|
let sql = "AS (SELECT temp FROM test WHERE temp IS NOT NONE GROUP BY temp)";
|
2020-06-29 15:36:01 +00:00
|
|
|
let res = view(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
2021-03-29 15:43:37 +00:00
|
|
|
assert_eq!("AS SELECT temp FROM test WHERE temp != NONE GROUP BY temp", format!("{}", out))
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
}
|