gryf 0.2.1

Graph data structure library with focus on convenience, versatility, correctness and performance.
Documentation
//! The [complement] of an undirected graph, that is, a graph with the same
//! vertices but they are connected if and only if they were not connected in
//! the original graph.
//!
//! [complement]: https://en.wikipedia.org/wiki/Complement_graph

use rustc_hash::FxHashSet;

use crate::core::{
    EdgeSet, GraphBase, GraphFull, Neighbors, VertexSet,
    base::{NeighborRef, NeighborReference},
    borrow::OwnableRef,
    facts,
    id::{AsIdRef, IdType, IntegerIdType},
    marker::{Direction, Undirected},
};

use gryf_derive::{GraphBase, VertexSet};

/// The [complement] of an undirected graph, that is, a graph with the same
/// vertices but they are connected if and only if they were not connected in
/// the original graph.
///
/// [complement]: https://en.wikipedia.org/wiki/Complement_graph
///
/// # Examples
///
/// ```
/// use gryf::{Graph, adapt::Complement, core::Neighbors};
///
/// let mut graph = Graph::new_undirected();
///
/// let v0 = graph.add_vertex(());
/// let v1 = graph.add_vertex(());
/// let v2 = graph.add_vertex(());
///
/// graph.add_edge(&v0, &v1, "original");
///
/// assert_eq!(graph.edge_count(), 1);
///
/// let neighbor = graph.neighbors_undirected(&v0).next().unwrap();
/// assert_eq!(neighbor.id, v1);
/// assert_eq!(graph.edge(&neighbor.edge), Some(&"original"));
///
/// // Create graph complement.
/// let complement = Complement::new(graph, "complement");
///
/// assert_eq!(complement.edge_count(), 2);
///
/// let neighbor = complement.neighbors_undirected(&v0).next().unwrap();
/// assert_eq!(neighbor.id, v2);
/// assert_eq!(complement.edge(&neighbor.edge), Some(&"complement"));
/// ```
#[derive(Debug, GraphBase, VertexSet)]
#[gryf_crate]
pub struct Complement<E, G> {
    #[graph]
    graph: G,
    edge: E,
}

impl<E, G> Complement<E, G>
where
    G: GraphBase<EdgeType = Undirected> + VertexSet + EdgeSet,
{
    /// Creates a complement of the given graph, using `edge` value for all
    /// complement edges.
    pub fn new(graph: G, edge: E) -> Self {
        Self { graph, edge }
    }

    /// Consumes the adapter and returns the wrapped graph.
    pub fn into_inner(self) -> G {
        self.graph
    }

    #[doc = include_str!("../../docs/include/edge_set.edge_count.md")]
    pub fn edge_count(&self) -> usize {
        facts::complete_graph_edge_count::<Undirected>(self.graph.vertex_count())
            - self.graph.edge_count()
    }

    #[doc = include_str!("../../docs/include/graph_ref.edge.md")]
    pub fn edge<EI>(&self, id: EI) -> Option<&E>
    where
        EI: AsIdRef<G::EdgeId>,
    {
        if id.as_id().is_sentinel() {
            Some(&self.edge)
        } else {
            None
        }
    }

    #[doc(hidden)]
    pub fn apply<V>(self) -> G
    where
        G: GraphFull<V, E>,
        E: Clone,
    {
        let mut graph = self.graph;
        let original_edges = graph
            .edges_by_id()
            .map(|e| graph.endpoints(&e).unwrap())
            .collect::<FxHashSet<_>>();
        let vertices = graph.vertices_by_id().collect::<Vec<_>>();

        graph.clear_edges();

        for (i, u) in vertices.iter().enumerate() {
            for v in vertices.iter().skip(i) {
                if !original_edges.contains(&(u.clone(), v.clone()))
                    && !original_edges.contains(&(v.clone(), u.clone()))
                {
                    graph.add_edge(u, v, self.edge.clone());
                }
            }
        }

        graph
    }
}

impl<E, G> Neighbors for Complement<E, G>
where
    G: Neighbors + VertexSet,
    G::EdgeId: IntegerIdType,
{
    type NeighborRef<'a>
        = NeighborRef<G::VertexId, G::EdgeId>
    where
        Self: 'a;

    type NeighborsIter<'a>
        = NeighborsIter<'a, G>
    where
        Self: 'a;

    fn neighbors_undirected(&self, from: &G::VertexId) -> Self::NeighborsIter<'_> {
        NeighborsIter {
            from: from.clone(),
            dir: Direction::Outgoing,
            neighbors: self
                .graph
                .neighbors_undirected(from)
                .map(|n| n.id().into_owned().into())
                .collect(),
            vertices: self.graph.vertices_by_id(),
        }
    }

    fn neighbors_directed(&self, from: &G::VertexId, dir: Direction) -> Self::NeighborsIter<'_> {
        NeighborsIter {
            from: from.clone(),
            dir,
            neighbors: self
                .graph
                .neighbors_directed(from, dir)
                .map(|n| n.id().into_owned().into())
                .collect(),
            vertices: self.graph.vertices_by_id(),
        }
    }
}

pub struct NeighborsIter<'a, G>
where
    G: VertexSet + 'a,
{
    from: G::VertexId,
    dir: Direction,
    neighbors: FxHashSet<OwnableRef<'a, G::VertexId>>,
    vertices: G::VerticesByIdIter<'a>,
}

impl<'a, G> Iterator for NeighborsIter<'a, G>
where
    G: VertexSet + 'a,
    G::EdgeId: IntegerIdType,
{
    type Item = NeighborRef<G::VertexId, G::EdgeId>;

    fn next(&mut self) -> Option<Self::Item> {
        loop {
            let v = self.vertices.next()?;

            if v != self.from && !self.neighbors.contains(&v) {
                return Some(NeighborRef {
                    id: v,
                    edge: IdType::sentinel(),
                    pred: self.from.clone(),
                    dir: self.dir,
                });
            }
        }
    }
}

#[cfg(test)]
mod tests {
    use crate::{
        core::{
            GraphAdd,
            id::{DefaultId, VertexId},
        },
        storage::{AdjList, Stable},
    };

    use super::*;

    use std::collections::HashSet;

    #[test]
    fn edge_count() {
        let mut graph: AdjList<_, _, _, DefaultId> = AdjList::new();

        let v0 = graph.add_vertex(());
        let v1 = graph.add_vertex(());
        let v2 = graph.add_vertex(());
        let v3 = graph.add_vertex(());

        graph.add_edge(&v0, &v1, ());
        graph.add_edge(&v1, &v2, ());
        graph.add_edge(&v2, &v3, ());
        graph.add_edge(&v3, &v1, ());

        let complement = Complement::new(graph, ());
        assert_eq!(complement.edge_count(), 2);
    }

    #[test]
    fn apply() {
        let mut graph: AdjList<_, _, _, DefaultId> = AdjList::new();

        let v0 = graph.add_vertex(());
        let v1 = graph.add_vertex(());
        let v2 = graph.add_vertex(());
        let v3 = graph.add_vertex(());

        graph.add_edge(&v0, &v1, ());
        graph.add_edge(&v1, &v2, ());
        graph.add_edge(&v2, &v3, ());
        graph.add_edge(&v3, &v1, ());

        let complement: AdjList<_, _, _, _> = Complement::new(graph, ()).apply();

        assert!(complement.edge_id_any(&v0, &v1).is_none());
        assert!(complement.edge_id_any(&v1, &v2).is_none());
        assert!(complement.edge_id_any(&v2, &v3).is_none());
        assert!(complement.edge_id_any(&v3, &v1).is_none());

        assert!(complement.edge_id_any(&v0, &v3).is_some());
        assert!(complement.edge_id_any(&v0, &v2).is_some());
    }

    #[test]
    fn neighbors() {
        let mut graph: AdjList<_, _, _, DefaultId> = AdjList::new();

        let v0 = graph.add_vertex(());
        let v1 = graph.add_vertex(());
        let v2 = graph.add_vertex(());
        let v3 = graph.add_vertex(());

        graph.add_edge(&v0, &v1, ());
        graph.add_edge(&v1, &v2, ());
        graph.add_edge(&v2, &v3, ());
        graph.add_edge(&v3, &v1, ());

        let complement = Complement::new(graph, ());

        assert_eq!(
            complement
                .neighbors_undirected(&v0)
                .map(|n| n.id().into_owned())
                .collect::<HashSet<VertexId>>(),
            vec![v2, v3].into_iter().collect()
        );
        assert_eq!(complement.neighbors_undirected(&v1).count(), 0);
        assert_eq!(
            complement
                .neighbors_undirected(&v2)
                .map(|n| n.id().into_owned())
                .collect::<HashSet<_>>(),
            vec![v0].into_iter().collect()
        );
        assert_eq!(
            complement
                .neighbors_undirected(&v3)
                .map(|n| n.id().into_owned())
                .collect::<HashSet<_>>(),
            vec![v0].into_iter().collect()
        );
    }

    #[test]
    fn apply_holes() {
        let mut graph: Stable<AdjList<_, _, _, DefaultId>> = Stable::new(AdjList::new());

        let v0 = graph.add_vertex(());
        let v1 = graph.add_vertex(());
        let v2 = graph.add_vertex(());
        let v3 = graph.add_vertex(());
        let v4 = graph.add_vertex(());

        graph.add_edge(&v0, &v1, ());
        graph.add_edge(&v1, &v2, ());
        graph.add_edge(&v2, &v3, ());
        graph.add_edge(&v3, &v4, ());
        graph.add_edge(&v4, &v1, ());

        graph.remove_vertex(&v3);

        let complement: Stable<AdjList<_, _, _, _>> = Complement::new(graph, ()).apply();

        assert!(complement.edge_id_any(&v0, &v1).is_none());
        assert!(complement.edge_id_any(&v1, &v2).is_none());
        assert!(complement.edge_id_any(&v4, &v1).is_none());

        assert!(complement.edge_id_any(&v0, &v2).is_some());
        assert!(complement.edge_id_any(&v0, &v4).is_some());
        assert!(complement.edge_id_any(&v2, &v4).is_some());
    }
}