Minor parser fixes (#2479)

This commit is contained in:
Mees Delzenne 2023-08-21 17:29:50 +02:00 committed by GitHub
parent 74c3551b45
commit 77c889f356
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 64 additions and 20 deletions

View file

@ -5,15 +5,16 @@ use nom::character::complete::char;
use nom::character::complete::multispace0; use nom::character::complete::multispace0;
use nom::character::complete::multispace1; use nom::character::complete::multispace1;
use nom::character::complete::not_line_ending; use nom::character::complete::not_line_ending;
use nom::multi::many0;
use nom::multi::many1; use nom::multi::many1;
pub fn mightbespace(i: &str) -> IResult<&str, ()> { pub fn mightbespace(i: &str) -> IResult<&str, ()> {
let (i, _) = alt((comment, blank))(i)?; let (i, _) = many0(alt((comment, space)))(i)?;
Ok((i, ())) Ok((i, ()))
} }
pub fn shouldbespace(i: &str) -> IResult<&str, ()> { pub fn shouldbespace(i: &str) -> IResult<&str, ()> {
let (i, _) = alt((comment, space))(i)?; let (i, _) = many1(alt((comment, space)))(i)?;
Ok((i, ())) Ok((i, ()))
} }
@ -58,12 +59,18 @@ pub fn hash(i: &str) -> IResult<&str, ()> {
Ok((i, ())) Ok((i, ()))
} }
fn blank(i: &str) -> IResult<&str, ()> {
let (i, _) = multispace0(i)?;
Ok((i, ()))
}
fn space(i: &str) -> IResult<&str, ()> { fn space(i: &str) -> IResult<&str, ()> {
let (i, _) = multispace1(i)?; let (i, _) = multispace1(i)?;
Ok((i, ())) Ok((i, ()))
} }
#[cfg(test)]
mod test {
use crate::sql::parse;
#[test]
fn any_whitespace() {
let sql = "USE /* white space and comment between */ NS test;";
assert!(parse(sql).is_ok());
}
}

View file

@ -9,6 +9,7 @@ use crate::sql::value::{value, Value};
use nom::branch::alt; use nom::branch::alt;
use nom::bytes::complete::tag_no_case; use nom::bytes::complete::tag_no_case;
use nom::combinator::map; use nom::combinator::map;
use nom::multi::separated_list1;
use nom::{multi::separated_list0, sequence::tuple}; use nom::{multi::separated_list0, sequence::tuple};
use revision::revisioned; use revision::revisioned;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -138,7 +139,7 @@ fn full(i: &str) -> IResult<&str, Permissions> {
} }
fn specific(i: &str) -> IResult<&str, Permissions> { fn specific(i: &str) -> IResult<&str, Permissions> {
let (i, perms) = separated_list0(commasorspace, permission)(i)?; let (i, perms) = separated_list1(commasorspace, permission)(i)?;
Ok(( Ok((
i, i,
Permissions { Permissions {
@ -277,4 +278,12 @@ mod tests {
} }
); );
} }
#[test]
fn no_empty_permissions() {
// This was previouslly allowed,
let sql = "PERMISSION ";
let res = dbg!(permission(sql));
assert!(dbg!(res.is_err()));
}
} }

View file

@ -4,7 +4,8 @@ use crate::sql::error::IResult;
use crate::sql::field::{fields, Fields}; use crate::sql::field::{fields, Fields};
use crate::sql::group::{group, Groups}; use crate::sql::group::{group, Groups};
use crate::sql::table::{tables, Tables}; use crate::sql::table::{tables, Tables};
use nom::bytes::complete::tag_no_case; use nom::branch::alt;
use nom::bytes::complete::{tag, tag_no_case};
use nom::combinator::opt; use nom::combinator::opt;
use nom::sequence::preceded; use nom::sequence::preceded;
use revision::revisioned; use revision::revisioned;
@ -34,19 +35,29 @@ impl fmt::Display for View {
} }
pub fn view(i: &str) -> IResult<&str, View> { pub fn view(i: &str) -> IResult<&str, View> {
let select_view = |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)?;
Ok((i, (expr, what, cond, group)))
};
let select_view_delimited = |i| {
let (i, _) = tag("(")(i)?;
let (i, res) = select_view(i)?;
let (i, _) = tag(")")(i)?;
Ok((i, res))
};
let (i, _) = tag_no_case("AS")(i)?; let (i, _) = tag_no_case("AS")(i)?;
let (i, _) = shouldbespace(i)?; let (i, _) = shouldbespace(i)?;
let (i, _) = opt(tag_no_case("("))(i)?; let (i, (expr, what, cond, group)) = alt((select_view, select_view_delimited))(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(( Ok((
i, i,
View { View {
@ -98,4 +109,21 @@ mod tests {
let out = res.unwrap().1; let out = res.unwrap().1;
assert_eq!("AS SELECT temp FROM test WHERE temp != NONE GROUP BY temp", format!("{}", out)) assert_eq!("AS SELECT temp FROM test WHERE temp != NONE GROUP BY temp", format!("{}", out))
} }
#[test]
fn view_disallow_unbalanced_brackets() {
let sql = "AS (SELECT temp FROM test WHERE temp IS NOT NONE GROUP BY temp";
let res = view(sql);
assert!(res.is_err());
let sql = "AS SELECT temp FROM test WHERE temp IS NOT NONE GROUP BY temp)";
let res = view(sql);
// The above test won't return an error since the trailing ) might be part of a another
// pair.
if let Ok((i, _)) = res {
// but it should not be parsed.
assert_eq!(i, ")")
} else {
panic!()
}
}
} }