use chrono::{DateTime, TimeZone, Utc};
use serde::Serialize;
use test_case::test_case;
use crate::scim_filter;
#[derive(Debug, Serialize, PartialEq)]
struct Resource {
a: String,
b: String,
c: String,
sub_resource: SubResource,
datetime: DateTime<Utc>,
decimal: rust_decimal::Decimal,
number: u32,
bool: bool,
bool_false: bool,
multi_simple_value: Vec<String>,
multi_simple_value_numbers: Vec<u32>,
nested_multi_value: Vec<SubResource>,
multi_bool_value: Vec<bool>,
}
#[derive(Debug, Serialize, PartialEq)]
struct SubResource {
first: String,
second: String,
}
impl Resource {
pub fn new(a: &str, b: &str, c: &str) -> Self {
Self {
a: a.to_string(),
b: b.to_string(),
c: c.to_string(),
sub_resource: SubResource {
first: "test-first".to_string(),
second: "test-second".to_string(),
},
datetime: Utc.with_ymd_and_hms(2021, 1, 1, 10, 0, 0).unwrap(),
decimal: rust_decimal::Decimal::new(102, 1),
number: 42,
bool: true,
bool_false: false,
multi_simple_value: vec![a.to_string(), b.to_string(), c.to_string()],
multi_simple_value_numbers: vec![1, 2, 3],
nested_multi_value: vec![
SubResource {
first: "test-first1".to_string(),
second: "test-second1".to_string(),
},
SubResource {
first: "test-first2".to_string(),
second: "test-second2".to_string(),
},
],
multi_bool_value: vec![true, true, true],
}
}
}
fn example_resources() -> Vec<Resource> {
vec![Resource::new("test1", "test2", "test3")]
}
fn example_resources2() -> Vec<Resource> {
vec![
Resource::new("a1", "b1", "c1"),
Resource::new("a2", "b2", "c2"),
Resource::new("a3", "b3", "c3"),
]
}
#[test_case("bool eq true"; "one resource do match with equals boolean")]
#[test_case("bool ne false"; "one resource do match with not equals boolean")]
#[test_case("a eq \"test1\""; "one resource do match with equals")]
#[test_case("b co \"est\""; "one resource do match with correct contains")]
#[test_case("b sw \"te\""; "one resource do match with correct starts with")]
#[test_case("c ew \"st3\""; "one resource do match with correct ends with")]
#[test_case("c pr"; "one resource do match with present")]
#[test_case("a eq \"test1\" or b eq \"test2\""; "two resources with a logical or")]
#[test_case("a eq \"test1\" and b eq \"test2\""; "two resources with a logical and")]
#[test_case("a eq \"test1\" or b eq \"test3\""; "two resources with a logical or where one is wrong")]
#[test_case("A eq \"test1\""; "matches should be case insensitive")]
#[test_case("(a eq \"test1\" or b eq \"test3\") and c pr"; "complex filter 1")]
#[test_case("datetime gt \"2020-01-01T10:10:10Z\""; "filter with date that should match")]
#[test_case("decimal gt 9.1"; "filter with decimal")]
#[test_case("a eq \"test1\" and sub_resource[first co \"test-\" and second co \"test-\"]"; "filter with complex attribute match")]
#[test_case("a eq \"test1\" and sub_resource[first sw \"test-\"]"; "filter with complex attribute and one single expression")]
#[test_case("a gt \"tess\""; "GreaterThan on strings")]
#[test_case("a ge \"tess\" and not (datetime lt \"2020-01-01T10:10:10Z\")"; "not expression")]
#[test_case("multi_simple_value eq \"test1\""; "simple multi-valued attribute")]
#[test_case("multi_simple_value ne \"testZ\""; "simple multi-valued attribute with ne")]
#[test_case("multi_simple_value ne \"test1\""; "simple multi-valued attribute with ne works")]
#[test_case("multi_simple_value co \"test1\""; "simple multi-valued attribute with contains")]
#[test_case("multi_simple_value sw \"tes\""; "simple multi-valued attribute with startsWith")]
#[test_case("multi_simple_value ew \"st1\""; "simple multi-valued attribute with endsWith")]
#[test_case("multi_simple_value_numbers gt 2"; "simple multi-valued attribute with greaterThan")]
#[test_case("multi_simple_value gt \"tess\""; "simple multi-valued attribute with greaterThan on strings")]
#[test_case("multi_simple_value ge \"test\""; "simple multi-valued attribute with greaterEqual on strings")]
#[test_case("multi_simple_value_numbers ge 3"; "simple multi-valued attribute with greaterEqual")]
#[test_case("multi_simple_value_numbers lt 2"; "simple multi-valued attribute with lessThan")]
#[test_case("multi_simple_value_numbers le 1"; "simple multi-valued attribute with lessThanEqual")]
#[test_case("nested_multi_value.first eq \"test-first1\""; "nested multi-valued attribute with eq")]
#[test_case("nested_multi_value.first ne \"test-firstZ\""; "nested multi-valued attribute with ne")]
#[test_case("number eq 42"; "number match on single value")]
#[test_case("multi_bool_value eq true"; "bool multi-valued attribute with Equal")]
#[test_case("multi_bool_value ne false"; "bool multi-valued attribute with NotEqual")]
#[test_case("bool eq true"; "bool(true) single-valued attribute with Equal")]
#[test_case("bool ne false"; "bool(true) single-valued attribute with NotEqual")]
#[test_case("bool_false eq false"; "bool(false) single-valued attribute with Equal")]
#[test_case("bool_false ne true"; "bool(false) single-valued attribute with NotEqual")]
#[test_case("nested_multi_value[first eq \"test-first1\"]"; "ValueFilter with search match")]
#[test_case("nested_multi_value[first eq \"test-first1\" and not(second eq \"test-second3\")]"; "ValueFilter with search match and logical expression")]
#[test_case("nested_multi_value[not(first eq \"nonono\")]"; "ValueFilter with search match and sub-filter")]
#[test_case("number ne 10"; "not equal with number")]
#[test_case("not (number co 10)"; "contains with number")]
#[test_case("not (number sw 10)"; "starts with with number")]
#[test_case("not (number ew 10)"; "ends with with number")]
#[test_case("A eq \"test1\""; "the match should be case insensitive - simple values")]
#[test_case("multi_simple_VALUE co \"t1\""; "the match should be case insensitive - multi simple values")]
#[test_case("nested_multi_value.FIRST ew \"first1\""; "the match should be case insensitive - multi complex values")]
fn match_ok_with_one_resource(filter: &str) {
let resources = example_resources();
let res = scim_filter(filter, resources);
assert_eq!(example_resources(), res.unwrap());
}
#[test_case("a eq \"no-match\""; "one resource do not match with wrong equals")]
#[test_case("a.test eq \"no-match\""; "one resource do not match with non-present sub attributes")]
#[test_case("b co \"zest\""; "one resource do not match with wrong contains")]
#[test_case("b sw \"ze\""; "one resource do not match with wrong starts with")]
#[test_case("c ew \"stX\""; "one resource do not match with wrong ends with")]
#[test_case("d pr"; "one resource do not match with present")]
#[test_case("a eq \"test1\" and b eq \"test2\" and (c eq \"wrong1\" or c eq \"wrong2\")"; "complex filter 2")]
#[test_case("datetime gt \"2022-01-01T10:10:10Z\""; "filter with date")]
#[test_case("a eq \"test1\" and sub_resource[first co \"test-\" and second ew \"test-\"]"; "filter with complex attribute should not match")]
#[test_case("multi_simple_value eq \"ZZZ\""; "simple multi-valued attribute")]
#[test_case("multi_simple_value co \"ZZZ\""; "simple multi-valued attribute with contains")]
#[test_case("multi_simple_value sw \"ZZZ\""; "simple multi-valued attribute with startsWith")]
#[test_case("multi_simple_value ew \"stZ\""; "simple multi-valued attribute with endsWith")]
#[test_case("multi_simple_value_numbers gt 3"; "simple multi-valued attribute with greaterThan")]
#[test_case("multi_simple_value gt \"testZ\""; "simple multi-valued attribute with greaterThan on strings")]
#[test_case("multi_simple_value ge \"test4\""; "simple multi-valued attribute with greaterEqual on strings")]
#[test_case("multi_simple_value_numbers ge 4"; "simple multi-valued attribute with greaterEqual")]
#[test_case("multi_simple_value_numbers lt 1"; "simple multi-valued attribute with lessThan")]
#[test_case("multi_simple_value_numbers le 0"; "simple multi-valued attribute with lessThanEqual")]
#[test_case("nested_multi_value.first eq \"ZZZZZ\""; "complex multi-valued attribute with eq")]
#[test_case("nested_multi_value.nonExistent eq \"ZZZZZ\""; "nested non-existent multi-valued attribute")]
#[test_case("bool eq \"test\""; "equals boolean with string")]
#[test_case("bool eq \"2022-01-01T10:10:10Z\""; "equals boolean with datetime")]
#[test_case("bool eq 19.2"; "equals boolean with decimal")]
#[test_case("bool eq 11"; "equals boolean with integer")]
#[test_case("datetime eq \"test\""; "equals datetime with string")]
#[test_case("datetime eq true"; "equals datetime with boolean")]
#[test_case("datetime eq 19.2"; "equals datetime with decimal")]
#[test_case("datetime eq 11"; "equals datetime with integer")]
#[test_case("decimal eq \"test\""; "equals decimal with string")]
#[test_case("decimal eq true"; "equals decimal with boolean")]
#[test_case("decimal eq \"2022-01-01T10:10:10Z\""; "equals decimal with datetime")]
#[test_case("decimal ew \"test\""; "equals decimal do not work with EndsWith")]
#[test_case("a eq 19.2"; "equals string with decimal")]
#[test_case("a eq 11"; "equals string with integer")]
#[test_case("a eq true"; "equals string with boolean")]
#[test_case("a eq \"2022-01-01T10:10:10Z\""; "equals string with datetime")]
fn match_none_with_one_resource(filter: &str) {
let resources = example_resources();
let res = scim_filter(filter, resources);
assert_eq!(Vec::<Resource>::new(), res.unwrap());
}
#[test_case("bool gt true"; "greater_than on bool")]
#[test_case("bool ge true"; "greater_than_equal on bool")]
#[test_case("bool lt true"; "less_than on bool")]
#[test_case("bool le true"; "less_than_equal on bool")]
#[test_case("a gt true"; "greater_than string with boolean")]
fn invalid_filter(filter: &str) {
let resources = example_resources();
let res = scim_filter(filter, resources);
assert!(res.is_err());
}
#[test_case("a eq \"a1\" or b eq \"b2\" or c eq \"c3\""; "or should match both")]
#[test_case("a sw \"a\""; "all starting with a should match")]
fn match_all_with_two_resources(filter: &str) {
let resources = example_resources2();
let res = scim_filter(filter, resources);
assert!(res.is_ok());
assert_eq!(example_resources2(), res.unwrap());
}
#[test_case("a eq \"a1\" and b eq \"b2\""; "and should not match")]
#[test_case("a eq \"aaa1\""; "none should not match")]
fn match_none_with_two_resources(filter: &str) {
let resources = example_resources2();
let res = scim_filter(filter, resources);
assert!(res.is_ok());
assert_eq!(Vec::<Resource>::new(), res.unwrap());
}
#[test_case("a eq \"a1\" and (b eq \"b1\" or b eq \"b2\")", vec![1]; "nested or should match only first resource")]
#[test_case("(a eq \"a1\" and b eq \"b1\") or a eq \"b2\"", vec![1]; "nested and should match only first resource")]
fn match_partial_with_two_resources(filter: &str, indexes: Vec<usize>) {
let resources = example_resources2();
let res = scim_filter(filter, resources);
assert!(res.is_ok());
let all = example_resources2();
let mut expected = vec![];
for i in indexes {
expected.push(all.get(i - 1).unwrap());
}
assert_eq!(expected, res.unwrap().iter().collect::<Vec<&Resource>>());
}