Metadata-Version: 2.4
Name: fast-sugiyama
Version: 0.5.3
Classifier: Development Status :: 4 - Beta
Classifier: Environment :: Console
Classifier: Intended Audience :: Science/Research
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: MIT License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3 :: Only
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Classifier: Programming Language :: Python :: 3.15
Classifier: Programming Language :: Rust
Classifier: Topic :: Scientific/Engineering
Classifier: Topic :: Multimedia :: Graphics :: Presentation
Classifier: Topic :: Scientific/Engineering :: Visualization
Classifier: Typing :: Typed
Requires-Dist: rectangle-packer>=2 ; extra == 'all'
Requires-Dist: fast-sugiyama[test] ; extra == 'dev'
Requires-Dist: pre-commit>=3,<5 ; extra == 'dev'
Requires-Dist: ruff>=0.13 ; extra == 'dev'
Requires-Dist: matplotlib>=3.10 ; extra == 'dev'
Requires-Dist: fast-sugiyama[all] ; extra == 'test'
Requires-Dist: pytest>=8 ; extra == 'test'
Requires-Dist: pytest-cov>=7 ; extra == 'test'
Requires-Dist: pytest-xdist>=3 ; extra == 'test'
Requires-Dist: networkx ; extra == 'test'
Provides-Extra: all
Provides-Extra: dev
Provides-Extra: test
License-File: LICENSE
Summary: A fast and flexible hierarchical graph layout engine.
Keywords: graph-display,directed-graph,digraph,graph,pydot,layout,dot layout,graph layout,network,networkx,graphviz
Author-email: Austin Orr <austinmartinorr@gmail.com>
License-Expression: MIT
Requires-Python: >=3.11
Description-Content-Type: text/markdown; charset=UTF-8; variant=GFM

# `Fast-Sugiyama`

[![codecov](https://codecov.io/gh/austinorr/fast-sugiyama/graph/badge.svg?token=0GWVZIWGIJ)](https://codecov.io/gh/austinorr/fast-sugiyama)
[![Rust Lint & Test](https://github.com/austinorr/fast-sugiyama/actions/workflows/rust_ci.yml/badge.svg)](https://github.com/austinorr/fast-sugiyama/actions/workflows/rust_ci.yml)
[![Python Lint & Test](https://github.com/austinorr/fast-sugiyama/actions/workflows/python_ci.yml/badge.svg)](https://github.com/austinorr/fast-sugiyama/actions/workflows/python_ci.yml)
<a href="https://pypi.org/project/fast-sugiyama" target="_blank">
<img src="https://img.shields.io/pypi/v/fast-sugiyama?color=%2334D058&label=pypi%20package" alt="Package version">
</a>

This project adds python bindings for the [`rust-sugiyama`](https://crates.io/crates/rust-sugiyama) crate, to produce directed graph layouts similar to the GraphViz `dot` program.

It's an implementation of Sugiyama's algorithm for displaying a layered, or hierarchical, directed graph like so:

<div align="center" >
    <img width=400 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/hero.png" alt="Graph Example">
</div>

Why?

We already have `nx.nx_pydot.pydot_layout(gn, prog="dot")`, `grandalf.layouts.SugiyamaLayout`, `netgraph.get_sugiyama_layout`, and `rustworkx.visualization.graphviz_draw`... why pile on?

All of the above either depend on either the `dot` program from `graphviz`, or the pure-python Sugiyama implementation from `grandalf`.

The `dot` program is probably the best solution to this problem that currently exists, and you should use the `dot` program if you are working in an environment that allows you to install `graphviz` and your python application only needs to render very small graphs and/or you don't need to produce them quickly.

The `grandalf`-based solutions are pure-python which is awesome for portability, but is slow for medium/large graphs (because python).
In addition, that project hasn't had maintenance in many years, including incorporating the 2020 Erratum from Brandes et., al on which the implementation appears to be based.
This implementation suffers from some occasions layouts where two nodes can share position, which either reveals a bug referenced in the erratum, or in the original implementation.
Anyway, the `grandalf` project has an excellent framework for this problem, and has a deep bench of potential mentors and prior contributors, so if you need a pure-python implementation you should support them.

This project is about having zero 3rd-party dependencies, reasonable multi-platform support (arm & x86 wheels are available for linux, windows, and mac), and speed.
In service of all three of these goals, this library is a thin wrapper around the excellent rust crate [`rust-sugiyama`](https://crates.io/crates/rust-sugiyama), which is likely the clearest-eyed implementation of the complex Sugiyama algorithm that's available for free on the internet.

With these shoulders to stand on, this project aims to offer very good `dot`-like layouts, very quickly, and with no 3rd-party runtime dependencies.
This is a great fit for folks who can't (or don't want to) install `graphviz`, or for web-applications that want smaller images, or for anyone needing a large directed graph layout to run _fast_.

## Quick Start

```python
import networkx as nx
from fast_sugiyama import from_edges

g = nx.gn_graph(42, seed=132, create_using=nx.DiGraph)
pos = from_edges(g.edges()).to_dict()
nx.draw_networkx(g, pos=pos, with_labels=False, node_size=150)
```

<div align="left" >
    <img width=500 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/quickstart.png" alt="Quick Start Output">
</div>

The above example is for a `networkx` graph, but the `fast-sugiyama` package can be used with any graph library.
The return type from the `to_dict()` method works equally well for the `rustworkx` `Pos2DMapping` type as the `networkx` pos arg.
It's a `{node_id: (x, y)}` coordinate mapping you can take where you please.

You might also wonder why we don't emulate the `networkx` layout api conventions exactly (e.g., immediately return exactly one layout), and we'll talk about that a little later.

## Install

This package is pip installable.

```console
pip install fast-sugiyama
```

As with all python endeavours, you should probably be using a virtual environment for your project via [`conda` \ `mamba`](https://github.com/conda-forge/miniforge) or [`venv`](https://docs.astral.sh/uv/pip/environments/) (preferrably via [`uv`](https://docs.astral.sh/uv/)).

```console
uv pip install fast-sugiyama
```

Either of the above two install commands should work well with either ARM or x86 systems running Linux, Mac, or Windows.
If you have another system and want to run this package and you can't install/build/run from the sdist or the repo please open a detailed issue and let me know what happened.
Maybe I can help.

## Usage

Purely for simplicity, this section will use `networkx` to facilitate usage demonstration and discussion.
Again, this project has no dependencies, and can be used to support high-quality layout visuals for any graph library that uses a node -> coord mapping.

The `from_edges` function returns a list of layouts, one for each weakly connected component of the input graph.
Each layout is a tuple containing the positions, width, height, and edges returned by the solver.
This makes it easy to deal with multi-graphs, and let's us write helper functions to support various ways of displaying separate graphs in one layout.

This section will use a multi-graph from the test suite that includes 40 component graphs, each with a range of sizes.
In these examples, the full graph `g` is made of 40 components, 435 nodes, and 395 edges.
There is also a sub-set of these components called `sg` that includes just 12 of the 40 components.

### `dot`-like layouts

This package has a helper function for users who would like to get a layout (or a multi-graph layout) similar to the `dot` program -- specifically, similar to the layout produced by `nx.pydot.pydot_layout`.
This layout aligns hierarchies to the top of the figure, lines them up side by side horizontally, and packs them efficiently such that nodes in neighboring graph components are separated by the same spacing interval as nodes within each component.
The following plot compares the layout results for a 12 component multi-graph `sg`.

<div align="center" >
    <img width=900 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/pydot_compare.png" alt="Pydot Compare">
</div>

Note that the bounding boxes of the layouts are allowed to overlap in the `pydot` layout, so this library supports that as well.

### rect-like / compact layouts

Lining up components horizontally quickly becomes unwieldy with the `pydot` layout though.
Here's all 40 components of our multi-graph `g`.

<div align="center" >
    <img width=900 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/difficult_pydot.png" alt="Difficult Pydot Layout">
</div>

Users can install an optional dependency `rectangle-packer` to produce an improved layout that compactly packs the components.

```python
pos = from_edges(g.edges()).rect_pack_layouts(max_width=2500).to_dict()
```

<div align="center" >
    <img height=500 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/rect_pack_layout.png" alt="Rect Pack Layout">
</div>

One can even utilize the 'horizontal compact' idea from the dot-like layouts to tighten things up even further.

```python
pos = from_edges(g.edges()).compact_layout(max_width=2500).to_dict()
# or equivalently
pos = (
    from_edges(g.edges())
    .rect_pack_layouts(max_width=2500)
    .sort_horizontal()
    .compact_layouts_horizontal()
)
```

<div align="center" >
    <img height=500 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/compact_layout.png" alt="Compact Layout">
</div>

### DIY layouts

The helper layouts above simply chain a couple layout 'primitive' operations.
For example, the `.dot_layout()` is equivalent to `.align_layouts_vertical_top().compact_layouts_horizontal()`
You can chain together the built in primitives, or adapt their code into your own custom functions to obtain any arrangement.

```python
pos = (
    from_edges(g.edges())
    [:6] # chaining even works after slices
    .align_layouts_vertical()
    .align_layouts_horizontal_center()
    .to_dict()
)
```

<div align="center" >
    <img height=500 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/slice_vert_center_layout.png" alt="Sliced, centered, vertically stacked">
</div>

If you're feeling wild and your graph is not simple and acyclic, you can even capture the dummy vertices and their edges from the Sugiyama solver and render those.

Here's another graph from the test suite that is directed but is not a DAG.
Let's say these edges are in a variable called `edges`.

```python
g = nx.DiGraph()
g.add_edges_from(edges)
pos = from_edges(edges).to_dict()

fig, ax = plt.subplots()
nx.draw_networkx(g, pos=pos, ax=ax)
```

<div align="left" >
    <img width=400 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/hero_no_dummies.png" alt="Hero with no dummies">
</div>

But there's an issue with the edge from (15, 1) -- the algorithm created a gap for the edge to go around node 8, not behind it.
We can render that edge path easily by using the edges returned by the solver in our layout.

```python
layouts = from_edges(edges)
pos = layouts.to_dict()
nodes = list({n for edge in edges for n in edge})

g = nx.DiGraph()
for *_, edgelist in layouts: # the solver returns the edges with the dummy's included
    if edgelist is not None:
        g.add_edges_from(edgelist)

fig, ax = plt.subplots()

nx.draw_networkx_nodes(g, pos, ax=ax, nodelist=nodes)
nx.draw_networkx_labels(g, pos, ax=ax, labels={n: n for n in nodes})

_ = nx.draw_networkx_edges(
    g,
    pos,
    [e for e in g.edges() if e[1] not in nodes],
    arrows=False,
    ax=ax,
)
_ = nx.draw_networkx_edges(
    g,
    pos,
    [e for e in g.edges() if e[1] in nodes],
    node_size=[300 if n in nodes else 0 for n in g.nodes()],
    ax=ax,
)
```

<div align="left" >
    <img width=400 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/hero.png" alt="Hero with dummies">
</div>

## Benchmarks

Speed is not the purpose of this package, but it is a welcome benefit of using rust for all the heavy lifting.
There are vanishingly few non-pathological reasons to produce a hierarchical graph plot with 5,000+ nodes, but for reasonable graph sizes, say, under 1000, this package is ~100-400x faster than using `pydot`.
NOTE: `pydot` will typically yield a nicer looking layout, and if you can install it in your environment and wait for `pydot` to do its serialization/deserialization, you should.

```python
for n in [100, 500, 1000, 5000, 10000]:
    g = nx.gn_graph(n, seed=42)

    %timeit _ = from_edges(g.edges()).dot_layout().to_dict()
    %timeit _ = nx.nx_pydot.pydot_layout(g, prog='dot')
```

<pre>
<b>786 μs ± 6.88 μs per loop (mean ± std. dev. of 7 runs, 1,000 loops each) <-- ours @ n==100 (410x faster)</b>
325 ms ± 9.01 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
<b>10.9 ms ± 112 μs per loop (mean ± std. dev. of 7 runs, 100 loops each) <-- ours @ n==500 (140x faster)</b>
1.57 s ± 27.7 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
<b>35.8 ms ± 48.2 μs per loop (mean ± std. dev. of 7 runs, 10 loops each) <-- ours @ n==1000 (89x faster)</b>
3.2 s ± 8.41 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
<b>855 ms ± 5.84 ms per loop (mean ± std. dev. of 7 runs, 1 loop each) <-- ours @ n==5000 (21x faster)</b>
18.7 s ± 234 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
<b>3.1 s ± 23.8 ms per loop (mean ± std. dev. of 7 runs, 1 loop each) <-- ours @ n==10000 (16x faster)</b>
49.6 s ± 1.07 s per loop (mean ± std. dev. of 7 runs, 1 loop each)
</pre>

And if we plot this up...

<div align="left" >
    <img width=500 src="https://raw.githubusercontent.com/austinorr/fast-sugiyama/main/crates/py-sugiyama/misc/single_graph_bench.png" alt="Single Graph Benchmark">
</div>

# Rust Sugiyama

This project contains a fork of the excellent [`rust-sugiyama`](https://crates.io/crates/rust-sugiyama) crate, and includes minor api changes to support layouts via the python bindings.
This fork is not intended to ever be published to crates.io.
Instead anything useful we do here will be offered to the upstream crate for consideration.

Differences from the upstream project currently include:

- Compute layout width and height in layout units rather than node count
- Optionally validate the layout to prevent/detect overlapping nodes & invalid layouts
- Return dummy vertices and edges if they're part of the layout
- Remove dev-dependencies that point to personal github repos and replace with alternatives

Big thank you to the developers of the `rust-sugiyama` crate for doing the heavy lifting!

