GraphViz - Как подключить подграфы?

В языке DOT для GraphViz я пытаюсь представить диаграмму зависимостей. Мне нужно иметь узлы внутри контейнера и иметь возможность создавать узлы и/или контейнеры в зависимости от других узлов и/или контейнеров.

Я использую subgraph для представления своих контейнеров. Node работает очень хорошо, но я не могу понять, как подключить подграфы.

Учитывая приведенную ниже программу, мне нужно подключить cluster_1 и cluster_2 со стрелкой, но все, что я пробовал, создает новые узлы вместо соединения кластеров:

digraph G {

    graph [fontsize=10 fontname="Verdana"];
    node [shape=record fontsize=10 fontname="Verdana"];

    subgraph cluster_0 {
        node [style=filled];
        "Item 1" "Item 2";
        label = "Container A";
        color=blue;
    }

    subgraph cluster_1 {
        node [style=filled];
        "Item 3" "Item 4";
        label = "Container B";
        color=blue;
    }

    subgraph cluster_2 {
        node [style=filled];
        "Item 5" "Item 6";
        label = "Container C";
        color=blue;
    }

    // Renders fine
    "Item 1" -> "Item 2";
    "Item 2" -> "Item 3";

    // Both of these create new nodes
    cluster_1 -> cluster_2;
    "Container A" -> "Container C";
}

enter image description here

Ответ 1

Руководство пользователя DOT дает следующий пример графика с кластерами с краями между кластерами

digraph G {
  compound=true;
  subgraph cluster0 {
    a -> b;
    a -> c;
    b -> d;
    c -> d;
  }
  subgraph cluster1 {
    e -> g;
    e -> f;
  }
  b -> f [lhead=cluster1];
  d -> e;
  c -> g [ltail=cluster0,lhead=cluster1];
  c -> e [ltail=cluster0];
  d -> h;
}

и ребра между узлами и кластерами.

enter image description here

Ответ 2

Для простоты справки решение, описанное в ответе HighPerformanceMark, применяемое непосредственно к исходному вопросу, выглядит следующим образом:

digraph G {

    graph [fontsize=10 fontname="Verdana" compound=true];
    node [shape=record fontsize=10 fontname="Verdana"];

    subgraph cluster_0 {
        node [style=filled];
        "Item 1" "Item 2";
        label = "Container A";
        color=blue;
    }

    subgraph cluster_1 {
        node [style=filled];
        "Item 3" "Item 4";
        label = "Container B";
        color=blue;
    }

    subgraph cluster_2 {
        node [style=filled];
        "Item 5" "Item 6";
        label = "Container C";
        color=blue;
    }

    // Edges between nodes render fine
    "Item 1" -> "Item 2";
    "Item 2" -> "Item 3";

    // Edges that directly connect one cluster to another
    "Item 1" -> "Item 3" [ltail=cluster_0 lhead=cluster_1];
    "Item 1" -> "Item 5" [ltail=cluster_0 lhead=cluster_2];
}

"Состав = истина" в декларации "graph" имеет жизненно важное значение. Это дает результат:

graph with connected clusters

Обратите внимание, что я сменил ребра на опорные узлы в кластере, добавил атрибуты stail и lhead к каждому ребру, указав имя кластера и добавив атрибут уровня графа 'compound = true'.

Что касается беспокойства о том, что можно захотеть подключить кластер без узлов внутри него, мое решение состояло в том, чтобы всегда добавлять node в каждый кластер, созданный с помощью стиля = plaintext. Используйте этот node для обозначения кластера (вместо встроенного в кластер атрибута метки, который должен быть установлен в пустую строку (в Python, label='""'). Это означает, что я больше не добавляю ребра, которые соединяются кластеров напрямую, но он работает в моей конкретной ситуации.

Ответ 3

Убедитесь, что вы используете макет fdp для файла. Я не думаю, что neato поддерживает кластеры.