All Downloads are FREE. Search and download functionalities are using the official Maven repository.

guru.nidi.graphviz.model.SerializerImpl Maven / Gradle / Ivy

There is a newer version: 0.18.1
Show newest version
/*
 * Copyright © 2015 Stefan Niederhauser ([email protected])
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package guru.nidi.graphviz.model;

import guru.nidi.graphviz.attribute.*;
import guru.nidi.graphviz.attribute.validate.AttributeValidator;
import guru.nidi.graphviz.attribute.validate.ValidatorMessage;

import java.util.*;
import java.util.Map.Entry;
import java.util.function.Consumer;
import java.util.stream.Stream;

import static guru.nidi.graphviz.attribute.validate.AttributeValidator.Scope;

class SerializerImpl {
    private final MutableGraph graph;
    private final StringBuilder str;
    private final AttributeValidator validator;
    private final Consumer messageConsumer;

    SerializerImpl(MutableGraph graph, AttributeValidator validator, Consumer messageConsumer) {
        this.graph = graph;
        this.validator = validator;
        this.messageConsumer = messageConsumer;
        str = new StringBuilder();
    }

    String serialize() {
        toplevelGraph(graph);
        return str.toString();
    }

    private void toplevelGraph(MutableGraph graph) {
        final boolean useDir = hasDifferentlyDirectedSubgraphs(graph);
        str.append(graph.strict ? "strict " : "").append(graph.directed || useDir ? "digraph " : "graph ");
        if (!graph.name.isContentEmpty()) {
            str.append(graph.name.serialized()).append(' ');
        }
        doGraph(graph, useDir, Scope.GRAPH);
    }

    private void subGraph(MutableGraph graph, boolean useDir) {
        if (!graph.name.isContentEmpty() || graph.cluster) {
            str.append("subgraph ")
                    .append((graph.cluster ? Label.of("cluster_" + graph.name) : graph.name).serialized())
                    .append(' ');
        }
        doGraph(graph, useDir, graph.cluster ? Scope.CLUSTER : Scope.SUB_GRAPH);
    }

    private void doGraph(MutableGraph graph, boolean useDir, Scope scope) {
        str.append("{\n");
        if (useDir && graph.graphAttrs.get("dir") == null) {
            attributes("edge", Attributes.attr("dir", graph.directed ? "forward" : "none"), Scope.EDGE, "");
        }
        graphAttrs(graph, scope);

        final List nodes = new ArrayList<>();
        final List graphs = new ArrayList<>();
        final Collection linkSources = linkedNodes(graph.nodes);
        linkSources.addAll(linkedNodes(graph.subgraphs));
        for (final LinkSource linkSource : linkSources) {
            if (linkSource instanceof MutableNode) {
                final MutableNode node = (MutableNode) linkSource;
                final int i = indexOfName(nodes, node.name);
                if (i < 0) {
                    nodes.add(node);
                } else {
                    nodes.set(i, node.copy().merge(nodes.get(i)));
                }
            } else {
                graphs.add((MutableGraph) linkSource);
            }
        }

        nodes(graph, nodes);
        graphs(graphs, nodes, useDir);

        edges(nodes, useDir);
        edges(graphs, useDir);
        str.append('}');
    }

    private boolean hasDifferentlyDirectedSubgraphs(MutableGraph graph) {
        return Stream.concat(linkedNodes(graph.nodes).stream(), linkedNodes(graph.subgraphs).stream())
                .filter(n -> n instanceof MutableGraph)
                .map(n -> (MutableGraph) n)
                .anyMatch(sub -> sub.directed != graph.directed);
    }

    private void graphAttrs(MutableGraph graph, Scope scope) {
        attributes("graph", graph.graphAttrs, scope, "Graph attrs of '" + graph.name.toString() + "'");
        attributes("node", graph.nodeAttrs, Scope.NODE, "Node attrs of '" + graph.name.toString() + "'");
        attributes("edge", graph.linkAttrs, Scope.EDGE, "Edge attrs of '" + graph.name.toString() + "'");
    }

    private int indexOfName(List nodes, Label name) {
        for (int i = 0; i < nodes.size(); i++) {
            if (nodes.get(i).name.equals(name)) {
                return i;
            }
        }
        return -1;
    }

    private void attributes(String name, Attributes attributed, Scope scope, String pos) {
        if (!attributed.isEmpty()) {
            str.append(name);
            attrs(attributed, scope, pos);
            str.append('\n');
        }
    }

    private Collection linkedNodes(Collection nodes) {
        final Set visited = new LinkedHashSet<>();
        for (final LinkSource node : nodes) {
            linkedNodes(node, visited);
        }
        return visited;
    }

    private void linkedNodes(LinkSource linkSource, Set visited) {
        if (!visited.contains(linkSource)) {
            visited.add(linkSource);
            for (final Link link : linkSource.links()) {
                linkedNodes(link.to.asLinkSource(), visited);
            }
        }
    }

    private void nodes(MutableGraph graph, List nodes) {
        for (final MutableNode node : nodes) {
            if (!node.attributes.isEmpty()
                    || (graph.nodes.contains(node) && node.links.isEmpty() && !isLinked(node, nodes))) {
                node(node);
                str.append('\n');
            }
        }
    }

    private void node(MutableNode node) {
        str.append(node.name.serialized());
        attrs(node.attributes, Scope.NODE, "Node '" + node.name.toString() + "'");
    }

    private boolean isLinked(MutableNode node, List nodes) {
        for (final MutableNode m : nodes) {
            for (final Link link : m.links) {
                if (isNode(link.to, node)) {
                    return true;
                }
            }
        }
        return false;
    }

    private boolean isLinked(MutableGraph graph, List linkSources) {
        for (final LinkSource linkSource : linkSources) {
            for (final Link link : linkSource.links()) {
                if (link.to.equals(graph)) {
                    return true;
                }
            }
        }
        return false;
    }

    private boolean isNode(LinkTarget target, MutableNode node) {
        return target == node || (target instanceof ImmutablePortNode && ((ImmutablePortNode) target).node() == node);
    }

    private void graphs(List graphs, List nodes, boolean useDir) {
        for (final MutableGraph graph : graphs) {
            if (graph.links.isEmpty() && !isLinked(graph, nodes) && !isLinked(graph, graphs)) {
                subGraph(graph, useDir);
                str.append('\n');
            }
        }
    }

    private void edges(List linkSources, boolean useDir) {
        for (final LinkSource linkSource : linkSources) {
            for (final Link link : linkSource.links()) {
                linkTarget(link.from, useDir);
                str.append(graph.directed || useDir ? " -> " : " -- ");
                linkTarget(link.to, useDir);
                attrs(link.attributes, Scope.EDGE,
                        "Edge '" + link.from.name().toString() + "--" + link.to.name().toString() + "'");
                str.append('\n');
            }
        }
    }

    private void linkTarget(Object linkable, boolean useDir) {
        if (linkable instanceof MutableNode) {
            str.append(((MutableNode) linkable).name.serialized());
        } else if (linkable instanceof ImmutablePortNode) {
            port((ImmutablePortNode) linkable);
        } else if (linkable instanceof MutableGraph) {
            subGraph((MutableGraph) linkable, useDir);
        } else {
            throw new IllegalStateException("unexpected link target " + linkable);
        }
    }

    private void port(ImmutablePortNode portNode) {
        str.append(portNode.name().serialized());
        final String record = portNode.port().record();
        if (record != null) {
            str.append(':').append(SimpleLabel.of(record).serialized());
        }
        final Compass compass = portNode.port().compass();
        if (compass != null) {
            str.append(':').append(compass.value);
        }
    }

    private void attrs(Attributes attrs, Scope scope, String pos) {
        if (!attrs.isEmpty()) {
            str.append(" [");
            boolean first = true;
            for (final Entry attr : attrs) {
                if (attr.getValue() != null) {
                    if (first) {
                        first = false;
                    } else {
                        str.append(',');
                    }
                    attr(attr.getKey(), attr.getValue(), scope, pos);
                }
            }
            str.append(']');
        }
    }

    private void attr(String key, Object value, Scope scope, String pos) {
        str.append(SimpleLabel.of(key).serialized())
                .append('=')
                .append(SimpleLabel.of(value).serialized());
        validate(key, value, scope, pos);
    }

    private void validate(String key, Object value, Scope scope, String pos) {
        validator.validate(key, value, scope).forEach(msg -> messageConsumer.accept(msg.at(pos)));
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy