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

org.parboiled.trees.GraphUtils Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2009-2011 Mathias Doenitz
 *
 * 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 org.parboiled.trees;

import org.parboiled.common.Formatter;
import org.parboiled.common.Preconditions;
import org.parboiled.common.Predicate;
import org.parboiled.common.Predicates;

import java.util.Collection;
import java.util.HashSet;

/**
 * General utility methods for operating on directed graphs (consisting of {@link GraphNode}s).
 */
public final class GraphUtils {

    private GraphUtils() {}

    /**
     * Returns true if this node is not null and has at least one child node.
     *
     * @param node a node
     * @return true if this node is not null and has at least one child node.
     */
    public static boolean hasChildren(GraphNode node) {
        return node != null && !node.getChildren().isEmpty();
    }

    /**
     * Returns the first child node of the given node or null if node is null or does not have any children.
     *
     * @param node a node
     * @return the first child node of the given node or null if node is null or does not have any children
     */
    public static > T getFirstChild(T node) {
        return hasChildren(node) ? node.getChildren().get(0) : null;
    }

    /**
     * Returns the last child node of the given node or null if node is null or does not have any children.
     *
     * @param node a node
     * @return the last child node of the given node or null if node is null or does not have any children
     */
    public static > T getLastChild(T node) {
        return hasChildren(node) ? node.getChildren().get(node.getChildren().size() - 1) : null;
    }

    /**
     * Counts all distinct nodes in the graph reachable from the given node.
     * This method can properly deal with cycles in the graph.
     *
     * @param node the root node
     * @return the number of distinct nodes
     */
    public static > int countAllDistinct(T node) {
        if (node == null) return 0;
        return collectAllNodes(node, new HashSet()).size();
    }

    /**
     * Collects all nodes from the graph reachable from the given node in the given collection.
     * This method can properly deal with cycles in the graph.
     *
     * @param node       the root node
     * @param collection the collection to collect into
     * @return the same collection passed as a parameter
     */
    public static , C extends Collection> C collectAllNodes(T node, C collection) {
        // we don't recurse if the collecion already contains the node
        // this costs a bit of performance but prevents infinite recursion in the case of graph cycles
        Preconditions.checkArgNotNull(collection, "collection");
        if (node != null && !collection.contains(node)) {
            collection.add(node);
            for (T child : node.getChildren()) {
                collectAllNodes(child, collection);
            }
        }
        return collection;
    }

    /**
     * Creates a string representation of the graph reachable from the given node using the given formatter.
     *
     * @param node      the root node
     * @param formatter the node formatter
     * @return a new string
     */
    public static > String printTree(T node, Formatter formatter) {
        Preconditions.checkArgNotNull(formatter, "formatter");
        return printTree(node, formatter, Predicates.alwaysTrue(), Predicates.alwaysTrue());
    }

    /**
     * Creates a string representation of the graph reachable from the given node using the given formatter.
     * The given filter predicated determines whether a particular node (and its subtree respectively) is to be
     * printed or not.
     *
     * @param node          the root node
     * @param formatter     the node formatter
     * @param nodeFilter    the predicate selecting the nodes to print
     * @param subTreeFilter the predicate determining whether to descend into a given nodes subtree or not
     * @return a new string
     */
    public static > String printTree(T node, Formatter formatter,
                                                            Predicate nodeFilter,
                                                            Predicate subTreeFilter) {
        Preconditions.checkArgNotNull(formatter, "formatter");
        Preconditions.checkArgNotNull(nodeFilter, "nodeFilter");
        Preconditions.checkArgNotNull(subTreeFilter, "subTreeFilter");
        return node == null ? "" :
                printTree(node, formatter, "", new StringBuilder(), nodeFilter, subTreeFilter).toString();
    }

    // private recursion helper

    private static > StringBuilder printTree(T node, Formatter formatter,
                                                                    String indent, StringBuilder sb,
                                                                    Predicate nodeFilter,
                                                                    Predicate subTreeFilter) {
        if (nodeFilter.apply(node)) {
            String line = formatter.format(node);
            if (line != null) {
                sb.append(indent).append(line).append("\n");
                indent += "  ";
            }
        }
        if (subTreeFilter.apply(node)) {
            for (T sub : node.getChildren()) {
                printTree(sub, formatter, indent, sb, nodeFilter, subTreeFilter);
            }
        }
        return sb;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy