ontoenv 0.1.4-alpha7

Rust library for managing ontologies and their imports in a local environment.
Documentation
// define a trait for a resolution policy. Given an ontology name and a set of possible ontologies,
// the policy should return the ontology that should be used.

use crate::consts::ONTOLOGY_VERSION_IRIS;
use crate::ontology::Ontology;
use oxigraph::model::NamedNode;
use serde::{Deserialize, Serialize};

pub trait ResolutionPolicy {
    fn resolve<'a>(&self, name: &str, ontologies: &'a [&'a Ontology]) -> Option<&'a Ontology>;
    fn policy_name(&self) -> &'static str;
}

pub fn policy_from_name(name: &str) -> Option<Box<dyn ResolutionPolicy>> {
    match name {
        "default" => Some(Box::new(DefaultPolicy)),
        "latest" => Some(Box::new(LatestPolicy)),
        "version" => Some(Box::new(VersionPolicy)),
        _ => None,
    }
}

/// A resolution policy that always returns the first ontology with the given name.
#[derive(Debug, Serialize, Deserialize, Default)]
pub struct DefaultPolicy;

impl ResolutionPolicy for DefaultPolicy {
    fn resolve<'a>(&self, name: &str, ontologies: &'a [&'a Ontology]) -> Option<&'a Ontology> {
        ontologies.iter().find(|o| o.name() == name).copied()
    }

    fn policy_name(&self) -> &'static str {
        "default"
    }
}

/// A resolution policy that returns the ontology which was most recently updated in the
/// environment
#[derive(Debug, Serialize, Deserialize, Default)]
pub struct LatestPolicy;

impl ResolutionPolicy for LatestPolicy {
    fn resolve<'a>(&self, name: &str, ontologies: &'a [&'a Ontology]) -> Option<&'a Ontology> {
        ontologies
            .iter()
            .filter(|o| o.name() == name)
            .max_by_key(|o| o.last_updated)
            .copied()
    }

    fn policy_name(&self) -> &'static str {
        "latest"
    }
}

/// A resolution policy that returns the ontology which has the most recent version, using
/// the various versioning information in the ontology.
#[derive(Debug, Serialize, Deserialize, Default)]
pub struct VersionPolicy;

impl ResolutionPolicy for VersionPolicy {
    fn resolve<'a>(&self, name: &str, ontologies: &'a [&'a Ontology]) -> Option<&'a Ontology> {
        // for each ontology, create a vector which contains the value for each of the ONTOLOGY_VERSION_IRIS values
        // if the ontology doesn't have a value for a given version, use "0" as the value
        let version_vectors: Vec<Vec<String>> = ontologies
            .iter()
            .filter_map(|o| {
                if o.name() != name {
                    return None;
                }
                ONTOLOGY_VERSION_IRIS
                    .iter()
                    .map(|iri| {
                        let iri: NamedNode = iri.to_owned().into();
                        Some(
                            o.version_properties()
                                .get(&iri)
                                .cloned()
                                .unwrap_or_else(|| "0".to_string()),
                        )
                    })
                    .collect()
            })
            .collect();

        // choose the "max" version by comparing the version vectors by lexicographically comparing
        // the values of the version vectors. REturn the ontology with the max version
        let max_version = version_vectors.iter().max().unwrap();
        let max_index = version_vectors
            .iter()
            .position(|v| v == max_version)
            .unwrap();
        Some(&ontologies[max_index]).copied()
    }

    fn policy_name(&self) -> &'static str {
        "version"
    }
}