jsonpath-rust 0.5.1

The library provides the basic functionality to find the set of the data according to the filtering query.
Documentation
use regex::{Error, Regex};
use serde_json::Value;
use std::collections::HashMap;
use std::sync::{Arc, Mutex, PoisonError};

/// The option to provide a cache for regex
/// ```
/// use serde_json::json;
/// use jsonpath_rust::JsonPathQuery;
/// use jsonpath_rust::path::config::cache::{DefaultRegexCacheInst, RegexCache};
/// use jsonpath_rust::path::config::JsonPathConfig;
///
/// let cfg = JsonPathConfig::new(RegexCache::Implemented(DefaultRegexCacheInst::default()));
///     let json = Box::new(json!({
///             "author":"abcd(Rees)",
///         }));
///
///     let _v = (json, cfg).path("$.[?(@.author ~= '.*(?i)d\\(Rees\\)')]")
///         .expect("the path is correct");
#[derive(Clone)]
pub enum RegexCache<T = DefaultRegexCacheInst>
where
    T: Clone + RegexCacheInst,
{
    Absent,
    Implemented(T),
}

impl<T> RegexCache<T>
where
    T: Clone + RegexCacheInst,
{
    pub fn is_implemented(&self) -> bool {
        match self {
            RegexCache::Absent => false,
            RegexCache::Implemented(_) => true,
        }
    }
    pub fn get_instance(&self) -> Result<&T, RegexCacheError> {
        match self {
            RegexCache::Absent => Err(RegexCacheError::new("the instance is absent".to_owned())),
            RegexCache::Implemented(inst) => Ok(inst),
        }
    }

    pub fn instance(instance: T) -> Self {
        RegexCache::Implemented(instance)
    }
}
#[allow(clippy::derivable_impls)]
impl Default for RegexCache {
    fn default() -> Self {
        RegexCache::Absent
    }
}

/// A trait that defines the behavior for regex cache
pub trait RegexCacheInst {
    fn validate(&self, regex: &str, values: Vec<&Value>) -> Result<bool, RegexCacheError>;
}

/// Default implementation for regex cache. It uses Arc and Mutex to be capable of working
/// among the threads.
#[derive(Default, Debug, Clone)]
pub struct DefaultRegexCacheInst {
    cache: Arc<Mutex<HashMap<String, Regex>>>,
}

impl RegexCacheInst for DefaultRegexCacheInst {
    fn validate(&self, regex: &str, values: Vec<&Value>) -> Result<bool, RegexCacheError> {
        let mut cache = self.cache.lock()?;
        if cache.contains_key(regex) {
            let r = cache.get(regex).unwrap();
            Ok(validate(r, values))
        } else {
            let new_reg = Regex::new(regex)?;
            let result = validate(&new_reg, values);
            cache.insert(regex.to_owned(), new_reg);
            Ok(result)
        }
    }
}

fn validate(r: &Regex, values: Vec<&Value>) -> bool {
    for el in values.iter() {
        if let Some(v) = el.as_str() {
            if r.is_match(v) {
                return true;
            }
        }
    }
    false
}

pub struct RegexCacheError {
    pub reason: String,
}

impl From<Error> for RegexCacheError {
    fn from(value: Error) -> Self {
        RegexCacheError::new(value.to_string())
    }
}

impl<T> From<PoisonError<T>> for RegexCacheError {
    fn from(value: PoisonError<T>) -> Self {
        RegexCacheError::new(value.to_string())
    }
}

impl RegexCacheError {
    pub fn new(reason: String) -> Self {
        Self { reason }
    }
}