devela 0.27.0

A development layer of coherence.
Documentation
// devela::data::collections::graph
//
//! Graphs are collections of vertices and edges, backed by two arrays or two `Vec`s.
//!
//! They enable efficient representation of complex relationships and structures between
//! pairs of nodes, with edges being potentially ordered and carrying optional data.
//

mod edge;
use edge::impl_graph_edge;

use crate::{
    Array, Bare,
    DataError::{IndexOutOfBounds, KeyAlreadyExists, NodeEmpty, NotEnoughSpace},
    DataResult as Result, Storage,
    num::niche::*,
    paste, unwrap,
};

macro_rules! impl_graph {
    () => {
        impl_graph!(u8:"_graph_u8", u16:"_graph_u16", u32:"_graph_u32", usize:"_graph_usize");
    };

    // $IDX:  the index primitive type. E.g. u8.
    // $cap:  the capability feature that enables the given implementation. E.g "_graph_u8".
    //
    // $Graph: the graph type name. E.g. GraphU8.
    // $Node: the node type name. E.g. NodeU8.
    // $Edge:  the edge type name. E.g. EdgeU8.
    // $Index: the index real type name. E.g. NonExtremeU8.
    ($( $IDX:ty : $cap:literal ),+) => { paste! {
        $(
            #[cfg(feature = $cap )]
            impl_graph!(@call
                [<Graph $IDX:camel>],      // $Graph
                [<Node $IDX:camel>],       // $Node
                [<Edge $IDX:camel>],       // $Edge
                [<NonExtreme $IDX:camel>], // $Index
                $IDX);
        )+
    }};
    (@call $($Graph:ty, $Node:ty, $Edge:ty, $Index:ty, $IDX:ty)+) => {
        $(
            impl_graph_edge!(@$Graph, $Node, $Edge, $Index, $IDX);
            impl_graph!(@$Graph, $Node, $Edge, $Index, $IDX);
        )+
    };

    (@$Graph:ty, $Node:ty, $Edge:ty, $Index:ty, $IDX:ty) => { paste! {
        use super::$Node;

        /* definition */

        #[doc = "A static graph with [`" $Index "`] indices,"]
        /// and two [`Array`]s for vertices and edges.
        ///
        /// It is generic in respect to the
        /// - vertices *const* capacity (`VCAP`),
        /// - edges *const* capacity (`ECAP`)
        /// - vertices data type (`V`),
        /// - edges data type (`E`),
        /// - storage (`S`).
        ///
        #[doc = "Edges are represented with the [`" $Edge "`]s type."]
        ///
        /// ## Features
        /// It supports multiple index sizes by enabling the corresponding features:
        /// `_graph_u[8|16|32|size]`.
        pub struct $Graph<const VCAP: usize, const ECAP: usize, V = (), E = (), S: Storage = Bare> {
            pub(super) verts: Array<Option<V>, VCAP, S>,
            pub(super) edges: Array<Option<$Edge<E>>, ECAP, S>,
        }

        /* constructors */

        // S: Bare, E: Copy
        impl<V, E: Copy, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, E, Bare> {
            /// Returns a disconnected graph, allocated in the stack,
            /// with the given array of `vertices`,
            /// where `E: Copy`.
            ///
            /// # Errors
            #[doc = "Returns [`IndexOutOfBounds`] if `(ECAP|VCAP) >= [`" $IDX "::MAX`]."]
            pub fn with_vertices(vertices: [Option<V>; VCAP]) -> Result<Self> {
                let _ = unwrap![ok? Self::check_capacity_bounds()];
                Ok(Self {
                    verts: Array::<Option<V>, VCAP, Bare>::new(vertices),
                    edges: Array::<Option<$Edge<E>>, ECAP, Bare>::with_copied(None),
                })
            }
        }

        // S: Bare, E: Clone
        impl<V, E: Clone, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, E, Bare> {
            /// Returns a disconnected graph, allocated in the stack,
            /// with the given array of `vertices`,
            /// where `E: Clone`.
            ///
            /// # Errors
            #[doc = "Returns [`IndexOutOfBounds`] if `(ECAP|VCAP) >= [`" $IDX "::MAX`]."]
            pub fn with_vertices_clone(vertices: [Option<V>; VCAP]) -> Result<Self> {
                let _ = unwrap![ok? Self::check_capacity_bounds()];
                Ok(Self {
                    verts: Array::<Option<V>, VCAP, Bare>::new(vertices),
                    edges: Array::<Option<$Edge<E>>, ECAP, Bare>::with_cloned(None),
                })
            }
        }

        // S: Bare, V: Copy, E: Copy
        impl<V: Copy, E: Copy, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, E, Bare> {
            /// Returns an empty graph, allocated in the stack, in *compile-time*,
            /// where `V: Copy, E: Copy`.
            ///
            /// # Errors
            #[doc = "Returns [`IndexOutOfBounds`] if `(ECAP|VCAP) >= [`" $IDX "::MAX`]."]
            ///
            /// # Examples
            /// ```
            #[doc = "use devela::" $Graph ";"]
            #[doc = "let g = " $Graph "::<10, 5, bool>::new_copy().unwrap();"]
            /// ```
            pub const fn new_copy() -> Result<Self> {
                let _ = unwrap![ok? Self::check_capacity_bounds()];
                Ok(Self {
                    verts: Array::<Option<V>, VCAP, Bare>::new_bare([None; VCAP]),
                    edges: Array::<Option<$Edge<E>>, ECAP, Bare>::new_bare([None; ECAP]),
                })
            }
        }

        // S: Bare, V: Clone, E: Clone.
        impl<V: Clone, E: Clone, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, E, Bare> {
            /// Returns an empty graph, allocated in the stack,
            /// where `V: Clone, E: Clone`.
            ///
            /// # Errors
            #[doc = "Returns [`IndexOutOfBounds`] if `(ECAP|VCAP) >= [`" $IDX "::MAX`]."]
            ///
            /// # Examples
            /// ```
            #[doc = "# use devela::" $Graph ";"]
            #[doc = "let g = " $Graph "::<10, 5, char>::new_clone().unwrap();"]
            /// ```
            pub fn new_clone() -> Result<Self> {
                Self::check_capacity_bounds()?;
                Ok(Self {
                    verts: Array::<Option<V>, VCAP, Bare>::with_cloned(None),
                    edges: Array::<Option<$Edge<E>>, ECAP, Bare>::with_cloned(None),
                })
            }
        }

        /* methods: vertices */

        impl<const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, Bare> {
            /// Adds a vertex to the graph without data.
            ///
            /// # Errors
            /// Returns [`NotEnoughSpace`] if there's no space left for the new vertex.
            pub fn add_vertex(&mut self) -> Result<$IDX> {
                for (idx, vertex) in self.verts.iter_mut().enumerate() {
                    if vertex.is_none() {
                        *vertex = Some(());
                        return Ok(idx as $IDX);
                    }
                }
                Err(NotEnoughSpace(Some(1)))
            }
        }

        // S: Bare
        impl<V, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, Bare> {
            /// Adds a vertex to the graph with the given `data`.
            ///
            /// # Errors
            /// Returns [`NotEnoughSpace`] if there's no space left for the new vertex.
            pub fn add_vertex_with(&mut self, data: V) -> Result<$IDX> {
                for (idx, vertex) in self.verts.iter_mut().enumerate() {
                    if vertex.is_none() {
                        *vertex = Some(data);
                        return Ok(idx as $IDX);
                    }
                }
                Err(NotEnoughSpace(Some(1)))
            }

            /// Returns `true` if there's no space left for vertices.
            pub const fn is_vertices_full(&self) -> bool {
                self.remaining_vertices_capacity() == 0
            }
            /// Returns the remaining capacity for additional vertices.
            pub const fn remaining_vertices_capacity(&self) -> $IDX {
                VCAP as $IDX - self.vertices_count()
            }
            /// Returns the number of existing vertices.
            pub const fn vertices_count(&self) -> $IDX {
                let mut i = 0;
                let mut count = 0;
                while i < VCAP {
                    count += self.verts.as_bare_slice()[i].is_some() as $IDX;
                    i += 1;
                }
                count
            }

            /// Returns `true` if a given `vertex` id exists.
            pub const fn vertex_exists(&self, vertex: $IDX) -> bool {
                self.verts.as_bare_slice()[vertex as usize].is_some()
            }

            /// Removes a vertex from the graph, with the given `id`.
            ///
            /// # Errors
            /// Returns [`IndexOutOfBounds`] if `id >= VCAP`,
            /// or [`NodeEmpty`] if the vertex didn't exist.
            pub fn remove_vertex(&mut self, id: $IDX) -> Result<()> {
                Self::check_vertex_bounds(id)?;
                let id = id as usize;
                if self.verts[id].is_none() {
                    Err(NodeEmpty(Some(id)))
                } else {
                    self.verts[id] = None;
                    Ok(())
                }
            }

            /* methods: edges */

            /// Returns `true` if there's no space left for edges.
            pub const fn is_edges_full(&self) -> bool {
                self.remaining_edges_capacity() == 0
            }
            /// Returns the remaining capacity for additional edges.
            pub const fn remaining_edges_capacity(&self) -> $IDX {
                ECAP as $IDX - self.edges_count()
            }

            /// Returns the number of existing edges.
            pub const fn edges_count(&self) -> $IDX {
                let mut i = 0;
                let mut count = 0;
                while i < ECAP {
                    count += self.edges.as_bare_slice()[i].is_some() as $IDX;
                    i += 1;
                }
                count
            }

            /// Returns `true` if a given edge `id` exists.
            pub const fn edge_exists(&self, id: $IDX) -> bool {
                if id as usize >= ECAP || self.edges.as_bare_slice()[id as usize].is_none() {
                    false
                } else {
                    true
                }
            }
            /// Returns `true` if a given edge `id` exists.
            ///
            /// # Panics
            /// Panics if `id >= ECAP`.
            pub const fn edge_exists_unchecked(&self, id: $IDX) -> bool {
                self.edges.as_bare_slice()[id as usize].is_some()
            }

            /// Adds an edge to the graph, connecting the given `orig` and `dest` vertices.
            ///
            /// Returns the new edge id, if there was space for it.
            ///
            /// # Errors
            /// Returns [`IndexOutOfBounds`] if `orig|dest >= VCAP`,
            /// or [`NodeEmpty`] if any of the given vertices is empty,
            /// or [`NotEnoughSpace`] if the array of edges is full.
            pub fn add_edge(&mut self, orig: $IDX, dest: $IDX) -> Result<$IDX> {
                Self::check_vertex_bounds(orig)?;
                Self::check_vertex_bounds(dest)?;

                if !self.vertex_exists(orig) { Err(NodeEmpty(Some(orig as usize))) }
                else if !self.vertex_exists(dest) { Err(NodeEmpty(Some(dest as usize))) }
                else {
                    // These can't fail since we've already checked their bounds
                    let orig = <$Index>::new(orig).unwrap();
                    let dest = <$Index>::new(dest).unwrap();

                    for (id, edge) in self.edges.iter_mut().enumerate() {
                        if edge.is_none() {
                            *edge = Some($Edge::new_some_valid(orig, dest));
                            return Ok(id as $IDX);
                        }
                    }
                    Err(NotEnoughSpace(Some(1)))
                }
            }

            /// Removes the edge with the given `id`.
            ///
            /// # Errors
            /// Returns [`IndexOutOfBounds`] if `id >= ECAP`.
            pub fn remove_edge(&mut self, id: $IDX) -> Result<()> {
                Self::check_edge_bounds(id)?;
                self.edges[id as usize] = None;
                Ok(())
            }
        }

        // V: PartialEq
        impl<V: PartialEq, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V> {
            /// Adds a vertex to the graph with the given `data`, but only if the data is unique.
            ///
            /// # Errors
            /// Returns [`NotEnoughSpace`] if there's no space left for the new vertex,
            /// or returns [`KeyAlreadyExists`] if there's already a vertex with the same data.
            pub fn add_vertex_unique(&mut self, data: V) -> Result<$IDX> {
                let (found, free) = self.find_vertex_and_first_free(&data);

                if free.is_none() {
                    return Err(NotEnoughSpace(Some(1)));
                } else if found.is_none() {
                    if let Some(id) = free {
                        self.verts[id] = Some(data);
                        return Ok(id as $IDX);
                    }
                }
                Err(KeyAlreadyExists)
            }

            /// Checks for the presence of the given vertex `data` and returns its id.
            #[must_use]
            pub fn find_vertex(&self, data: &V) -> Option<$IDX> {
                for (id, vertex) in self.verts.iter().enumerate() {
                    if let Some(v) = vertex {
                        if v == data {
                            return Some(id as $IDX);
                        }
                    }
                }
                None
            }
        }

        /* methods: helpers */

        impl<V: PartialEq, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V> {
            // Tries to find the given vertex `data` and returns its id,
            // as well as the first free vertex id.
            fn find_vertex_and_first_free(&self, data: &V) -> (Option<usize>, Option<usize>) {
                let mut found = None;
                let mut free = None;

                for (id, vertex) in self.verts.iter().enumerate() {
                    if found.is_none() && vertex.as_ref() == Some(data) { found = Some(id); }
                    else if free.is_some() { break; }

                    if free.is_none() && vertex.is_none() { free = Some(id); }
                    else if found.is_some() { break; }
                }
                (found, free)
            }
        }

        #[allow(dead_code)]
        impl<V, E, const VCAP: usize, const ECAP: usize> $Graph<VCAP, ECAP, V, E, Bare> {
            // Makes sure the given vertex `id` is in bounds,
            const fn check_vertex_bounds(id: $IDX) -> Result<()> {
                if id == $IDX::MAX || id >= VCAP as $IDX {
                    Err(IndexOutOfBounds(Some(id as usize)))
                } else {
                    Ok(())
                }
            }
            // Makes sure the given edge `id` is in bounds,
            const fn check_edge_bounds(id: $IDX) -> Result<()> {
                if id == $IDX::MAX || id >= ECAP as $IDX {
                    Err(IndexOutOfBounds(Some(id as usize)))
                } else {
                    Ok(())
                }
            }

            // Makes sure the capacity const-generic arguments are in bounds.
            const fn check_capacity_bounds() -> Result<()> {
                if ECAP >= $IDX::MAX as usize {
                    Err(IndexOutOfBounds(Some(ECAP)))
                } else if VCAP >= $IDX::MAX as usize {
                    Err(IndexOutOfBounds(Some(VCAP)))
                } else {
                    Ok(())
                }
            }
        }
    }};
}
impl_graph!();