001    /**
002     * Copyright 2005-2012 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.kew.engine.node;
017    
018    import org.apache.log4j.MDC;
019    import org.kuali.rice.kew.actionrequest.ActionRequestValue;
020    import org.kuali.rice.kew.api.exception.WorkflowException;
021    import org.kuali.rice.kew.engine.RouteContext;
022    import org.kuali.rice.kew.engine.RouteHelper;
023    import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
024    import org.kuali.rice.kew.service.KEWServiceLocator;
025    import org.kuali.rice.kew.api.KewApiConstants;
026    import org.kuali.rice.kew.util.PerformanceLogger;
027    import org.kuali.rice.kew.util.Utilities;
028    
029    import java.util.ArrayList;
030    import java.util.Collections;
031    import java.util.Iterator;
032    import java.util.List;
033    
034    
035    /**
036     * A node which will iteratively activate any requests pending on it.  Subclasses are responsible for generating the requests
037     * to be activated, and then delegating to superclass {@link #process(RouteContext, RouteHelper)} to activate
038     * those requests.
039     *
040     * This node can be used to serve multiple batches of requests, and can be re-entered multiple times.  The algorithm it implements is:
041     * 
042     * <ol>
043     *   <li>activate any existing requests (requests may have been generated by an external party)
044     *   <li>while request fulfillment criteria indicates that any pending requests have been satisfied
045     *     <ol>
046     *       <li>generate new requests</li>
047     *       <li><b>if no requests were generated, we are done, return and transition</b></li>
048     *       <li>otherwise if requests were generated
049     *         <ol>
050     *           <li>activate requests (depending on activation policy, serial or parallel, <i>NOT ALL GENERATED REQUESTS MAY BE ACTIVATED AT THIS POINT</i>)</li>
051     *           <li>determine a request fulfillment criteria</li>
052     *           <li>continue in loop</li>
053     *         </ol>
054     *       </li>
055     *   </li>
056     *   <li>otherwise if pending requests have not been satisfied, block until invoked again</li>
057     * </ol>
058     * This node transitions/completes when there are no remaining <i>blocking</i> action requests (i.e., no approval or
059     * completion requests).
060     *
061     * @author Kuali Rice Team (rice.collab@kuali.org)
062     */
063    public class IteratedRequestActivationNode implements SimpleNode {
064    
065        protected final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(getClass());
066        private static long generatedRequestPriority = 0;
067    
068        protected static interface RequestFulfillmentCriteria {
069            public boolean pendingRequestsAreFulfilled(boolean activatedBlockingRequests, RouteContext routeContext);
070        }
071    
072        private static class AllBlockingRequestsCompleteCriteria implements RequestFulfillmentCriteria {
073            public boolean pendingRequestsAreFulfilled(boolean activatedBlockingRequests, RouteContext routeContext) {
074                // determine whether there are any pending blocking requests
075                if (activatedBlockingRequests) {
076                    // if we just activated blocking requests then surely there are pending blocking requests
077                    assert(activatedBlockingRequests == blockingRequestsArePending(routeContext.getDocument(), routeContext.getNodeInstance())) : "Blocking requests were activated but none were subsequently found";
078                    return false;
079                } else {
080                    // otherwise let's see if there are any pre-existing blocking requests associated with this node instance
081                    return !blockingRequestsArePending(routeContext.getDocument(), routeContext.getNodeInstance());
082                }
083            }
084        }
085    
086        private static class SimulatingCriteria implements RequestFulfillmentCriteria {
087            public boolean pendingRequestsAreFulfilled(boolean activatedBlockingRequests, RouteContext routeContext) {
088                // when simulating, never block for requests to be fulfilled
089                return true;
090            }
091        }
092    
093        public SimpleResult process(RouteContext routeContext, RouteHelper routeHelper) throws Exception {
094            // if previous requests were satisfied (if no requests have been generated yet, this should return true
095            RequestFulfillmentCriteria criteria = getRequestFulfillmentCriteria(routeContext);
096            // generate new requests until they are not satisfied or none are generated
097            // if none are generated then we should transition immediately
098            boolean activatedBlockingRequests = activateRequests(routeContext); // activation should always occur as something might have generated unactived requests
099            boolean initialRequestGeneration = routeContext.getNodeInstance().isInitial();
100            while (criteria.pendingRequestsAreFulfilled(activatedBlockingRequests, routeContext)) {
101                boolean newRequestsGenerated = generateNewRequests(initialRequestGeneration, routeContext, routeHelper);
102                initialRequestGeneration = false;
103    
104                if (!newRequestsGenerated) {
105                    // if the pending requests were fulfilled
106                    // ...and we didn't generate any new requests
107                    // ...then there are no further requests to activate (XXX: WRONG)
108                    // ...and we have no more processing to do
109                    return new SimpleResult(true);
110                }
111    
112                // activate any requests that were generated
113                activatedBlockingRequests = activateRequests(routeContext);
114    
115                // set the request fulfillment criteria for the new set of requests
116                // if we are simulating, always set a criteria that indicates fulfillment
117                if (routeContext.isSimulation()) {
118                    criteria = new SimulatingCriteria();
119                } else {
120                    criteria = getRequestFulfillmentCriteria(routeContext);
121                }
122            }
123            // if we got here, then for some reason pending requests have not been fulfilled
124            // so wait/block for request fulfillment
125            return new SimpleResult(false);
126        }
127    
128        /**
129         * Template method that subclasses should override to indicate when activated requests have been fulfilled.
130         * The default implementation is that all pending blocking requests must be fulfilled.
131         * @return a RequestFulfillmentCriteria that indicates when activated requests have been fulfilled.
132         */
133        protected RequestFulfillmentCriteria getRequestFulfillmentCriteria(RouteContext routeContext) {
134            return new AllBlockingRequestsCompleteCriteria();
135        }
136    
137        /**
138         * Template method that subclasses should override to generates new requests
139         * @param initial whether this is the very first request generation, that is, the first generation in the first invocation of the node
140         * @return whether new requests were generated
141         * @throws Exception 
142         */
143        protected boolean generateNewRequests(boolean initial, RouteContext context, RouteHelper routeHelper) throws WorkflowException, Exception {
144            return false;
145        }
146    
147        /**
148         * Activates any pending requests and returns whether there are outstanding blocking requests
149         * @param context the RouteContext
150         * @throws org.kuali.rice.kew.api.exception.WorkflowException if anything goes wrong...
151         * @return whether there are outstanding blocking requests
152         */
153        protected boolean activateRequests(RouteContext routeContext) throws WorkflowException {
154            DocumentRouteHeaderValue document = routeContext.getDocument();
155            RouteNodeInstance nodeInstance = routeContext.getNodeInstance();
156            if (routeContext.isSimulation()) {
157                // this seems to indicate whether, when we are simulating, to activate requests...
158                if (routeContext.getActivationContext().isActivateRequests()) {
159                    activateRequests(routeContext, document, nodeInstance);
160                }
161                // if we are in simulation, don't block, just transition out
162                return false;
163            } else {
164                // activate any unactivated pending requests on this node instance
165                return activateRequests(routeContext, document, nodeInstance);
166            }
167        }
168    
169        /**
170         * @return whether there are any pending requests at the given route node instance which are blocking (i.e., 'approve' or 'complete')
171         */
172        private static boolean blockingRequestsArePending(DocumentRouteHeaderValue document, RouteNodeInstance nodeInstance) {
173            // returns blocking requests that are *activated*
174            List<ActionRequestValue> requests = KEWServiceLocator.getActionRequestService().findPendingRootRequestsByDocIdAtRouteNode(document.getDocumentId(), nodeInstance.getRouteNodeInstanceId());
175            boolean blockingRequestsArePending = false;
176            for (ActionRequestValue request: requests) {
177                if (request.isApproveOrCompleteRequest()) {
178                    blockingRequestsArePending = true;
179                    break;
180                }
181            }
182            return blockingRequestsArePending;
183        }
184    
185        /**
186         * Activates the action requests that are pending at this routelevel of the document. The requests are processed by priority and then request ID.
187         * It is implicit in the access that the requests are activated according to the route level above all.
188         * <p>
189         * FYI and acknowledgement requests do not cause the processing to stop. Only action requests for approval or completion cause the processing to
190         * stop and then only for route level with a serialized activation policy. Only requests at the current document's current route level are activated.
191         * Inactive requests at a lower level cause a routing exception.
192         * <p>
193         * Exception routing and adhoc routing are processed slightly differently.
194         * 
195         * 
196         * @param context the RouteContext
197         * @param document the document we are processing
198         * @param nodeInstance the node instance we are processing
199         * @return True if the any blocking actions requests (approve or complete) were activated.
200         * @throws org.kuali.rice.kew.api.exception.ResourceUnavailableException
201         * @throws org.kuali.rice.kew.api.exception.WorkflowException
202         */
203        private boolean activateRequests(RouteContext context, DocumentRouteHeaderValue document, RouteNodeInstance nodeInstance) throws WorkflowException {
204            MDC.put("docId", document.getDocumentId());
205            PerformanceLogger performanceLogger = new PerformanceLogger(document.getDocumentId());
206            List generatedActionItems = new ArrayList();
207            List<ActionRequestValue> requests = KEWServiceLocator.getActionRequestService().findPendingRootRequestsByDocIdAtRouteNode(document.getDocumentId(), nodeInstance.getRouteNodeInstanceId());
208            if (context.isSimulation()) {
209                requests.addAll(context.getEngineState().getGeneratedRequests());
210            }
211            // this will sort higher priority requests to the front
212            // blocking requests are higher priority, so all blocking requests will be
213            // activated before non-blocking requests
214            Collections.sort(requests, new Utilities.PrioritySorter());
215            LOG.info("Pending Root Requests " + requests.size());
216            String activationType = nodeInstance.getRouteNode().getActivationType();
217            boolean isParallel = KewApiConstants.ROUTE_LEVEL_PARALLEL.equals(activationType);
218            boolean activatedApproveRequest = false;
219            for (Iterator iter = requests.iterator(); iter.hasNext();) {
220                if (activatedApproveRequest && !isParallel) {
221                    LOG.info("Already activated an apprve request and serial, so not activating any more");
222                    break;
223                }
224                ActionRequestValue request = (ActionRequestValue) iter.next();
225                LOG.info("ActionRequestValue: " + request);
226                if (request.getParentActionRequest() != null || request.getNodeInstance() == null) {
227                    // 1. disregard request if it's not a top-level request
228                    // 2. disregard request if it's a "future" request and hasn't been attached to a node instance yet
229                    continue; 
230                }
231                if (request.isActive()) {
232                    activatedApproveRequest = activatedApproveRequest || request.isApproveOrCompleteRequest();
233                    continue;
234                }
235                logProcessingMessage(request);   
236                LOG.info("Activating request. " + request);
237                activatedApproveRequest = activateRequest(context, request, nodeInstance, generatedActionItems) || activatedApproveRequest;
238            }
239            // now let's send notifications, since this code needs to be able to activate each request individually, we need
240            // to collection all action items and then notify after all have been generated
241            if (!context.isSimulation()) {
242                KEWServiceLocator.getNotificationService().notify(generatedActionItems);
243            }
244            performanceLogger.log("Time to activate requests.");
245            return activatedApproveRequest;
246        }
247    
248        private boolean activateRequest(RouteContext context, ActionRequestValue actionRequest, RouteNodeInstance nodeInstance, List generatedActionItems) {
249            if (actionRequest.isRoleRequest()) {
250                List actionRequests = KEWServiceLocator.getActionRequestService().findPendingRootRequestsByDocIdAtRouteNode(actionRequest.getDocumentId(), nodeInstance.getRouteNodeInstanceId());
251                for (Iterator iterator = actionRequests.iterator(); iterator.hasNext();) {
252                    ActionRequestValue siblingRequest = (ActionRequestValue) iterator.next();
253                    if (actionRequest.getRoleName().equals(siblingRequest.getRoleName())) {
254                        generatedActionItems.addAll(KEWServiceLocator.getActionRequestService().activateRequestNoNotification(siblingRequest, context.getActivationContext()));
255                    }
256                }
257            }
258            generatedActionItems.addAll(KEWServiceLocator.getActionRequestService().activateRequestNoNotification(actionRequest, context.getActivationContext()));
259            return actionRequest.isApproveOrCompleteRequest() && ! actionRequest.isDone();
260        }
261        
262        protected void saveActionRequest(RouteContext context, ActionRequestValue actionRequest) {
263            if (!context.isSimulation()) {
264                KEWServiceLocator.getActionRequestService().saveActionRequest(actionRequest);
265            } else {
266                actionRequest.setActionRequestId(String.valueOf(generatedRequestPriority++));
267                context.getEngineState().getGeneratedRequests().add(actionRequest);    
268            }
269            
270        }
271        
272        private void logProcessingMessage(ActionRequestValue request) {
273            //if (LOG.isDebugEnabled()) {
274                    RouteNodeInstance nodeInstance = request.getNodeInstance();
275                StringBuffer buffer = new StringBuffer();
276                buffer.append("Processing AR: ").append(request.getActionRequestId()).append("\n");
277                buffer.append("AR Node Name: ").append(nodeInstance != null ? nodeInstance.getName() : "null").append("\n");
278                buffer.append("AR RouteLevel: ").append(request.getRouteLevel()).append("\n");
279                buffer.append("AR Request Code: ").append(request.getActionRequested()).append("\n");
280                buffer.append("AR Request priority: ").append(request.getPriority()).append("\n");
281                LOG.info(buffer);
282            //}
283        }
284                
285    }