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

de.viadee.bpm.vPAV.processing.model.graph.Graph Maven / Gradle / Ivy

Go to download

The tool checks Camunda projects for consistency and discovers errors in process-driven applications. Called as a Maven plugin or JUnit test, it discovers esp. inconsistencies of a given BPMN model in the classpath and the sourcecode of an underlying java project, such as a delegate reference to a non-existing java class or a non-existing Spring bean.

There is a newer version: 3.0.8
Show newest version
/**
 * Copyright © 2017, viadee Unternehmensberatung GmbH
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *    This product includes software developed by the viadee Unternehmensberatung GmbH.
 * 4. Neither the name of the viadee Unternehmensberatung GmbH nor the
 *    names of its contributors may be used to endorse or promote products
 *    derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY  ''AS IS'' AND ANY
 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL  BE LIABLE FOR ANY
 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package de.viadee.bpm.vPAV.processing.model.graph;

/**
 * University of Washington, Computer Science & Engineering, Course 373, Winter 2011, Jessica Miller
 *
 * A class for a directed graph. Implemented by an adjacency list representation of a graph.
 */
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;

import de.viadee.bpm.vPAV.processing.model.data.Anomaly;
import de.viadee.bpm.vPAV.processing.model.data.AnomalyContainer;
import de.viadee.bpm.vPAV.processing.model.data.BpmnElement;
import de.viadee.bpm.vPAV.processing.model.data.InOutState;

public class Graph implements IGraph {

    private String processId;

    private Map> adjacencyListSucessor; // [vertices] -> [edge]

    private Map> adjacencyListPredecessor; // [vertices] -> [edge]

    private Map vertexInfo; // [vertex] -> [info]

    private Collection startNodes = new ArrayList();

    private Collection endNodes = new ArrayList();

    public Graph(final String processId) {
        this.processId = processId;
        this.adjacencyListSucessor = new HashMap>();
        this.adjacencyListPredecessor = new HashMap>();
        this.vertexInfo = new HashMap();
    }

    @Override
    public String getProcessId() {
        return processId;
    }

    @Override
    public void addStartNode(final BpmnElement node) {
        startNodes.add(node);
    }

    @Override
    public Collection getStartNodes() {
        return startNodes;
    }

    @Override
    public void addEndNode(final BpmnElement node) {
        endNodes.add(node);
    }

    @Override
    public Collection getEndNodes() {
        return endNodes;
    }

    @Override
    public void addVertex(BpmnElement v) {
        if (v == null) {
            throw new IllegalArgumentException("null");
        }

        adjacencyListSucessor.put(v, new ArrayList());
        adjacencyListPredecessor.put(v, new ArrayList());
        vertexInfo.put(v, new VertexInfo(v));
    }

    @Override
    public Collection getVertices() {
        return vertexInfo.keySet();
    }

    @Override
    public Collection> getEdges() {
        return adjacencyListSucessor.values();
    }

    @Override
    public void addEdge(BpmnElement from, BpmnElement to, int weight) {
        // add successor
        List edgeSucessorList = adjacencyListSucessor.get(from);
        if (edgeSucessorList == null) {
            throw new IllegalArgumentException("source vertex not in graph");
        }

        Edge newSucessorEdge = new Edge(from, to, weight);
        edgeSucessorList.add(newSucessorEdge);

        // add predecessor
        List edgePredecessorList = adjacencyListPredecessor.get(to);
        if (edgePredecessorList == null) {
            throw new IllegalArgumentException("source vertex not in graph");
        }

        Edge newPredecessorEdge = new Edge(to, from, weight);
        edgePredecessorList.add(newPredecessorEdge);
    }

    @Override
    public void removeEdge(BpmnElement from, BpmnElement to) {
        final List edgeSucessorList = adjacencyListSucessor.get(from);
        Edge foundEdge = null;
        for (final Edge e : edgeSucessorList) {
            if (e.from.toString().equals(from.toString()) && e.to.toString().equals(to.toString())) {
                // delete
                foundEdge = e;
            }
        }
        edgeSucessorList.remove(foundEdge);

        final List edgePredecessorList = adjacencyListPredecessor.get(to);
        foundEdge = null;
        for (final Edge e : edgePredecessorList) {
            if (e.to.toString().equals(from.toString()) && e.from.toString().equals(to.toString())) {
                // delete
                foundEdge = e;
            }
        }
        edgePredecessorList.remove(foundEdge);
    }

    @Override
    public boolean hasEdge(BpmnElement from, BpmnElement to) {
        return getEdge(from, to) != null;
    }

    @Override
    public Edge getEdge(BpmnElement from, BpmnElement to) {
        List edgeList = adjacencyListSucessor.get(from);
        if (edgeList == null) {
            throw new IllegalArgumentException("source vertex not in graph");
        }

        for (Edge e : edgeList) {
            if (e.to.equals(to)) {
                return e;
            }
        }

        return null;
    }

    /**
     * set anomaly information on data flow graph
     *
     */
    @Override
    public void setAnomalyInformation(final BpmnElement source) {
        setAnomalyInformationRecursive(source, new LinkedList());
    }

    /**
     * set anomaly information recursive on data flow graph (forward)
     *
     * @param startNode
     * @param currentPath
     */
    private void setAnomalyInformationRecursive(final BpmnElement startNode,
            final LinkedList currentPath) {

        currentPath.add(startNode);

        final boolean isGateway = startNode.getBaseElement().getElementType().getBaseType()
                .getTypeName().equals("gateway");
        final boolean isNodeParallelGateway = startNode.getBaseElement().getElementType().getTypeName()
                .equals("parallelGateway");
        final boolean isEndEvent = startNode.getBaseElement().getElementType().getTypeName()
                .equals("endEvent")
                && startNode.getBaseElement().getParentElement().getElementType().getTypeName()
                        .equals("process");

        final List predecessorEdges = this.adjacencyListPredecessor.get(startNode);
        Map outSuccessors = new HashMap();
        if (predecessorEdges != null) {
            for (final Edge t : predecessorEdges) {
                if (isGateway) {
                    if (isNodeParallelGateway) {
                        // If the node is a parallel gateway, take all predecessor variables.
                        // If variables are identical, take the variable with the following precedence
                        // 1) DELETED
                        // 2) READ
                        // 3) DEFINED
                        if (outSuccessors.isEmpty()) {
                            outSuccessors.putAll(t.to.getOut());
                        } else {
                            outSuccessors.putAll(unionWithStatePrecedence(outSuccessors, t.to.getOut()));
                        }
                    } else {
                        // If the node is an other gateway, take the intersection of all predecessor variables.
                        // Follow the precedence rule (look above)
                        if (outSuccessors.isEmpty()) {
                            outSuccessors.putAll(t.to.getOut());
                        } else {
                            outSuccessors.putAll(intersection(outSuccessors, t.to.getOut()));
                        }
                    }
                } else {
                    outSuccessors.putAll(t.to.getOut());
                }
            }
        }

        startNode.setIn(outSuccessors);
        if (!isEndEvent) {
            // end element has not an out set
            startNode.setOut();
        }

        if (startNode.getBaseElement() != null) {
            // save the path, if the the search has reached the begin of the process
            if (isEndEvent) {
                currentPath.remove(startNode);
                return;
            }
        }

        final List edges = this.adjacencyListSucessor.get(startNode);

        for (final Edge t : edges) {
            int occurrences = Collections.frequency(currentPath, t.to);
            if (occurrences < 2) { // case iterations n=1 and n=2 for loops
                setAnomalyInformationRecursive(t.to, currentPath);
            }
        }

        currentPath.remove(startNode);
    }

    /**
     * get nodes with data flow anomalies
     */
    @Override
    public Map> getNodesWithAnomalies() {

        final Map> anomalies = new HashMap>();
        for (final BpmnElement node : adjacencyListSucessor.keySet()) {
            anomalies.putAll(node.getAnomalies());
        }
        return anomalies;
    }

    /**
     * search all paths with variables, which has not been set
     *
     * source: http://codereview.stackexchange.com/questions/45678/find-all-paths-from-source-to-destination
     */
    @Override
    public List getAllInvalidPaths(final BpmnElement source, final AnomalyContainer anomaly) {
        final List paths = getAllInvalidPathsRecursive(source, anomaly,
                new LinkedList());
        return paths;
    }

    /**
     * search all paths with variables, which has not been set (backward)
     *
     * source: http://codereview.stackexchange.com/questions/45678/find-all-paths-from-source-to-destination
     *
     * @param startNode
     * @param varName
     * @param currentPath
     * @param maxSize
     * @return paths
     */
    private List getAllInvalidPathsRecursive(final BpmnElement startNode,
            final AnomalyContainer anomaly, final LinkedList currentPath) {

        final List invalidPaths = new ArrayList();

        currentPath.add(startNode);

        final List edges = this.adjacencyListPredecessor.get(startNode);

        final Map in = startNode.getIn();
        final Map out = startNode.getOut();

        final List returnPathsUrAnomaly = exitConditionUrAnomaly(startNode, anomaly, currentPath,
                invalidPaths, in, out);
        final List returnPathsDdDuAnomaly = exitConditionDdDuAnomaly(startNode, anomaly,
                currentPath, invalidPaths, in);

        if (anomaly.getAnomaly() == Anomaly.UR && !in.containsKey(anomaly.getName())
                && out.containsKey(anomaly.getName())) {
            return invalidPaths;
        } else if (returnPathsUrAnomaly != null) {
            return returnPathsUrAnomaly;
        } else if (returnPathsDdDuAnomaly != null) {
            return returnPathsDdDuAnomaly;
        }

        for (final Edge t : edges) {
            if (!currentPath.contains(t.to) || t.to == anomaly.getVariable().getElement()) {
                invalidPaths.addAll(getAllInvalidPathsRecursive(t.to, anomaly, currentPath));
            }
        }

        currentPath.remove(startNode);

        return invalidPaths;
    }

    /**
     * exit condition for path finding (ur anomaly)
     *
     * @param startNode
     * @param anomaly
     * @param currentPath
     * @param invalidPaths
     * @param in
     * @param out
     */
    private List exitConditionUrAnomaly(final BpmnElement startNode,
            final AnomalyContainer anomaly, final LinkedList currentPath,
            final List invalidPaths, final Map in,
            final Map out) {

        // go back to the node, where the variable was deleted
        // or go back to the start
        if (anomaly.getAnomaly() == Anomaly.UR && (variableDeleted(anomaly, in, out)
                || ((startNode.getBaseElement().getElementType().getTypeName().equals("startEvent")
                        && startNode.getBaseElement().getParentElement().getElementType().getTypeName()
                                .equals("process"))))) {

            final List newPath = new ArrayList(currentPath);
            invalidPaths.add(new Path(newPath));

            currentPath.remove(startNode);
            return invalidPaths;
        }
        return null;
    }

    /**
     * is variable deleted
     *
     * @param anomaly
     * @param in
     * @param out
     * @return
     */
    private boolean variableDeleted(final AnomalyContainer anomaly, final Map in,
            final Map out) {

        return ((in.containsKey(anomaly.getName()) && in.get(anomaly.getName()) != InOutState.DELETED))
                && (out.containsKey(anomaly.getName()) && out.get(anomaly.getName()) == InOutState.DELETED);
    }

    /**
     * exit condition for path finding (du / dd anomaly)
     *
     * @param startNode
     * @param anomaly
     * @param currentPath
     * @param invalidPaths
     */
    private List exitConditionDdDuAnomaly(final BpmnElement startNode,
            final AnomalyContainer anomaly, final LinkedList currentPath,
            final List invalidPaths, Map in) {

        // go back to the node where the element is defined
        // skip the startpoint
        if (startNode.defined().containsKey(anomaly.getName())
                && (anomaly.getAnomaly() == Anomaly.DD || anomaly.getAnomaly() == Anomaly.DU)
                && currentPath.size() > 1) {
            final List newPath = new ArrayList(currentPath);
            invalidPaths.add(new Path(newPath));

            currentPath.remove(startNode);
            return invalidPaths;
        }
        return null;
    }

    @Override
    public String toString() {
        Set keys = adjacencyListSucessor.keySet();
        String str = "digraph G {\n";

        for (BpmnElement v : keys) {
            str += " ";

            List edgeList = adjacencyListSucessor.get(v);

            for (Edge edge : edgeList) {
                str += edge;
                str += "\n";
            }
        }
        str += "}";
        return str;
    }

    protected final void clearVertexInfo() {
        for (VertexInfo info : this.vertexInfo.values()) {
            info.clear();
        }
    }

    /**
     * generate intersection for variable maps and remind precedence rule for variable states
     *
     * @param mapA
     *            mapA
     * @param mapB
     *            mapB
     * @return intersection
     */
    public static Map intersection(final Map mapA,
            final Map mapB) {
        final Map intersectionMap = new HashMap();
        final Set variables = new HashSet();
        variables.addAll(mapA.keySet());
        variables.addAll(mapB.keySet());
        for (final String varName : variables) {
            if (mapA.containsKey(varName) && mapB.containsKey(varName)) {
                final InOutState state1 = mapA.get(varName);
                final InOutState state2 = mapB.get(varName);

                final InOutState intersectionElement = getStatePrecedence(state1, state2);
                intersectionMap.put(varName, intersectionElement);
            } else {
                mapA.remove(varName);
                mapB.remove(varName);
            }
        }
        return intersectionMap;
    }

    /**
     * get union and remind precedence rule for variable states
     *
     * @param mapA
     *            mapA
     * @param mapB
     *            mapB
     * @return union
     */
    public static Map unionWithStatePrecedence(final Map mapA,
            final Map mapB) {

        final Map unionMap = new HashMap();
        unionMap.putAll(mapA);
        unionMap.putAll(mapB);
        unionMap.putAll(intersection(mapA, mapB));

        return unionMap;

    }

    /**
     * precedence rule for variable states
     *
     * 1) delete 2) read 3) define
     *
     * @param element1
     * @param element2
     * @return
     */
    private static InOutState getStatePrecedence(final InOutState state1, final InOutState state2) {
        if (state1 == InOutState.DELETED || state2 == InOutState.DELETED
                || (state1 == InOutState.DELETED && state2 == InOutState.DELETED)) {
            return InOutState.DELETED;
        } else if (state1 == InOutState.READ || state2 == InOutState.READ
                || (state1 == InOutState.READ && state2 == InOutState.READ)) {
            return InOutState.READ;
        }
        return InOutState.DEFINED;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy