Extract from limo mono repo commit 6d5297c814588061ba58ec4b40f5e377956b1fe5.

This commit is contained in:
James Pace 2026-06-20 09:05:28 -04:00
commit ef90912bfa
5 changed files with 209 additions and 0 deletions

1
.gitignore vendored Normal file
View File

@ -0,0 +1 @@
/target

8
Cargo.toml Normal file
View File

@ -0,0 +1,8 @@
[package]
name = "limbo_graph"
version = "0.1.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]

3
README.md Normal file
View File

@ -0,0 +1,3 @@
# limbo_graph
An implementation of a directed graph in Rust.

148
src/lib.rs Normal file
View File

@ -0,0 +1,148 @@
//
// Copyright 2023 James Pace
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at https://mozilla.org/MPL/2.0/.
//
// This Source Code Form is "Incompatible With Secondary Licenses", as
// defined by the Mozilla Public License, v. 2.0.
//
#![no_std]
extern crate alloc;
mod node;
use alloc::string::{String, ToString};
use alloc::vec;
use alloc::vec::Vec;
pub use node::*;
/// A generic graph type holding values connected to other values.
/// Values can be added to the graph, but not removed.
pub struct Graph<NodeValueT: NodeValue> {
nodes: Vec<Node<NodeValueT>>,
}
impl<NodeValueT: NodeValue> Graph<NodeValueT> {
/// Make a new graph with a root node with value `root`.
pub fn new(root: NodeValueT) -> Self {
// Make root node from its value.
let root_node = Node::new(root, None);
// Make graph with root_node as the one value in the vec.
Graph {
nodes: vec![root_node],
}
}
/// Add a child wth value `val` to the parent with Key `parent`.
/// If the parent key is not in the graph, returns an error.
/// Returns a result with the key of the new node or an error.
pub fn add(&mut self, val: NodeValueT, parent: Key) -> Result<Key, GraphError> {
// Make sure parent is valid.
if parent >= self.nodes.len() {
return Err(GraphError::from_msg("Parent node not in graph."));
}
// Add new node to graph, get it's key.
let new_node = Node::new(val, Some(parent));
self.nodes.push(new_node.clone());
let new_node_key = self.nodes.len() - 1;
// Add it's key to parent's children.
self.nodes[parent].add_child(new_node_key);
Ok(new_node_key)
}
/// Get the value of key `key` if the key is valid.
pub fn value_of(&self, key: &Key) -> Result<NodeValueT, GraphError> {
if key >= &self.nodes.len() {
return Err(GraphError::from_msg("Can't get value of invalid key."));
}
Ok(self.nodes[*key].value())
}
/// Get the children (as a list of keys) of key `key` if the key is valid.
pub fn children_of(&self, key: &Key) -> Result<Vec<Key>, GraphError> {
if key >= &self.nodes.len() {
return Err(GraphError::from_msg("Can't get children of invalid key."));
}
Ok(self.nodes[*key].children())
}
/// Get the parent of key `key` if the key is valid.
/// Will return None if the node at `key` as no parent (i.e. is the root node).
pub fn parent_of(&self, key: &Key) -> Result<Option<Key>, GraphError> {
if key >= &self.nodes.len() {
return Err(GraphError::from_msg("Can't get parent of invalid key."));
}
Ok(self.nodes[*key].parent())
}
/// Get the key for the root of the graph.
pub fn root_key(&self) -> Key {
// This is always 0.
0
}
}
#[derive(Debug, Clone)]
pub struct GraphError {
pub msg: String,
}
impl GraphError {
pub fn from_msg(msg: &str) -> Self {
GraphError {
msg: msg.to_string(),
}
}
}
impl core::fmt::Display for GraphError {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
write!(f, "Error manipulating graph: {}", self.msg)
}
}
impl core::error::Error for GraphError {}
#[cfg(test)]
mod tests {
use super::*;
use core::cmp::PartialEq;
#[derive(Debug, Clone, PartialEq)]
struct NodeType {
pub x: f64,
}
impl NodeValue for NodeType {}
impl NodeType {
fn new(val: f64) -> Self {
NodeType { x: val }
}
}
#[test]
fn node_manipulation() {
let mut node = Node::new(NodeType::new(1.0), None);
assert!(node.parent().is_none());
assert!(node.value() == NodeType::new(1.0));
let child_key: Key = 1;
node.add_child(child_key.clone());
assert!(node.children().len() == 1);
assert!(node.children()[0] == child_key);
}
#[test]
fn graph_manipulation() {
let root_val = NodeType::new(1.0);
let mut graph = Graph::new(root_val);
let second_val = NodeType::new(2.0);
let second_add_res = graph.add(second_val, 0);
assert!(second_add_res.is_ok());
assert!(second_add_res.unwrap() == 1);
assert!(graph.value_of(&1).unwrap() == NodeType::new(2.0));
assert!(graph.parent_of(&1).unwrap() == Some(0));
assert!(graph.children_of(&1).unwrap().len() == 0);
}
}

49
src/node.rs Normal file
View File

@ -0,0 +1,49 @@
//
// Copyright 2023 James Pace
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at https://mozilla.org/MPL/2.0/.
//
// This Source Code Form is "Incompatible With Secondary Licenses", as
// defined by the Mozilla Public License, v. 2.0.
//
use alloc::vec;
use alloc::vec::Vec;
pub type Key = usize;
pub trait NodeValue: Clone {}
#[derive(Debug, Clone)]
pub struct Node<NodeValueT: NodeValue> {
value: NodeValueT,
children: Vec<Key>,
parent: Option<Key>,
}
impl<NodeValueT: NodeValue> Node<NodeValueT> {
pub fn new(value: NodeValueT, parent: Option<Key>) -> Self {
Node {
value: value,
children: vec![],
parent: parent,
}
}
pub fn add_child(&mut self, child_key: Key) {
self.children.push(child_key);
}
pub fn children(&self) -> Vec<Key> {
self.children.clone()
}
pub fn parent(&self) -> Option<Key> {
self.parent.clone()
}
pub fn value(&self) -> NodeValueT {
self.value.clone()
}
}