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

net.spals.appbuilder.graph.writer.TextServiceGraphWriterPlugin Maven / Gradle / Ivy

package net.spals.appbuilder.graph.writer;

import net.spals.appbuilder.graph.model.IServiceTreeVertex;
import net.spals.appbuilder.graph.model.ServiceTree;
import org.jgrapht.event.TraversalListenerAdapter;
import org.jgrapht.event.VertexTraversalEvent;
import org.jgrapht.graph.DefaultEdge;
import org.jgrapht.traverse.DepthFirstIterator;

import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @author tkral
 */
class TextServiceGraphWriterPlugin implements ServiceGraphWriterPlugin, ServiceTree> {

    @Override
    public String writeServiceGraph(final ServiceTree serviceTree) {
        final StringBuilder textGraph = new StringBuilder().append(String.format("%n"));
        final DepthFirstIterator, DefaultEdge> dfs =
            new DepthFirstIterator<>(serviceTree, serviceTree.getRoot());

        // Walk the graph depth-first and use the service tree listener
        // to create a text representation of the service tree
        final TextServiceTreeListener listener = new TextServiceTreeListener(serviceTree, textGraph);
        dfs.addTraversalListener(listener);
        while (dfs.hasNext()) {
            dfs.next();
        }

        return textGraph.toString();
    }

    /**
     * A {@link org.jgrapht.event.TraversalListener} which generates a text
     * tree from a tree traversal.
     *
     * This is shamelessly copied from the maven-dependency-plugin:tree target.
     *
     * @author tkral
     */
    static class TextServiceTreeListener extends TraversalListenerAdapter, DefaultEdge> {

        private static final String VERTEX_INDENT = "+- ";
        private static final String LAST_VERTEX_INDENT = "\\- ";
        private static final String FILL_INDENT = "|  ";
        private static final String LAST_FILL_INDENT = "   ";

        private final ServiceTree serviceTree;
        private final StringBuilder textGraph;

        private final AtomicInteger graphDepth = new AtomicInteger(0);
        private final Set> visitedVertices = new HashSet<>();
        TextServiceTreeListener(final ServiceTree serviceTree,
                                final StringBuilder textGraph) {
            this.serviceTree = serviceTree;
            this.textGraph = textGraph;
        }

        @Override
        public void vertexTraversed(final VertexTraversalEvent> e) {
            visitedVertices.add(e.getVertex());

            indent(e.getVertex());
            textGraph.append(e.getVertex()).append(String.format("%n"));

            graphDepth.incrementAndGet();
        }

        @Override
        public void vertexFinished(final VertexTraversalEvent> e) {
            graphDepth.decrementAndGet();
        }

        private void indent(final IServiceTreeVertex vertex) {
            for (int i = 1; i < graphDepth.get(); i++) {
                textGraph.append(isLast(vertex, i) ? LAST_FILL_INDENT : FILL_INDENT);
            }

            if (graphDepth.get() > 0) {
                textGraph.append(isLast(vertex) ? LAST_VERTEX_INDENT : VERTEX_INDENT);
            }
        }

        private boolean isLast(final IServiceTreeVertex vertex) {
            if (vertex.isRoot()) {
                return true;
            }

            final Set> siblings = serviceTree.getSiblings(vertex);
            return visitedVertices.containsAll(siblings);
        }

        private boolean isLast(final IServiceTreeVertex vertex, final int ancestorDepth) {
            IServiceTreeVertex depthVertex = vertex;
            int distance = graphDepth.get() - ancestorDepth;

            while (distance-- > 0) {
                depthVertex = depthVertex.getParent().get();
            }

            return isLast(depthVertex);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy