1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.rice.kew.engine.node;
17
18 import org.kuali.rice.core.framework.persistence.jpa.OrmUtils;
19 import org.kuali.rice.kew.api.WorkflowRuntimeException;
20 import org.kuali.rice.kew.engine.RouteContext;
21 import org.kuali.rice.kew.service.KEWServiceLocator;
22
23 import java.util.HashSet;
24 import java.util.Iterator;
25 import java.util.Set;
26 import java.util.StringTokenizer;
27
28
29
30
31
32
33
34
35 public class BasicJoinEngine implements JoinEngine {
36
37 public static final String EXPECTED_JOINERS = "ExpectedJoiners";
38 public static final String ACTUAL_JOINERS = "ActualJoiners";
39
40 public void createExpectedJoinState(RouteContext context, RouteNodeInstance joinInstance, RouteNodeInstance previousNodeInstance) {
41 RouteNodeInstance splitNode = previousNodeInstance.getBranch().getSplitNode();
42 if (splitNode == null) {
43 throw new WorkflowRuntimeException("The split node retrieved from node with name '" + previousNodeInstance.getName() + "' and branch with name '" + previousNodeInstance.getBranch().getName() + "' was null");
44 }
45 for (Iterator iter = splitNode.getNextNodeInstances().iterator(); iter.hasNext();) {
46 RouteNodeInstance splitNodeNextNode = (RouteNodeInstance) iter.next();
47 splitNodeNextNode.getBranch().setJoinNode(joinInstance);
48
49
50 if (!OrmUtils.isJpaEnabled("rice.kew")) {
51 saveBranch(context, splitNodeNextNode.getBranch());
52 }
53 addExpectedJoiner(joinInstance, splitNodeNextNode.getBranch());
54 }
55 joinInstance.setBranch(splitNode.getBranch());
56 joinInstance.setProcess(splitNode.getProcess());
57 }
58
59 public void addExpectedJoiner(RouteNodeInstance nodeInstance, Branch branch) {
60 addJoinState(nodeInstance, branch, EXPECTED_JOINERS);
61 }
62
63 public void addActualJoiner(RouteNodeInstance nodeInstance, Branch branch) {
64 addJoinState(nodeInstance, branch, ACTUAL_JOINERS);
65 }
66
67 private void addJoinState(RouteNodeInstance nodeInstance, Branch branch, String key) {
68 NodeState state = nodeInstance.getNodeState(key);
69 if (state == null) {
70 state = new NodeState();
71 state.setKey(key);
72 state.setValue("");
73 state.setNodeInstance(nodeInstance);
74 nodeInstance.addNodeState(state);
75 }
76 state.setValue(state.getValue()+branch.getBranchId()+",");
77 }
78
79 public boolean isJoined(RouteNodeInstance nodeInstance) {
80 NodeState expectedState = nodeInstance.getNodeState(EXPECTED_JOINERS);
81 if (expectedState == null || org.apache.commons.lang.StringUtils.isEmpty(expectedState.getValue())) {
82 return true;
83 }
84 NodeState actualState = nodeInstance.getNodeState(ACTUAL_JOINERS);
85 Set expectedSet = loadIntoSet(expectedState);
86 Set actualSet = loadIntoSet(actualState);
87 for (Iterator iterator = expectedSet.iterator(); iterator.hasNext();) {
88 String value = (String) iterator.next();
89 if (actualSet.contains(value)) {
90 iterator.remove();
91 }
92 }
93 return expectedSet.size() == 0;
94 }
95
96 private Set loadIntoSet(NodeState state) {
97 Set set = new HashSet();
98 StringTokenizer tokenizer = new StringTokenizer(state.getValue(), ",");
99 while (tokenizer.hasMoreTokens()) {
100 set.add(tokenizer.nextToken());
101 }
102 return set;
103 }
104
105 private void saveBranch(RouteContext context, Branch branch) {
106 if (!context.isSimulation()) {
107 KEWServiceLocator.getRouteNodeService().save(branch);
108 }
109 }
110
111 }