rbe 0.2.20

RDF data shapes implementation in Rust
Documentation
use crate::Context;
use crate::{Key, Ref, Value, rbe_cond::RbeCond};
use crate::{Pending, rbe_error::RbeError};
use std::collections::HashSet;
use tracing::debug;

#[derive(Default)]
pub struct RbeCondMatcher<K, V, R, Ctx>
where
    K: Key,
    V: Value,
    R: Ref,
    Ctx: Context,
{
    rbe: RbeCond<K, V, R, Ctx>,
    open: bool,
    controlled: HashSet<K>,
}

impl<K, V, R, Ctx> RbeCondMatcher<K, V, R, Ctx>
where
    K: Key,
    V: Value,
    R: Ref,
    Ctx: Context,
{
    pub fn new() -> RbeCondMatcher<K, V, R, Ctx> {
        RbeCondMatcher::default()
    }

    pub fn with_rbe(mut self, rbe: &RbeCond<K, V, R, Ctx>) -> Self {
        self.rbe = (*rbe).clone();
        self.controlled = self.rbe.symbols();
        self
    }

    pub fn extend_controlled(mut self, controlled: HashSet<K>) -> Self {
        self.controlled.extend(controlled);
        self
    }

    pub fn with_open(mut self, open: bool) -> Self {
        self.open = open;
        self
    }

    pub fn matches<T: IntoIterator<Item = (K, V, Ctx)>>(
        &self,
        iter: T,
    ) -> Result<Pending<V, R>, RbeError<K, V, R, Ctx>> {
        let mut pending = Pending::new();
        let mut processed: Vec<(K, V, Ctx)> = Vec::new();
        match self.matches_iter(iter, &mut pending, &mut processed) {
            RbeCond::Fail { error } => Err(error.clone()),
            d => {
                if d.nullable() {
                    Ok(pending)
                } else {
                    Err(RbeError::NonNullableMatch {
                        non_nullable_rbe: Box::new(d.clone()),
                        expr: Box::new(self.rbe.clone()),
                    })
                }
            },
        }
    }

    fn matches_iter<T: IntoIterator<Item = (K, V, Ctx)>>(
        &self,
        iter: T,
        pending: &mut Pending<V, R>,
        processed: &mut Vec<(K, V, Ctx)>,
    ) -> RbeCond<K, V, R, Ctx> {
        let mut current = self.rbe.clone();
        for (key, value, ctx) in iter {
            let deriv = current.deriv(&key, &value, &ctx, 1, self.open, &self.controlled, pending);
            match deriv {
                RbeCond::Fail { error } => {
                    current = RbeCond::Fail {
                        error: RbeError::DerivIterError {
                            error_msg: format!("{error}"),
                            processed: processed.clone(),
                            expr: Box::new(self.rbe.clone()),
                            current: Box::new(current.clone()),
                            key: key.clone(),
                            open: self.open,
                        },
                    };
                    break;
                },
                _ => {
                    debug!("Processing: {key}/{value}\ncurrent:{current}\nderiv:{deriv}");
                    processed.push((key.clone(), value.clone(), ctx.clone()));
                    current = deriv;
                },
            }
        }
        current
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use crate::{MatchCond, Max, Min, SingleCond};

    impl Value for i32 {}
    impl Ref for String {}

    fn is_even(v: &i32, _ctx: &char) -> Result<Pending<i32, String>, RbeError<char, i32, String, char>> {
        if v % 2 == 0 {
            Ok(Pending::new())
        } else {
            Err(RbeError::MsgError {
                msg: format!("Value {v} is not even"),
            })
        }
    }

    fn ref_x(v: &i32, _ctx: &char) -> Result<Pending<i32, String>, RbeError<char, i32, String, char>> {
        let ps = vec![(*v, vec!["X".to_string()])].into_iter();
        Ok(Pending::from(ps))
    }

    impl Value for String {}

    fn cond_name(name: String) -> MatchCond<char, String, String, char> {
        MatchCond::single(SingleCond::new().with_cond(move |v: &String, _ctx: &char| {
            if *v == name {
                Ok(Pending::new())
            } else {
                Err(RbeError::MsgError {
                    msg: format!("Value {v} is not equal to {name}"),
                })
            }
        }))
    }

    fn cond_len(len: usize) -> MatchCond<char, String, String, char> {
        MatchCond::single(SingleCond::new().with_cond(move |v: &String, _ctx: &char| {
            if v.len() == len {
                Ok(Pending::new())
            } else {
                Err(RbeError::MsgError {
                    msg: format!("Value {v} has no length {len}"),
                })
            }
        }))
    }

    #[test]
    fn test_rbe_matcher_len_name() {
        let rbe: RbeCond<char, String, String, char> = RbeCond::and(vec![
            RbeCond::symbol_cond('a', cond_len(3), Min::from(1), Max::IntMax(1)),
            RbeCond::symbol_cond('b', cond_name("foo".to_string()), Min::from(0), Max::IntMax(1)),
        ]);
        let expected = Pending::new();
        let rbe_matcher = RbeCondMatcher::new().with_rbe(&rbe);

        assert_eq!(
            rbe_matcher.matches(vec![('a', "baz".to_string(), 'a')].into_iter()),
            Ok(expected)
        );
    }

    #[test]
    fn test_rbe_matcher_even_ref() {
        let cond_even = MatchCond::single(SingleCond::new().with_cond(is_even));
        let cond_ref_x = MatchCond::single(SingleCond::new().with_cond(ref_x));

        let rbe: RbeCond<char, i32, String, char> = RbeCond::and(vec![
            RbeCond::symbol_cond('a', cond_even, Min::from(1), Max::IntMax(1)),
            RbeCond::symbol_cond('b', cond_ref_x, Min::from(0), Max::IntMax(1)),
        ]);
        let expected = Pending::from(vec![(42, vec!["X".to_string()])]);
        let rbe_matcher = RbeCondMatcher::new().with_rbe(&rbe);

        assert_eq!(
            rbe_matcher.matches(vec![('a', 2, 'a'), ('b', 42, 'b')].into_iter()),
            Ok(expected)
        );
    }

    #[test]
    fn test_rbe_matcher_even_ref_unordered() {
        let cond_even = MatchCond::single(SingleCond::new().with_cond(is_even));
        let cond_ref_x = MatchCond::single(SingleCond::new().with_cond(ref_x));

        let rbe: RbeCond<char, i32, String, char> = RbeCond::and(vec![
            RbeCond::symbol_cond('a', cond_even, Min::from(1), Max::IntMax(1)),
            RbeCond::symbol_cond('b', cond_ref_x, Min::from(0), Max::IntMax(1)),
        ]);
        let expected = Pending::from(vec![(42, vec!["X".to_string()])]);
        let rbe_matcher = RbeCondMatcher::new().with_rbe(&rbe);

        assert_eq!(
            rbe_matcher.matches(vec![('b', 42, 'b'), ('a', 2, 'a')].into_iter()),
            Ok(expected)
        );
    }

    #[test]
    fn test_rbe_matcher_even_fails_odd() {
        let cond_even = MatchCond::single(SingleCond::new().with_cond(is_even));
        let cond_ref_x = MatchCond::single(SingleCond::new().with_cond(ref_x));

        let rbe: RbeCond<char, i32, String, char> = RbeCond::and(vec![
            RbeCond::symbol_cond('a', cond_even, Min::from(1), Max::IntMax(1)),
            RbeCond::symbol_cond('b', cond_ref_x, Min::from(0), Max::IntMax(1)),
        ]);
        let rbe_matcher = RbeCondMatcher::new().with_rbe(&rbe);
        let iter = vec![('b', 42, 'b'), ('a', 3, 'a')].into_iter();
        assert!(rbe_matcher.matches(iter).is_err());
    }
}