|
|
"""Unit tests for pydot drawing functions.""" |
|
|
|
|
|
from io import StringIO |
|
|
|
|
|
import pytest |
|
|
|
|
|
import networkx as nx |
|
|
from networkx.utils import graphs_equal |
|
|
|
|
|
pydot = pytest.importorskip("pydot") |
|
|
|
|
|
|
|
|
class TestPydot: |
|
|
@pytest.mark.parametrize("G", (nx.Graph(), nx.DiGraph())) |
|
|
@pytest.mark.parametrize("prog", ("neato", "dot")) |
|
|
def test_pydot(self, G, prog, tmp_path): |
|
|
""" |
|
|
Validate :mod:`pydot`-based usage of the passed NetworkX graph with the |
|
|
passed basename of an external GraphViz command (e.g., `dot`, `neato`). |
|
|
""" |
|
|
|
|
|
|
|
|
|
|
|
G.graph["name"] = "G" |
|
|
|
|
|
|
|
|
G.add_edges_from([("A", "B"), ("A", "C"), ("B", "C"), ("A", "D")]) |
|
|
G.add_node("E") |
|
|
|
|
|
|
|
|
graph_layout = nx.nx_pydot.pydot_layout(G, prog=prog) |
|
|
assert isinstance(graph_layout, dict) |
|
|
|
|
|
|
|
|
P = nx.nx_pydot.to_pydot(G) |
|
|
|
|
|
|
|
|
G2 = G.__class__(nx.nx_pydot.from_pydot(P)) |
|
|
|
|
|
|
|
|
assert graphs_equal(G, G2) |
|
|
|
|
|
fname = tmp_path / "out.dot" |
|
|
|
|
|
|
|
|
P.write_raw(fname) |
|
|
|
|
|
|
|
|
Pin_list = pydot.graph_from_dot_file(path=fname, encoding="utf-8") |
|
|
|
|
|
|
|
|
assert len(Pin_list) == 1 |
|
|
|
|
|
|
|
|
Pin = Pin_list[0] |
|
|
|
|
|
|
|
|
n1 = sorted(p.get_name() for p in P.get_node_list()) |
|
|
|
|
|
|
|
|
n2 = sorted(p.get_name() for p in Pin.get_node_list()) |
|
|
|
|
|
|
|
|
assert n1 == n2 |
|
|
|
|
|
|
|
|
e1 = sorted((e.get_source(), e.get_destination()) for e in P.get_edge_list()) |
|
|
|
|
|
|
|
|
e2 = sorted((e.get_source(), e.get_destination()) for e in Pin.get_edge_list()) |
|
|
|
|
|
|
|
|
assert e1 == e2 |
|
|
|
|
|
|
|
|
Hin = nx.nx_pydot.read_dot(fname) |
|
|
Hin = G.__class__(Hin) |
|
|
|
|
|
|
|
|
assert graphs_equal(G, Hin) |
|
|
|
|
|
def test_read_write(self): |
|
|
G = nx.MultiGraph() |
|
|
G.graph["name"] = "G" |
|
|
G.add_edge("1", "2", key="0") |
|
|
fh = StringIO() |
|
|
nx.nx_pydot.write_dot(G, fh) |
|
|
fh.seek(0) |
|
|
H = nx.nx_pydot.read_dot(fh) |
|
|
assert graphs_equal(G, H) |
|
|
|
|
|
|
|
|
def test_pydot_issue_7581(tmp_path): |
|
|
"""Validate that `nx_pydot.pydot_layout` handles nodes |
|
|
with characters like "\n", " ". |
|
|
|
|
|
Those characters cause `pydot` to escape and quote them on output, |
|
|
which caused #7581. |
|
|
""" |
|
|
G = nx.Graph() |
|
|
G.add_edges_from([("A\nbig test", "B"), ("A\nbig test", "C"), ("B", "C")]) |
|
|
|
|
|
graph_layout = nx.nx_pydot.pydot_layout(G, prog="dot") |
|
|
assert isinstance(graph_layout, dict) |
|
|
|
|
|
|
|
|
P = nx.nx_pydot.to_pydot(G) |
|
|
G2 = nx.Graph(nx.nx_pydot.from_pydot(P)) |
|
|
assert graphs_equal(G, G2) |
|
|
|
|
|
|
|
|
@pytest.mark.parametrize( |
|
|
"graph_type", [nx.Graph, nx.DiGraph, nx.MultiGraph, nx.MultiDiGraph] |
|
|
) |
|
|
def test_hashable_pydot(graph_type): |
|
|
|
|
|
G = graph_type() |
|
|
G.add_edge("5", frozenset([1]), t='"Example:A"', l=False) |
|
|
G.add_edge("1", 2, w=True, t=("node1",), l=frozenset(["node1"])) |
|
|
G.add_edge("node", (3, 3), w="string") |
|
|
|
|
|
assert [ |
|
|
{"t": '"Example:A"', "l": "False"}, |
|
|
{"w": "True", "t": "('node1',)", "l": "frozenset({'node1'})"}, |
|
|
{"w": "string"}, |
|
|
] == [ |
|
|
attr |
|
|
for _, _, attr in nx.nx_pydot.from_pydot(nx.nx_pydot.to_pydot(G)).edges.data() |
|
|
] |
|
|
|
|
|
assert {str(i) for i in G.nodes()} == set( |
|
|
nx.nx_pydot.from_pydot(nx.nx_pydot.to_pydot(G)).nodes |
|
|
) |
|
|
|
|
|
|
|
|
def test_pydot_numerical_name(): |
|
|
G = nx.Graph() |
|
|
G.add_edges_from([("A", "B"), (0, 1)]) |
|
|
graph_layout = nx.nx_pydot.pydot_layout(G, prog="dot") |
|
|
assert isinstance(graph_layout, dict) |
|
|
assert "0" not in graph_layout |
|
|
assert 0 in graph_layout |
|
|
assert "1" not in graph_layout |
|
|
assert 1 in graph_layout |
|
|
assert "A" in graph_layout |
|
|
assert "B" in graph_layout |
|
|
|