mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 04:56:28 +00:00 
			
		
		
		
	Implements the experimental contains filter operator«
This commit is contained in:
		| @@ -26,6 +26,7 @@ pub enum Condition<'a> { | ||||
|     LowerThan(Token<'a>), | ||||
|     LowerThanOrEqual(Token<'a>), | ||||
|     Between { from: Token<'a>, to: Token<'a> }, | ||||
|     Contains(Token<'a>), | ||||
| } | ||||
|  | ||||
| /// condition      = value ("==" | ">" ...) value | ||||
| @@ -92,6 +93,23 @@ pub fn parse_not_exists(input: Span) -> IResult<FilterCondition> { | ||||
|     Ok((input, FilterCondition::Not(Box::new(FilterCondition::Condition { fid: key, op: Exists })))) | ||||
| } | ||||
|  | ||||
| /// contains        = value "CONTAINS" value | ||||
| pub fn parse_contains(input: Span) -> IResult<FilterCondition> { | ||||
|     let (input, (fid, _, value)) = tuple((parse_value, tag("CONTAINS"), cut(parse_value)))(input)?; | ||||
|     Ok((input, FilterCondition::Condition { fid, op: Contains(value) })) | ||||
| } | ||||
|  | ||||
| /// contains        = value "NOT" WS+ "CONTAINS" value | ||||
| pub fn parse_not_contains(input: Span) -> IResult<FilterCondition> { | ||||
|     let keyword = tuple((tag("NOT"), multispace1, tag("CONTAINS"))); | ||||
|     let (input, (fid, _, value)) = tuple((parse_value, keyword, cut(parse_value)))(input)?; | ||||
|  | ||||
|     Ok(( | ||||
|         input, | ||||
|         FilterCondition::Not(Box::new(FilterCondition::Condition { fid, op: Contains(value) })), | ||||
|     )) | ||||
| } | ||||
|  | ||||
| /// to             = value value "TO" WS+ value | ||||
| pub fn parse_to(input: Span) -> IResult<FilterCondition> { | ||||
|     let (input, (key, from, _, _, to)) = | ||||
|   | ||||
| @@ -146,7 +146,7 @@ impl<'a> Display for Error<'a> { | ||||
|             } | ||||
|             ErrorKind::InvalidPrimary => { | ||||
|                 let text = if input.trim().is_empty() { "but instead got nothing.".to_string() } else { format!("at `{}`.", escaped_input) }; | ||||
|                 writeln!(f, "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` {}", text)? | ||||
|                 writeln!(f, "Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` {}", text)? | ||||
|             } | ||||
|             ErrorKind::InvalidEscapedNumber => { | ||||
|                 writeln!(f, "Found an invalid escaped sequence number: `{}`.", escaped_input)? | ||||
|   | ||||
| @@ -48,8 +48,8 @@ use std::fmt::Debug; | ||||
|  | ||||
| pub use condition::{parse_condition, parse_to, Condition}; | ||||
| use condition::{ | ||||
|     parse_exists, parse_is_empty, parse_is_not_empty, parse_is_not_null, parse_is_null, | ||||
|     parse_not_exists, | ||||
|     parse_contains, parse_exists, parse_is_empty, parse_is_not_empty, parse_is_not_null, | ||||
|     parse_is_null, parse_not_contains, parse_not_exists, | ||||
| }; | ||||
| use error::{cut_with_err, ExpectedValueKind, NomErrorExt}; | ||||
| pub use error::{Error, ErrorKind}; | ||||
| @@ -147,7 +147,37 @@ pub enum FilterCondition<'a> { | ||||
|     GeoBoundingBox { top_right_point: [Token<'a>; 2], bottom_left_point: [Token<'a>; 2] }, | ||||
| } | ||||
|  | ||||
| pub enum TraversedElement<'a> { | ||||
|     FilterCondition(&'a FilterCondition<'a>), | ||||
|     Condition(&'a Condition<'a>), | ||||
| } | ||||
|  | ||||
| impl<'a> FilterCondition<'a> { | ||||
|     pub fn use_contains_operator(&self) -> Option<&Token> { | ||||
|         match self { | ||||
|             FilterCondition::Condition { fid: _, op } => match op { | ||||
|                 Condition::GreaterThan(_) | ||||
|                 | Condition::GreaterThanOrEqual(_) | ||||
|                 | Condition::Equal(_) | ||||
|                 | Condition::NotEqual(_) | ||||
|                 | Condition::Null | ||||
|                 | Condition::Empty | ||||
|                 | Condition::Exists | ||||
|                 | Condition::LowerThan(_) | ||||
|                 | Condition::LowerThanOrEqual(_) | ||||
|                 | Condition::Between { .. } => None, | ||||
|                 Condition::Contains(tok) => Some(tok), | ||||
|             }, | ||||
|             FilterCondition::Not(this) => this.use_contains_operator(), | ||||
|             FilterCondition::Or(seq) | FilterCondition::And(seq) => { | ||||
|                 seq.iter().find_map(|filter| filter.use_contains_operator()) | ||||
|             } | ||||
|             FilterCondition::GeoLowerThan { .. } | ||||
|             | FilterCondition::GeoBoundingBox { .. } | ||||
|             | FilterCondition::In { .. } => None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Returns the first token found at the specified depth, `None` if no token at this depth. | ||||
|     pub fn token_at_depth(&self, depth: usize) -> Option<&Token> { | ||||
|         match self { | ||||
| @@ -452,6 +482,8 @@ fn parse_primary(input: Span, depth: usize) -> IResult<FilterCondition> { | ||||
|         parse_exists, | ||||
|         parse_not_exists, | ||||
|         parse_to, | ||||
|         parse_contains, | ||||
|         parse_not_contains, | ||||
|         // the next lines are only for error handling and are written at the end to have the less possible performance impact | ||||
|         parse_geo, | ||||
|         parse_geo_distance, | ||||
| @@ -534,6 +566,7 @@ impl<'a> std::fmt::Display for Condition<'a> { | ||||
|             Condition::LowerThan(token) => write!(f, "< {token}"), | ||||
|             Condition::LowerThanOrEqual(token) => write!(f, "<= {token}"), | ||||
|             Condition::Between { from, to } => write!(f, "{from} TO {to}"), | ||||
|             Condition::Contains(token) => write!(f, "CONTAINS {token}"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -558,6 +591,7 @@ pub mod tests { | ||||
|         unsafe { Span::new_from_raw_offset(offset, lines as u32, value, "") }.into() | ||||
|     } | ||||
|  | ||||
|     #[track_caller] | ||||
|     fn p(s: &str) -> impl std::fmt::Display + '_ { | ||||
|         Fc::parse(s).unwrap().unwrap() | ||||
|     } | ||||
| @@ -639,6 +673,13 @@ pub mod tests { | ||||
|         insta::assert_snapshot!(p("NOT subscribers NOT EXISTS"), @"{subscribers} EXISTS"); | ||||
|         insta::assert_snapshot!(p("subscribers NOT   EXISTS"), @"NOT ({subscribers} EXISTS)"); | ||||
|  | ||||
|         // Test CONTAINS + NOT CONTAINS | ||||
|         insta::assert_snapshot!(p("subscribers CONTAINS 'hello'"), @"{subscribers} CONTAINS {hello}"); | ||||
|         insta::assert_snapshot!(p("NOT subscribers CONTAINS 'hello'"), @"NOT ({subscribers} CONTAINS {hello})"); | ||||
|         insta::assert_snapshot!(p("subscribers NOT CONTAINS hello"), @"NOT ({subscribers} CONTAINS {hello})"); | ||||
|         insta::assert_snapshot!(p("NOT subscribers NOT CONTAINS 'hello'"), @"{subscribers} CONTAINS {hello}"); | ||||
|         insta::assert_snapshot!(p("subscribers NOT   CONTAINS 'hello'"), @"NOT ({subscribers} CONTAINS {hello})"); | ||||
|  | ||||
|         // Test nested NOT | ||||
|         insta::assert_snapshot!(p("NOT NOT NOT NOT x = 5"), @"{x} = {5}"); | ||||
|         insta::assert_snapshot!(p("NOT NOT (NOT NOT x = 5)"), @"{x} = {5}"); | ||||
| @@ -710,7 +751,7 @@ pub mod tests { | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p("'OR'"), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `\'OR\'`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `\'OR\'`. | ||||
|         1:5 'OR' | ||||
|         "###); | ||||
|  | ||||
| @@ -720,12 +761,12 @@ pub mod tests { | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p("channel Ponce"), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `channel Ponce`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `channel Ponce`. | ||||
|         1:14 channel Ponce | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p("channel = Ponce OR"), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` but instead got nothing. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` but instead got nothing. | ||||
|         19:19 channel = Ponce OR | ||||
|         "###); | ||||
|  | ||||
| @@ -810,12 +851,12 @@ pub mod tests { | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p("colour NOT EXIST"), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `colour NOT EXIST`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `colour NOT EXIST`. | ||||
|         1:17 colour NOT EXIST | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p("subscribers 100 TO1000"), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `subscribers 100 TO1000`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `subscribers 100 TO1000`. | ||||
|         1:23 subscribers 100 TO1000 | ||||
|         "###); | ||||
|  | ||||
| @@ -878,35 +919,35 @@ pub mod tests { | ||||
|         "###); | ||||
|  | ||||
|         insta::assert_snapshot!(p(r#"value NULL"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value NULL`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value NULL`. | ||||
|         1:11 value NULL | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value NOT NULL"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value NOT NULL`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value NOT NULL`. | ||||
|         1:15 value NOT NULL | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value EMPTY"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value EMPTY`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value EMPTY`. | ||||
|         1:12 value EMPTY | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value NOT EMPTY"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value NOT EMPTY`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value NOT EMPTY`. | ||||
|         1:16 value NOT EMPTY | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value IS"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value IS`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value IS`. | ||||
|         1:9 value IS | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value IS NOT"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value IS NOT`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value IS NOT`. | ||||
|         1:13 value IS NOT | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value IS EXISTS"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value IS EXISTS`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value IS EXISTS`. | ||||
|         1:16 value IS EXISTS | ||||
|         "###); | ||||
|         insta::assert_snapshot!(p(r#"value IS NOT EXISTS"#), @r###" | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `_geoRadius`, or `_geoBoundingBox` at `value IS NOT EXISTS`. | ||||
|         Was expecting an operation `=`, `!=`, `>=`, `>`, `<=`, `<`, `IN`, `NOT IN`, `TO`, `EXISTS`, `NOT EXISTS`, `IS NULL`, `IS NOT NULL`, `IS EMPTY`, `IS NOT EMPTY`, `CONTAINS`, `NOT CONTAINS`, `_geoRadius`, or `_geoBoundingBox` at `value IS NOT EXISTS`. | ||||
|         1:20 value IS NOT EXISTS | ||||
|         "###); | ||||
|     } | ||||
|   | ||||
| @@ -211,6 +211,7 @@ fn is_keyword(s: &str) -> bool { | ||||
|             | "IS" | ||||
|             | "NULL" | ||||
|             | "EMPTY" | ||||
|             | "CONTAINS" | ||||
|             | "_geoRadius" | ||||
|             | "_geoBoundingBox" | ||||
|     ) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user