YogEx 🌳
যোগ • (jōg) noun
- connection, link, union
- addition, sum
λ
/|\
/ | \
/ | \
Y | O--------G
/ | \ /
/ | \ /
/ | \ /
যো------+-------গ
/ \ | / \
/ \ | / \
/ \ | / \
✦ ✦ | ✦ ✦
Yog is a set of Graph and Network algorithms and data structures implemented in Elixir and packaged as a common API.
It started off as a wrapper around the Gleam Yog library, but now YogEx is now fully implemented in Elixir and a superset of the original Gleam version.
See the Gleam vs Elixir Comparison for feature comparison between the two implementations.
[!WARNING] API Stability: Until the version reaches 0.98.0
1.0.0, there may bebreaking changes. While I'll try my best to keep the API stable, there's no guaranteesome hiccups in performance, and documentation quality maybe less than optimal. Before version 0.99.0 is released in 12-12-2026, the primary focus is on performance, documentation, and bugfixes. From v0.96.0 onwards, there will be no breaking changes until v2.0 (No plans for that).
Features
YogEx provides comprehensive graph algorithms organized into modules:
Core Capabilities
Pathfinding & Flow — Shortest paths (Dijkstra, A*, Bellman-Ford, Floyd-Warshall, Johnson's), maximum flow (Edmonds-Karp), min-cut (Stoer-Wagner), and implicit state-space search for on-demand graphs.
Network Analysis — Centrality measures (PageRank, betweenness, closeness, eigenvector, Katz), community detection (Louvain, Leiden, Infomap, Walktrap), and network health metrics.
Connectivity & Structure — SCCs (Tarjan/Kosaraju), bridges, articulation points, K-core decomposition, and reachability analysis with exact and HyperLogLog-based estimation.
Graph Operations — Union, intersection, difference, Cartesian product, power, isomorphism, and O(1) transpose.
Developer Experience
Generators & Builders — Classic patterns (complete, cycle, grid, Petersen), random models (SBM, R-MAT), and a comprehensive Maze Generation suite (Recursive Backtracker, Wilson's, Kruskal's, Eller's, etc.) with labeled and grid builders.
I/O & Visualization — GraphML, GDF, Pajek, LEDA, TGF, JSON serialization plus ASCII, DOT, and Mermaid rendering.
Functional Graphs(Experimental) — Pure inductive graph library (FGL) for elegant recursive algorithms.
Complete Algorithm Catalog — See all 60+ algorithms, underlying data structures (Pairing Heap, Union-Find, HyperLogLog), and selection guidance with Big-O complexities.
Installation
Basic Installation
Add YogEx to your list of dependencies in mix.exs:
def deps do
[
{:yog_ex, "~> 0.95.0"}
]
endThen run:
mix deps.getOptional Dependencies
YogEx includes several optional dependencies that enable additional I/O and interoperability features:
| Dependency | Module | Purpose |
|---|---|---|
:saxy | Yog.IO.GraphML |
Fast streaming XML parser for GraphML files (3-4x faster than default :xmerl) |
:jason | Yog.IO.JSON | JSON serialization/deserialization for D3.js, Cytoscape, vis.js, NetworkX formats |
:libgraph | Yog.IO.Libgraph | Bidirectional conversion with libgraph library |
To use these features, add the optional dependencies to your mix.exs:
def deps do
[
{:yog_ex, "~> 0.95.0"},
{:saxy, "~> 1.5"}, # For fast GraphML/XML parsing
{:jason, "~> 1.4"}, # For JSON import/export
{:libgraph, "~> 0.16"} # For libgraph interoperability
]
endXML/GraphML with Saxy
# Reading large GraphML files is significantly faster with saxy
{:ok, graph} = Yog.IO.GraphML.read("large_network.graphml")
# Writing GraphML
Yog.IO.GraphML.write("output.graphml", graph)JSON Serialization with Jason
# Export to various JSON formats
json = Yog.IO.JSON.to_json(graph, Yog.IO.JSON.export_options_for(:d3_force))
# Import from JSON
{:ok, graph} = Yog.IO.JSON.from_json(json_string)Libgraph Interoperability
# Convert Yog graph to libgraph
libgraph = Yog.IO.Libgraph.to_libgraph(graph)
# Convert libgraph back to Yog
{:ok, yog_graph} = Yog.IO.Libgraph.from_libgraph(libgraph)Livebook
For livebook, add the following:
Mix.install(
{:yog_ex, "~> 0.95.0"}
)There is a Kino App that can be used to explore the library and create and render graphs.
Usage
alias Yog.Pathfinding
# Create a directed graph
graph =
Yog.directed()
|> Yog.add_node(1, "Start")
|> Yog.add_node(2, "Middle")
|> Yog.add_node(3, "End")
|> Yog.add_edge_ensure(from: 1, to: 2, with: 5)
|> Yog.add_edge_ensure(from: 2, to: 3, with: 3)
|> Yog.add_edge_ensure(from: 1, to: 3, with: 10)
# Find shortest path using Dijkstra (uses :ok/:error tuples and Path struct)
case Pathfinding.shortest_path(
in: graph,
from: 1,
to: 3
) do
{:ok, path} ->
IO.puts("Found path with weight: #{path.weight}")
:error ->
IO.puts("No path found")
end
# => Found path with weight: 8Examples
Detailed examples are located in the examples/ directory
Advent of Code Solutions
YogEx is used to solve Advent of Code challenges.
See all Advent of Code solutions tagged with graph that demonstrate usage of YogEx algorithms in the Advent of Code repository.
Development
Running Tests
mix testRun tests for a specific module:
mix test test/yog/pathfinding/dijkstra_test.exsProject Structure
lib/yog/— Core graph library modules (pure Elixir)test/— Unit tests and doctestsexamples/— Real-world usage examples
Property-Based Testing
This library uses property-based testing (PBT) via StreamData to ensure that algorithms hold up against a wide range of automatically generated graph structures.
See the PROPERTIES.md for a complete catalog of all algorithmic invariants (hypotheses) verified by the test suite.
AI Assistance
Parts of this project were developed with the assistance of AI coding tools. All AI-generated code has been reviewed, tested, and validated by the maintainer.