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

org.jbpm.simulation.impl.JBPMBAMSimulationDataProvider Maven / Gradle / Ivy

There is a newer version: 7.74.1.Final
Show newest version
/*
 * Copyright 2015 Red Hat, Inc. and/or its affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * 
 *      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.jbpm.simulation.impl;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import javax.naming.InitialContext;
import javax.sql.DataSource;

import org.apache.commons.math3.stat.StatUtils;
import org.jbpm.simulation.NodeStatistic;
import org.jbpm.simulation.SimulationDataProvider;
import org.kie.api.definition.process.Node;

public class JBPMBAMSimulationDataProvider implements SimulationDataProvider {

    private DataSource bamDataSource;
    private boolean processLoaded = false;
    private Map> processStatistics = new HashMap>();
    
    private String processId;
    
    private static final String GET_PROCESS_INFO_QUERY = 
            "select nodeid, count(processinstanceid), min(log_date), max(log_date) from nodeinstancelog where processid = ? and type = ? group by nodeid;";
    
    private static final String PROCESS_INSTANCE_COUNT_QUERY = 
            "select count(processinstanceid) from processinstancelog where processid = ?;";
    
    private static final String PROCESS_INSTANCE_COUNT_FOR_PATH_QUERY = 
            "select count(processinstanceid) from nodeinstancelog where processid = ? and nodeid in (@1) group by processinstanceid;";
    
    public JBPMBAMSimulationDataProvider(String bamDataSource, String processId) {
        try {
            InitialContext ctx = new InitialContext();
            this.bamDataSource = (DataSource) ctx.lookup(bamDataSource);
        
        } catch (Exception e) {
            throw new IllegalStateException("Unable to get data source: " + bamDataSource, e);
        }
        this.processId = processId;
    }

    public JBPMBAMSimulationDataProvider(DataSource bamDataSource, String processId) {
        this.bamDataSource = bamDataSource;
        this.processId = processId;
    }


    public Map getSimulationDataForNode(Node node) {
        
        if (!processLoaded) {
            loadProcessInfo(processId);
        }
        
        
        return processStatistics.get(Long.toString(node.getId()));
    } 
    

    protected void loadProcessInfo(String processId) {
        Connection connection = null;
        PreparedStatement pstmt = null;
        ResultSet rs = null;
        
        Map enterStats = new HashMap();
        Map exitStats = new HashMap();
        
        try {
            connection = bamDataSource.getConnection();
            
            // collect information about node enter events
            pstmt = connection.prepareStatement(GET_PROCESS_INFO_QUERY);
            pstmt.setString(1, processId);
            pstmt.setInt(2, 0);
            
            rs = pstmt.executeQuery();

            while (rs.next()) {
                String nodeIdbam = rs.getString(1);
                Long numberOfProcessInstances = rs.getLong(2);
                Timestamp minTs = rs.getTimestamp(3);
                Timestamp maxTs = rs.getTimestamp(4);
                

                enterStats.put(nodeIdbam, new NodeStatistic(nodeIdbam, minTs.getTime(), maxTs.getTime(), numberOfProcessInstances));
            }
            
            
            // collect information about node exit events
            pstmt = connection.prepareStatement(GET_PROCESS_INFO_QUERY);
            pstmt.setString(1, processId);
            pstmt.setInt(2, 1);
            
            rs = pstmt.executeQuery();

            while (rs.next()) {
                String nodeIdbam = rs.getString(1);
                Long numberOfProcessInstances = rs.getLong(2);
                Timestamp minTs = rs.getTimestamp(3);
                Timestamp maxTs = rs.getTimestamp(4);
                
                exitStats.put(nodeIdbam, new NodeStatistic(nodeIdbam, minTs.getTime(), maxTs.getTime(), numberOfProcessInstances));
            }
        } catch (SQLException e) {
            e.printStackTrace();
        } finally {
            if (rs != null) {
                try {
                    rs.close();
                } catch (SQLException e) {

                }
            }
            
            if (pstmt != null) {
                try {
                    pstmt.close();
                } catch (SQLException e) {

                }
            }
            
            if (connection != null) {
                try {
                    connection.close();
                } catch (SQLException e) {

                }
            }
        }
        
        // process db information and construct single map with data
        
        Iterator nodes = enterStats.keySet().iterator();
        while (nodes.hasNext()) {
            Map nodeProperties = new HashMap();
            String node = (String) nodes.next();
            NodeStatistic enterStat = enterStats.get(node);
            NodeStatistic exitStat = exitStats.remove(node);
            if (exitStat != null) {
                long minExec = exitStat.getMinTimeStamp() - enterStat.getMinTimeStamp();
                long maxExec = exitStat.getMaxTimeStamp() - enterStat.getMaxTimeStamp();
            
                nodeProperties.put("duration", new Double(StatUtils.mean(new double[]{minExec, maxExec})).longValue());
                nodeProperties.put("max-exec", maxExec);
                nodeProperties.put("min-exec", minExec);
                nodeProperties.put("range", (maxExec - minExec)/2);
                nodeProperties.put("numberOfInstance", enterStat.getInstances());
            } else {
                nodeProperties.put("duration", 0);
                nodeProperties.put("max-exec", 0);
                nodeProperties.put("min-exec", 0);
                nodeProperties.put("range", 0);
                nodeProperties.put("numberOfInstance", enterStat.getInstances());
            }
            this.processStatistics.put(node, nodeProperties);
        }
        
        if (!exitStats.isEmpty()) {
            nodes = exitStats.keySet().iterator();
            while (nodes.hasNext()) {
                Map nodeProperties = new HashMap();
                String node = (String) nodes.next();
                NodeStatistic exitStat = exitStats.get(node);

                nodeProperties.put("duration", 0);
                nodeProperties.put("max-exec", 0);
                nodeProperties.put("min-exec", 0);
                nodeProperties.put("range", 0);
                nodeProperties.put("numberOfInstance", exitStat.getInstances());
                
                this.processStatistics.put(node, nodeProperties);
            }
            
        }
        this.processLoaded = true;
    }
    
    /*
     * Experimental  - not really working yet....
     */
    public double calculatePathProbability(SimulationPath path) {
        
        Connection connection = null;
        PreparedStatement pstmt = null;
        ResultSet rs = null;
        
        try {
            
            connection = bamDataSource.getConnection();
            
            // collect information about node enter events
            pstmt = connection.prepareStatement(PROCESS_INSTANCE_COUNT_QUERY);
            pstmt.setString(1, processId);
            rs = pstmt.executeQuery();
            Integer instanceCount = 1;
            if (rs.next()) {
                instanceCount = rs.getInt(1);
            }
            
            String parameters = buildParameterPlaceHolder(path.getActivityIds().size());
            String query = PROCESS_INSTANCE_COUNT_FOR_PATH_QUERY.replaceFirst("@1", parameters);
            pstmt = connection.prepareStatement(query);
            pstmt.setString(1, processId);
            int parameterIndex = 2;
            
            for (String node : path.getActivityIds()) {
                pstmt.setString(parameterIndex, node.replaceFirst("_", ""));
                parameterIndex++;
            }
            
            rs = pstmt.executeQuery();
            Integer pathcount = 0;
            if (rs.next()) {
                pathcount = rs.getInt(1)/2;
            }
            
            double result = (100 * pathcount) / instanceCount;
            
            path.setProbability(result);
            
            return result;
            
        } catch (SQLException e) {
            e.printStackTrace();
        } finally {
            if (rs != null) {
                try {
                    rs.close();
                } catch (SQLException e) {

                }
            }
            
            if (pstmt != null) {
                try {
                    pstmt.close();
                } catch (SQLException e) {

                }
            }
            
            if (connection != null) {
                try {
                    connection.close();
                } catch (SQLException e) {

                }
            }
        }
        
        return -1;
        
    }

    public Map getProcessDataForNode(Node node) {
        return null;
    }

    private String buildParameterPlaceHolder(int size) {
        StringBuffer result = new StringBuffer();
        for (int i = 0; i < size; i++) {
            result.append("?,");
        }
        // remove last comma
        result.deleteCharAt(result.length()-1);
        return result.toString();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy