shex_ast 0.2.18

RDF data shapes implementation in Rust
Documentation
use crate::shapemap::{Association, NodeSelector, ShapeSelector, ShapemapError};
use crate::{Node, ShapeExprLabel, ir::shape_label::ShapeLabel, object_value::ObjectValue};
use prefixmap::{DerefError, PrefixMap};
use rudof_iri::IriS;
use rudof_rdf::rdf_core::query::QueryRDF;
use serde::Serialize;
use std::fmt::Display;

#[derive(Debug, Default, PartialEq, Clone, Serialize)]
pub struct QueryShapeMap {
    associations: Vec<Association>,
    nodes_prefixmap: PrefixMap,
    shapes_prefixmap: PrefixMap,
}

impl QueryShapeMap {
    pub fn new() -> Self {
        Self::default()
    }

    pub fn nodes_prefixmap(&self) -> PrefixMap {
        self.nodes_prefixmap.clone()
    }

    pub fn shapes_prefixmap(&self) -> PrefixMap {
        self.shapes_prefixmap.clone()
    }

    pub fn with_nodes_prefixmap(mut self, prefixmap: &PrefixMap) -> Self {
        self.nodes_prefixmap = prefixmap.clone();
        self
    }

    pub fn with_shapes_prefixmap(mut self, prefixmap: &PrefixMap) -> Self {
        self.shapes_prefixmap = prefixmap.clone();
        self
    }

    pub fn add_association(
        &mut self,
        node_selector: NodeSelector,
        base_nodes: &Option<IriS>,
        shape_selector: ShapeSelector,
        base_shapes: &Option<IriS>,
    ) -> Result<(), DerefError> {
        let association = Association::new(
            node_selector,
            base_nodes,
            Some(&self.nodes_prefixmap),
            shape_selector,
            base_shapes,
            Some(&self.shapes_prefixmap),
        )?;
        self.associations.push(association);
        Ok(())
    }

    pub fn iter(&self) -> impl Iterator<Item = &Association> + '_ {
        self.associations.iter()
    }

    pub fn node_shapes<'a, R>(&'a self, rdf: &'a R) -> Result<Vec<(R::Term, &'a ShapeExprLabel)>, ShapemapError>
    where
        R: QueryRDF,
    {
        let mut result = Vec::new();
        for assoc in self.iter() {
            let node_shapes = assoc.iter_node_shape(rdf)?;
            for pair in node_shapes {
                result.push(pair)
            }
        }
        Ok(result)
    }

    pub fn from_node_shape(node: &Node, shape: &ShapeLabel) -> Result<Self, crate::SchemaJsonError> {
        let mut sm = QueryShapeMap::new();
        let object_value: ObjectValue = node.try_into()?;
        let shape: ShapeExprLabel = shape.into();
        sm.add_association(
            NodeSelector::Node(object_value),
            &None,
            ShapeSelector::label(shape),
            &None,
        )
        .map_err(|e| crate::SchemaJsonError::DerefError { error: e.to_string() })?;
        Ok(sm)
    }
}

impl Display for QueryShapeMap {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        let mut iter = self.associations.iter();
        if let Some(first) = iter.next() {
            write!(f, "{first}")?;
            for assoc in iter {
                write!(f, ",\n{assoc}")?;
            }
        }
        Ok(())
    }
}