1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
//! Stores arbitrarliy interrelated elements.
//!
//! This is not used in Xvc at the moment.
//! It uses `XvcStore<ChildEntity<T, U>>` and `XvcStore<ChildEntity<U, T>>` to specify two
//! [crate::R1NStore] 's that keep each other's relationship. It can be deprecated if there seems
//! not to be a use case.
//!
//! A possible use case might be in experiments. Experiments vs files may have M-N relationships,
//! each file can be affected by multiple experiments and each experiment may be affecting multiple
//! files. I'll wait to deprecate this until all major features are implemented.
use std::fmt::Debug;
use std::path::Path;
use crate::error::Result;
use crate::ChildEntity;
use crate::{Storable, XvcStore};
/// RMNStore is M-N RelationStore, where we store arbitrary relationships between two entities. It
/// doesn't have any semantics except binding two entities together.
#[derive(Debug, Clone)]
pub struct RMNStore<T, U>
where
    T: Storable,
    U: Storable,
{
    /// The store for M side of the relationships
    pub left: XvcStore<T>,
    /// The store for N side of the relationships
    pub right: XvcStore<U>,
    /// Parent-child relationships from left to right
    pub left_to_right: XvcStore<ChildEntity<T, U>>,
    /// Parent-child relationships from right to left
    pub right_to_left: XvcStore<ChildEntity<U, T>>,
}
impl<T, U> RMNStore<T, U>
where
    T: Storable,
    U: Storable,
{
    /// Loads store from the directory under store root named after types
    pub fn load_rmnstore(store_root: &Path) -> Result<RMNStore<T, U>> {
        let left = XvcStore::<T>::load_store(store_root)?;
        let right = XvcStore::<U>::load_store(store_root)?;
        let left_to_right = XvcStore::<ChildEntity<T, U>>::load_store(store_root)?;
        let right_to_left = XvcStore::<ChildEntity<U, T>>::load_store(store_root)?;
        Ok(RMNStore {
            left,
            right,
            left_to_right,
            right_to_left,
        })
    }
    /// Saves the store to directory created by store root and type names
    pub fn save_rmnstore(store: &Self, store_root: &Path) -> Result<()> {
        store.left.save(store_root)?;
        store.right.save(store_root)?;
        store.left_to_right.save(store_root)?;
        store.right_to_left.save(store_root)
    }
}