ion-schema 0.15.0

Implementation of Amazon Ion Schema
Documentation
//! Provides a way to construct [`DocumentAuthority`].
//!
//! A [`DocumentAuthority`] is responsible for resolving a particular class of
//! schema identifiers as per *[Ion Schema spec]*.
//!
//! There are two types of a [`DocumentAuthority`] as defined below.
//! * [`FileSystemDocumentAuthority`] : Attempts to resolve schema ids to files relative to a basePath.
//! * [`MapDocumentAuthority`] : Attempts to resolve schema ids to ion elements using the map of (id, ion content).
//!
//! [Ion Schema spec]: https://amazon-ion.github.io/ion-schema/docs/isl-1-0/spec#schema-authorities
//!
//! ## Example usage of `authority` module to create a [`DocumentAuthority`]:
//! ```
//! use ion_schema::authority::{MapDocumentAuthority, FileSystemDocumentAuthority};
//! use std::path::Path;
//!
//! // map with (id, ion content) to represent `sample_number` schema
//! let map_authority = [
//!     (
//!         "sample_number.isl",
//!         r#"
//!             schema_header::{
//!                 imports: [{ id: "sample_decimal.isl", type: my_decimal, as: other_decimal }],
//!             }
//!             type::{
//!                 name: my_int,
//!                 type: int,
//!             }
//!             type::{
//!                 name: my_number,
//!                 all_of: [
//!                     my_int,
//!                     other_decimal,
//!                  ],
//!             }
//!             schema_footer::{
//!            }
//!         "#,
//!   ),
//!   (
//!         "sample_decimal.isl",
//!         r#"
//!             schema_header::{
//!                 imports: [],
//!             }
//!             type::{
//!                 name: my_decimal,
//!                 type: decimal,
//!              }
//!              schema_footer::{
//!              }
//!         "#,
//!    ),
//! ];
//!
//! let map_document_authority = MapDocumentAuthority::new(map_authority);
//!
//! let file_system_document_authority = FileSystemDocumentAuthority::new(Path::new(
//!     "sample_schemas",
//! ));
//! ```

use crate::result::{unresolvable_schema_error_raw, IonSchemaResult};
use ion_rs::Element;
use std::collections::HashMap;
use std::fmt::Debug;
use std::fs;
use std::path::{Path, PathBuf};

/// An [`DocumentAuthority`] is responsible for resolving a particular class of
/// schema identifiers.
///
/// The structure of a schema identifier string is defined by the
/// Authority responsible for the schema/type(s) being imported.
///
/// [`DocumentAuthority`] is *[Send and Sync]* i.e. it is safe to send it to another thread and to be shared between threads.
/// For cases when one does need thread-safe interior mutability, they can use the explicit locking via [`std::sync::Mutex`] and [`std::sync::RwLock`].
///
/// [Send and Sync]: https://doc.rust-lang.org/nomicon/send-and-sync.html
pub trait DocumentAuthority: Debug + Send + Sync {
    fn elements(&self, id: &str) -> IonSchemaResult<Vec<Element>>;
}

/// An [`DocumentAuthority`] implementation that attempts to resolve schema ids to files
/// relative to a basePath.
#[derive(Debug, Clone)]
pub struct FileSystemDocumentAuthority {
    base_path: PathBuf,
}

impl FileSystemDocumentAuthority {
    pub fn new(base_path: &Path) -> Self {
        Self {
            base_path: base_path.to_path_buf(),
        }
    }

    /// Returns the base path for this [`FileSystemDocumentAuthority`]
    pub fn base_path(&self) -> &Path {
        self.base_path.as_path()
    }
}

impl DocumentAuthority for FileSystemDocumentAuthority {
    /// Returns a vector of [`Element`]s based on given schema id
    fn elements(&self, id: &str) -> IonSchemaResult<Vec<Element>> {
        let absolute_path = self.base_path().join(id);
        // if absolute_path exists for the given id then load schema with file contents
        let ion_content = fs::read(absolute_path)?;
        let schema_content = Element::read_all(ion_content)?;
        Ok(schema_content.into_iter().collect())
    }
}

/// An [`DocumentAuthority`] implementation that attempts to resolve schema ids to ion elements using the map.
#[derive(Debug, Clone)]
pub struct MapDocumentAuthority {
    ion_content_by_id: HashMap<String, String>, // This map represents (id, ion content) which can used to resolve schema ids to Vec<Element>
}

impl MapDocumentAuthority {
    pub fn new<'a, I: IntoIterator<Item = (&'a str, &'a str)>>(ion_content_by_id: I) -> Self {
        Self {
            ion_content_by_id: ion_content_by_id
                .into_iter()
                .map(|(id, ion)| (id.to_owned(), ion.to_owned()))
                .collect(),
        }
    }
}

impl DocumentAuthority for MapDocumentAuthority {
    /// Returns a vector of [`Element`]s based on given schema id using ion_content_by_id map
    fn elements(&self, id: &str) -> IonSchemaResult<Vec<Element>> {
        // if ion content exists for the given id  in the map then return ion content as Elements
        let ion_content = self.ion_content_by_id.get(id).ok_or_else(|| {
            unresolvable_schema_error_raw(format!(
                "MapDocumentAuthority does not contain schema with id: {id}"
            ))
        })?;
        let schema_content = Element::read_all(ion_content.as_bytes())?;
        Ok(schema_content.into_iter().collect())
    }
}