001    /**
002     * Copyright 2005-2014 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.actionrequest.service.impl;
017    
018    import java.util.ArrayList;
019    import java.util.Collection;
020    import java.util.Collections;
021    import java.util.HashMap;
022    import java.util.HashSet;
023    import java.util.Iterator;
024    import java.util.List;
025    import java.util.Map;
026    import java.util.Set;
027    
028    import org.apache.commons.lang.ObjectUtils;
029    import org.apache.commons.lang.StringUtils;
030    import org.apache.log4j.Logger;
031    import org.kuali.rice.core.api.config.CoreConfigHelper;
032    import org.kuali.rice.core.api.exception.RiceRuntimeException;
033    import org.kuali.rice.core.api.config.property.ConfigContext;
034    import org.kuali.rice.coreservice.framework.CoreFrameworkServiceLocator;
035    import org.kuali.rice.kew.actionitem.ActionItem;
036    import org.kuali.rice.kew.actionlist.service.ActionListService;
037    import org.kuali.rice.kew.actionrequest.ActionRequestValue;
038    import org.kuali.rice.kew.actionrequest.Recipient;
039    import org.kuali.rice.kew.actionrequest.dao.ActionRequestDAO;
040    import org.kuali.rice.kew.actionrequest.service.ActionRequestService;
041    import org.kuali.rice.kew.api.KewApiServiceLocator;
042    import org.kuali.rice.kew.api.document.DocumentRefreshQueue;
043    import org.kuali.rice.kew.actiontaken.ActionTakenValue;
044    import org.kuali.rice.kew.actiontaken.service.ActionTakenService;
045    import org.kuali.rice.kew.api.action.ActionRequestPolicy;
046    import org.kuali.rice.kew.api.action.ActionRequestStatus;
047    import org.kuali.rice.kew.api.action.RecipientType;
048    import org.kuali.rice.kew.doctype.bo.DocumentType;
049    import org.kuali.rice.kew.engine.ActivationContext;
050    import org.kuali.rice.kew.engine.node.RouteNodeInstance;
051    import org.kuali.rice.kew.exception.WorkflowServiceErrorException;
052    import org.kuali.rice.kew.exception.WorkflowServiceErrorImpl;
053    import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
054    import org.kuali.rice.kew.routeheader.service.RouteHeaderService;
055    import org.kuali.rice.kew.routemodule.RouteModule;
056    import org.kuali.rice.kew.service.KEWServiceLocator;
057    import org.kuali.rice.kew.util.FutureRequestDocumentStateManager;
058    import org.kuali.rice.kew.api.KewApiConstants;
059    import org.kuali.rice.kew.util.PerformanceLogger;
060    import org.kuali.rice.kew.util.ResponsibleParty;
061    import org.kuali.rice.kim.api.group.Group;
062    import org.kuali.rice.kim.api.identity.principal.Principal;
063    import org.kuali.rice.kim.api.services.KimApiServiceLocator;
064    import org.kuali.rice.krad.util.KRADConstants;
065    
066    
067    /**
068     * Default implementation of the {@link ActionRequestService}.
069     *
070     * @author Kuali Rice Team (rice.collab@kuali.org)
071     */
072    public class ActionRequestServiceImpl implements ActionRequestService {
073        private static final Logger LOG = Logger.getLogger(ActionRequestServiceImpl.class);
074    
075        private ActionRequestDAO actionRequestDAO;
076    
077        public ActionRequestValue findByActionRequestId(String actionRequestId) {
078            return getActionRequestDAO().getActionRequestByActionRequestId(actionRequestId);
079        }
080    
081        public Map<String, String> getActionsRequested(DocumentRouteHeaderValue routeHeader, String principalId, boolean completeAndApproveTheSame) {
082            return getActionsRequested(principalId, routeHeader.getActionRequests(), completeAndApproveTheSame);
083        }
084        
085        /**
086         * Returns a Map of actions that are requested for the given principalId in the given list of action requests.
087         * @param principalId
088         * @param actionRequests
089         * @param completeAndApproveTheSame
090         * @return
091         */
092        protected Map<String, String> getActionsRequested(String principalId, List<ActionRequestValue> actionRequests, boolean completeAndApproveTheSame) {
093            Map<String, String> actionsRequested = new HashMap<String, String>();
094            actionsRequested.put(KewApiConstants.ACTION_REQUEST_FYI_REQ, "false");
095            actionsRequested.put(KewApiConstants.ACTION_REQUEST_ACKNOWLEDGE_REQ, "false");
096            actionsRequested.put(KewApiConstants.ACTION_REQUEST_APPROVE_REQ, "false");
097            actionsRequested.put(KewApiConstants.ACTION_REQUEST_COMPLETE_REQ, "false");
098            String topActionRequested = KewApiConstants.ACTION_REQUEST_FYI_REQ;
099            for (ActionRequestValue actionRequest : actionRequests) {
100                // we are getting the full list of requests here, so no need to look at role requests, if we did this then
101                // we could get a "false positive" for "all approve" roles where only part of the request graph is marked
102                // as "done"
103                if (!RecipientType.ROLE.getCode().equals(actionRequest.getRecipientTypeCd()) &&
104                        actionRequest.isRecipientRoutedRequest(principalId) && actionRequest.isActive()) {
105                    int actionRequestComparison = ActionRequestValue.compareActionCode(actionRequest.getActionRequested(), topActionRequested, completeAndApproveTheSame);
106                    if (actionRequest.isFYIRequest() && actionRequestComparison >= 0) {
107                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_FYI_REQ, "true");
108                    } else if (actionRequest.isAcknowledgeRequest() && actionRequestComparison >= 0) {
109                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_ACKNOWLEDGE_REQ, "true");
110                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_FYI_REQ, "false");
111                        topActionRequested = actionRequest.getActionRequested();
112                    } else if (actionRequest.isApproveRequest() && actionRequestComparison >= 0) {
113                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_APPROVE_REQ, "true");
114                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_ACKNOWLEDGE_REQ, "false");
115                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_FYI_REQ, "false");
116                        topActionRequested = actionRequest.getActionRequested();
117                    } else if (actionRequest.isCompleteRequst() && actionRequestComparison >= 0) {
118                            actionsRequested.put(KewApiConstants.ACTION_REQUEST_COMPLETE_REQ, "true");
119                            actionsRequested.put(KewApiConstants.ACTION_REQUEST_APPROVE_REQ, "false");
120                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_ACKNOWLEDGE_REQ, "false");
121                        actionsRequested.put(KewApiConstants.ACTION_REQUEST_FYI_REQ, "false");
122                            if (completeAndApproveTheSame) {
123                                    actionsRequested.put(KewApiConstants.ACTION_REQUEST_APPROVE_REQ, "true");
124                            }
125                        topActionRequested = actionRequest.getActionRequested();
126                    }
127                }
128            }
129            return actionsRequested;
130        }
131    
132        public ActionRequestValue initializeActionRequestGraph(ActionRequestValue actionRequest,
133                DocumentRouteHeaderValue document, RouteNodeInstance nodeInstance) {
134            if (actionRequest.getParentActionRequest() != null) {
135                LOG.warn("-->A non parent action request from doc " + document.getDocumentId());
136                actionRequest = KEWServiceLocator.getActionRequestService().getRoot(actionRequest);
137            }
138            propagatePropertiesToRequestGraph(actionRequest, document, nodeInstance);
139            return actionRequest;
140        }
141    
142        private void propagatePropertiesToRequestGraph(ActionRequestValue actionRequest, DocumentRouteHeaderValue document,
143                RouteNodeInstance nodeInstance) {
144            setPropertiesToRequest(actionRequest, document, nodeInstance);
145            for (ActionRequestValue actionRequestValue : actionRequest.getChildrenRequests())
146            {
147                propagatePropertiesToRequestGraph(actionRequestValue, document, nodeInstance);
148            }
149        }
150    
151        private void setPropertiesToRequest(ActionRequestValue actionRequest, DocumentRouteHeaderValue document,
152                RouteNodeInstance nodeInstance) {
153            actionRequest.setDocumentId(document.getDocumentId());
154            actionRequest.setDocVersion(document.getDocVersion());
155            actionRequest.setRouteLevel(document.getDocRouteLevel());
156            actionRequest.setNodeInstance(nodeInstance);
157            actionRequest.setStatus(ActionRequestStatus.INITIALIZED.getCode());
158        }
159    
160    
161    
162        public void activateRequests(Collection actionRequests) {
163            activateRequests(actionRequests, new ActivationContext(!ActivationContext.CONTEXT_IS_SIMULATION));
164        }
165    
166        public void activateRequests(Collection actionRequests, boolean simulate) {
167            activateRequests(actionRequests, new ActivationContext(simulate));
168        }
169    
170        public void activateRequests(Collection actionRequests, ActivationContext activationContext) {
171            if (actionRequests == null) {
172                return;
173            }
174            PerformanceLogger performanceLogger = null;
175            if ( LOG.isInfoEnabled() ) {
176                    performanceLogger = new PerformanceLogger();
177            }
178            activationContext.setGeneratedActionItems(new ArrayList<ActionItem>());
179            activateRequestsInternal(actionRequests, activationContext);
180            if (!activationContext.isSimulation()) {
181                KEWServiceLocator.getNotificationService().notify(ActionItem.to(activationContext.getGeneratedActionItems()));
182            }
183            if ( LOG.isInfoEnabled() ) {
184                    performanceLogger.log("Time to " + (activationContext.isSimulation() ? "simulate activation of " : "activate ")
185                                    + actionRequests.size() + " action requests.");
186            }
187            if ( LOG.isDebugEnabled() ) {
188                    LOG.debug("Generated " + activationContext.getGeneratedActionItems().size() + " action items.");
189            }
190        }
191    
192        public void activateRequest(ActionRequestValue actionRequest) {
193            activateRequests(Collections.singletonList(actionRequest), new ActivationContext(!ActivationContext.CONTEXT_IS_SIMULATION));
194        }
195    
196        public void activateRequest(ActionRequestValue actionRequest, boolean simulate) {
197            activateRequests(Collections.singletonList(actionRequest), new ActivationContext(simulate));
198        }
199    
200        public void activateRequest(ActionRequestValue actionRequest, ActivationContext activationContext) {
201            activateRequests(Collections.singletonList(actionRequest), activationContext);
202        }
203    
204        public List activateRequestNoNotification(ActionRequestValue actionRequest, boolean simulate) {
205            return activateRequestNoNotification(actionRequest, new ActivationContext(simulate));
206        }
207    
208        public List activateRequestNoNotification(ActionRequestValue actionRequest, ActivationContext activationContext) {
209            activationContext.setGeneratedActionItems(new ArrayList<ActionItem>());
210            activateRequestInternal(actionRequest, activationContext);
211            return activationContext.getGeneratedActionItems();
212        }
213    
214        /**
215         * Internal helper method for activating a Collection of action requests and their children. Maintains an accumulator
216         * for generated action items.
217         * @param actionRequests
218         * @param activationContext
219         */
220        private void activateRequestsInternal(Collection actionRequests, ActivationContext activationContext) {
221            if (actionRequests == null) {
222                return;
223            }
224            List<?> actionRequestList = new ArrayList<Object>(actionRequests);
225            for (int i = 0; i < actionRequestList.size(); i++) {
226                    activateRequestInternal((ActionRequestValue) actionRequestList.get(i), activationContext);
227            }
228        }
229    
230        /**
231         * Internal helper method for activating a single action requests and it's children. Maintains an accumulator for
232         * generated action items.
233         */
234        private void activateRequestInternal(ActionRequestValue actionRequest, ActivationContext activationContext) {
235            PerformanceLogger performanceLogger = null;
236            if ( LOG.isInfoEnabled() ) {
237                    performanceLogger = new PerformanceLogger();
238            }
239            if (actionRequest == null || actionRequest.isActive() || actionRequest.isDeactivated()) {
240                return;
241            }
242            processResponsibilityId(actionRequest);
243            if (deactivateOnActionAlreadyTaken(actionRequest, activationContext)) {
244                return;
245            }
246            if (deactivateOnInactiveGroup(actionRequest, activationContext)) {
247                return;
248            }
249            if (deactivateOnEmptyGroup(actionRequest, activationContext)) {
250                    return;
251            }
252            actionRequest.setStatus(ActionRequestStatus.ACTIVATED.getCode());
253            if (!activationContext.isSimulation()) {
254                saveActionRequest(actionRequest);
255                activationContext.getGeneratedActionItems().addAll(generateActionItems(actionRequest, activationContext));
256            }
257            activateRequestsInternal(actionRequest.getChildrenRequests(), activationContext);
258            activateRequestInternal(actionRequest.getParentActionRequest(), activationContext);
259            if ( LOG.isInfoEnabled() ) {
260                    if (activationContext.isSimulation()) {
261                    performanceLogger.log("Time to simulate activation of request.");
262                    } else {
263                        performanceLogger.log("Time to activate action request with id " + actionRequest.getActionRequestId());
264                    }
265            }
266        }
267    
268        /**
269         * Generates ActionItems for the given ActionRequest and returns the List of generated Action Items.
270         *
271         * @param actionRequest
272         * @param activationContext
273         * @return the List of generated ActionItems
274         */
275        private List<ActionItem> generateActionItems(ActionRequestValue actionRequest, ActivationContext activationContext) {
276            if ( LOG.isDebugEnabled() ) {
277                    LOG.debug("generating the action items for request " + actionRequest.getActionRequestId());
278            }
279            List<ActionItem> actionItems = new ArrayList<ActionItem>();
280            if (!actionRequest.isPrimaryDelegator()) {
281                if (actionRequest.isGroupRequest()) {
282                    List<String> principalIds =  KimApiServiceLocator.getGroupService().getMemberPrincipalIds(actionRequest.getGroupId());
283                    actionItems.addAll(createActionItemsForPrincipals(actionRequest, principalIds));
284                } else if (actionRequest.isUserRequest()) {
285                    ActionItem actionItem = getActionListService().createActionItemForActionRequest(actionRequest);
286                    actionItems.add(actionItem);
287                }
288            }
289            if (!activationContext.isSimulation()) {
290                for (ActionItem actionItem: actionItems) {
291                    if ( LOG.isDebugEnabled() ) {
292                            LOG.debug("Saving action item: " + actionItems);
293                    }
294                    getActionListService().saveActionItem(actionItem);
295                }
296            } else {
297                    actionRequest.getSimulatedActionItems().addAll(actionItems);
298            }
299            return actionItems;
300        }
301    
302        private List<ActionItem> createActionItemsForPrincipals(ActionRequestValue actionRequest, List<String> principalIds) {
303            List<ActionItem> actionItems = new ArrayList<ActionItem>();
304            for (String principalId: principalIds) {
305    
306                ActionItem actionItem = getActionListService().createActionItemForActionRequest(actionRequest);
307                actionItem.setPrincipalId(principalId);
308                actionItem.setRoleName(actionRequest.getQualifiedRoleName());
309    
310                //KULRICE-3307 Prevent workflow from attempting to activate requests for null principals
311                String ignoreUnknownPrincipalIdsValue = ConfigContext.getCurrentContextConfig().getProperty(KewApiConstants.WORKFLOW_ACTION_IGNORE_UNKOWN_PRINCIPAL_IDS);
312                boolean ignoreUnknownPrincipalIds = Boolean.parseBoolean(ignoreUnknownPrincipalIdsValue);
313    
314                if(principalId==null && ignoreUnknownPrincipalIds)
315                {
316                    LOG.warn("Ignoring action item with actionRequestID of " + actionRequest.getActionRequestId()  + " due to null principalId.");
317                }
318                else
319                {
320                    if(principalId==null)
321                    {
322                        IllegalArgumentException e = new IllegalArgumentException("Exception thrown when trying to add action item with null principalId");
323                        LOG.error(e);
324                        throw e;
325                    }
326                    else
327                    {
328                        actionItems.add(actionItem);
329                    }
330                }
331            }
332            return actionItems;
333        }
334    
335        private void processResponsibilityId(ActionRequestValue actionRequest) {
336            if (actionRequest.getResolveResponsibility()) {
337                    String responsibilityId = actionRequest.getResponsibilityId();
338                    try {
339                        RouteModule routeModule = KEWServiceLocator.getRouteModuleService().findRouteModule(actionRequest);
340                        if (responsibilityId != null && actionRequest.isRouteModuleRequest()) {
341                            if ( LOG.isDebugEnabled() ) {
342                                    LOG.debug("Resolving responsibility id for action request id=" + actionRequest.getActionRequestId()
343                                    + " and responsibility id=" + actionRequest.getResponsibilityId());
344                            }
345                            ResponsibleParty responsibleParty = routeModule.resolveResponsibilityId(actionRequest.getResponsibilityId());
346                            if (responsibleParty == null) {
347                                return;
348                            }
349                            if (responsibleParty.getPrincipalId() != null) {
350                                Principal user = KimApiServiceLocator.getIdentityService()
351                                        .getPrincipal(responsibleParty.getPrincipalId());
352                                actionRequest.setPrincipalId(user.getPrincipalId());
353                            } else if (responsibleParty.getGroupId() != null) {
354                                    actionRequest.setGroupId(responsibleParty.getGroupId());
355                            } else if (responsibleParty.getRoleName() != null) {
356                                actionRequest.setRoleName(responsibleParty.getRoleName());
357                            }
358                        }
359                    } catch (Exception e) {
360                        LOG.error("Exception thrown when trying to resolve responsibility id " + responsibilityId, e);
361                        throw new RuntimeException(e);
362                    }
363            }
364        }
365    
366        protected boolean deactivateOnActionAlreadyTaken(ActionRequestValue actionRequestToActivate,
367                ActivationContext activationContext) {
368    
369            FutureRequestDocumentStateManager futureRequestStateMngr = null;
370    
371            if (actionRequestToActivate.isGroupRequest()) {
372                futureRequestStateMngr = new FutureRequestDocumentStateManager(actionRequestToActivate.getRouteHeader(), actionRequestToActivate.getGroup());
373            } else if (actionRequestToActivate.isUserRequest()) {
374                futureRequestStateMngr = new FutureRequestDocumentStateManager(actionRequestToActivate.getRouteHeader(), actionRequestToActivate.getPrincipalId());
375            } else {
376                return false;
377            }
378    
379            if (futureRequestStateMngr.isReceiveFutureRequests()) {
380                return false;
381            }
382            if (!actionRequestToActivate.getForceAction() || futureRequestStateMngr.isDoNotReceiveFutureRequests()) {
383                ActionTakenValue previousActionTaken = null;
384                if (!activationContext.isSimulation()) {
385                    previousActionTaken = getActionTakenService().getPreviousAction(actionRequestToActivate);
386                } else {
387                    previousActionTaken = getActionTakenService().getPreviousAction(actionRequestToActivate,
388                            activationContext.getSimulatedActionsTaken());
389                }
390                if (previousActionTaken != null) {
391                    if ( LOG.isDebugEnabled() ) {
392                            LOG.debug("found a satisfying action taken so setting this request done.  Action Request Id "
393                                + actionRequestToActivate.getActionRequestId());
394                    }
395                    // set up the delegation for an action taken if this is a delegate request and the delegate has
396                    // already taken action.
397                    if (!previousActionTaken.isForDelegator() && actionRequestToActivate.getParentActionRequest() != null) {
398                        previousActionTaken.setDelegator(actionRequestToActivate.getParentActionRequest().getRecipient());
399                        if (!activationContext.isSimulation()) {
400                            getActionTakenService().saveActionTaken(previousActionTaken);
401                        }
402                    }
403                    deactivateRequest(previousActionTaken, actionRequestToActivate, null, activationContext);
404                    return true;
405                }
406            }
407            if ( LOG.isDebugEnabled() ) {
408                    LOG.debug("Forcing action for action request " + actionRequestToActivate.getActionRequestId());
409            }
410            return false;
411        }
412        
413        /**
414         * Checks if the action request which is being activated has a group with no members.  If this is the case then it will immediately
415         * initiate de-activation on the request since a group with no members will result in no action items being generated so should be
416         * effectively skipped.
417         */
418        protected boolean deactivateOnEmptyGroup(ActionRequestValue actionRequestToActivate, ActivationContext activationContext) {
419            if (actionRequestToActivate.isGroupRequest()) {
420                     if (KimApiServiceLocator.getGroupService().getMemberPrincipalIds(actionRequestToActivate.getGroup().getId()).isEmpty()) {
421                             deactivateRequest(null, actionRequestToActivate, null, activationContext);
422                             return true;
423                    }
424            }
425            return false;
426        }
427    
428        /**
429         * Checks if the action request which is being activated is being assigned to an inactive group.  If this is the case and if the FailOnInactiveGroup 
430         * policy is set to false then it will immediately initiate de-activation on the request
431         */
432        protected boolean deactivateOnInactiveGroup(ActionRequestValue actionRequestToActivate, ActivationContext activationContext) {
433            if (actionRequestToActivate.isGroupRequest()) {
434                if (!actionRequestToActivate.getGroup().isActive() && !actionRequestToActivate.getRouteHeader().getDocumentType().getFailOnInactiveGroup().getPolicyValue()) {
435                    deactivateRequest(null, actionRequestToActivate, null, activationContext);
436                    return true;
437                }
438            }
439            return false;
440        }
441        
442        public void deactivateRequest(ActionTakenValue actionTaken, ActionRequestValue actionRequest) {
443            deactivateRequest(actionTaken, actionRequest, null, new ActivationContext(!ActivationContext.CONTEXT_IS_SIMULATION));
444        }
445    
446        public void deactivateRequest(ActionTakenValue actionTaken, ActionRequestValue actionRequest, boolean simulate) {
447            deactivateRequest(actionTaken, actionRequest, null, new ActivationContext(simulate));
448        }
449    
450        public void deactivateRequest(ActionTakenValue actionTaken, ActionRequestValue actionRequest,
451                ActivationContext activationContext) {
452            deactivateRequest(actionTaken, actionRequest, null, activationContext);
453        }
454    
455        public void deactivateRequests(ActionTakenValue actionTaken, List actionRequests) {
456            deactivateRequests(actionTaken, actionRequests, null,
457                    new ActivationContext(!ActivationContext.CONTEXT_IS_SIMULATION));
458        }
459    
460        public void deactivateRequests(ActionTakenValue actionTaken, List actionRequests, boolean simulate) {
461            deactivateRequests(actionTaken, actionRequests, null, new ActivationContext(simulate));
462        }
463    
464        public void deactivateRequests(ActionTakenValue actionTaken, List actionRequests, ActivationContext activationContext) {
465            deactivateRequests(actionTaken, actionRequests, null, activationContext);
466        }
467    
468        private void deactivateRequests(ActionTakenValue actionTaken, Collection actionRequests,
469                ActionRequestValue deactivationRequester, ActivationContext activationContext) {
470            if (actionRequests == null) {
471                return;
472            }
473            for (Iterator iterator = actionRequests.iterator(); iterator.hasNext();) {
474                ActionRequestValue actionRequest = (ActionRequestValue) iterator.next();
475                deactivateRequest(actionTaken, actionRequest, deactivationRequester, activationContext);
476            }
477        }
478    
479        private void deactivateRequest(ActionTakenValue actionTaken, ActionRequestValue actionRequest,
480                ActionRequestValue deactivationRequester, ActivationContext activationContext) {
481            if (actionRequest == null || actionRequest.isDeactivated()
482                    || haltForAllApprove(actionRequest, deactivationRequester)) {
483                return;
484            }
485            actionRequest.setStatus(ActionRequestStatus.DONE.getCode());
486            actionRequest.setActionTaken(actionTaken);
487            if (actionTaken != null) {
488                actionTaken.getActionRequests().add(actionRequest);
489            }
490            if (!activationContext.isSimulation()) {
491                getActionRequestDAO().saveActionRequest(actionRequest);
492                deleteActionItems(actionRequest);
493            }
494            deactivateRequests(actionTaken, actionRequest.getChildrenRequests(), actionRequest, activationContext);
495            deactivateRequest(actionTaken, actionRequest.getParentActionRequest(), actionRequest, activationContext);
496        }
497    
498        /**
499         * Returns true if we are dealing with an 'All Approve' request, the requester of the deactivation is a child of the
500         * 'All Approve' request, and all of the children have not been deactivated. If all of the children are already
501         * deactivated or a non-child request initiated deactivation, then this method returns false. false otherwise.
502         * @param actionRequest
503         * @param deactivationRequester
504         * @return
505         */
506        private boolean haltForAllApprove(ActionRequestValue actionRequest, ActionRequestValue deactivationRequester) {
507            if (ActionRequestPolicy.ALL.getCode().equals(actionRequest.getApprovePolicy())
508                    && actionRequest.hasChild(deactivationRequester)) {
509                boolean allDeactivated = true;
510                for (ActionRequestValue childRequest : actionRequest.getChildrenRequests())
511                {
512                    if (!(allDeactivated = childRequest.isDeactivated()))
513                    {
514                        return true;
515                    }
516                }
517            }
518            return false;
519        }
520    
521        public List<ActionRequestValue> getRootRequests(Collection<ActionRequestValue> actionRequests) {
522            Set<ActionRequestValue> unsavedRequests = new HashSet<ActionRequestValue>();
523            Map<String, ActionRequestValue> requestMap = new HashMap<String, ActionRequestValue>();
524            for (ActionRequestValue actionRequest1 : actionRequests)
525            {
526                    ActionRequestValue actionRequest = (ActionRequestValue) actionRequest1;
527                    ActionRequestValue rootRequest = getRoot(actionRequest);
528                    if (rootRequest.getActionRequestId() != null)
529                    {
530                            requestMap.put(rootRequest.getActionRequestId(), rootRequest);
531                    } else
532                    {
533                            unsavedRequests.add(rootRequest);
534                    }
535            }
536            List<ActionRequestValue> requests = new ArrayList<ActionRequestValue>();
537            requests.addAll(requestMap.values());
538            requests.addAll(unsavedRequests);
539            return requests;
540        }
541    
542        public ActionRequestValue getRoot(ActionRequestValue actionRequest) {
543            if (actionRequest == null) {
544                return null;
545            }
546            if (actionRequest.getParentActionRequest() != null) {
547                return getRoot(actionRequest.getParentActionRequest());
548            }
549            return actionRequest;
550        }
551        
552        /**
553         * Returns all pending requests for a given routing identity
554         * @param documentId the id of the document header being routed
555         * @return a List of all pending ActionRequestValues for the document
556         */
557        public List<ActionRequestValue> findAllPendingRequests(String documentId) {
558            ActionRequestDAO arDAO = getActionRequestDAO();
559            List<ActionRequestValue> pendingArs = arDAO.findByStatusAndDocId(ActionRequestStatus.ACTIVATED.getCode(), documentId);
560            return pendingArs;
561        }
562    
563        public List findAllValidRequests(String principalId, String documentId, String requestCode) {
564            ActionRequestDAO arDAO = getActionRequestDAO();
565            Collection pendingArs = arDAO.findByStatusAndDocId(ActionRequestStatus.ACTIVATED.getCode(), documentId);
566            return findAllValidRequests(principalId, pendingArs, requestCode);
567        }
568    
569        public List findAllValidRequests(String principalId, Collection actionRequests, String requestCode) {
570            List matchedArs = new ArrayList();
571            List<String> arGroups = KimApiServiceLocator.getGroupService().getGroupIdsByPrincipalId(principalId);
572            return filterActionRequestsByCode((List<ActionRequestValue>)actionRequests, principalId, arGroups, requestCode);
573        }
574        
575        /**
576             * Filters action requests based on if they occur after the given requestCode, and if they relate to 
577             * the given principal
578             * @param actionRequests the List of ActionRequestValues to filter
579             * @param principalId the id of the principal to find active requests for
580             * @param principalGroupIds List of group ids that the principal belongs to
581             * @param requestCode the request code for all ActionRequestValues to be after
582             * @return the filtered List of ActionRequestValues
583             */
584            public List<ActionRequestValue> filterActionRequestsByCode(List<ActionRequestValue> actionRequests, String principalId, List<String> principalGroupIds, String requestCode) {
585                    List<ActionRequestValue> filteredActionRequests = new ArrayList<ActionRequestValue>();
586                    
587                    List<String> arGroups = null;
588            for (ActionRequestValue ar : actionRequests) {
589                if (ActionRequestValue.compareActionCode(ar.getActionRequested(), requestCode, true) > 0) {
590                    continue;
591                }
592                if (ar.isUserRequest() && principalId.equals(ar.getPrincipalId())) {
593                    filteredActionRequests.add(ar);
594                } else if (ar.isGroupRequest() && principalGroupIds != null && !principalGroupIds.isEmpty()) {
595                    for (String groupId : principalGroupIds) {
596                            if (groupId.equals(ar.getGroupId())) {
597                                    filteredActionRequests.add(ar);
598                            }
599                    }
600                }
601            }
602                    
603                    return filteredActionRequests;
604            }
605    
606        public void updateActionRequestsForResponsibilityChange(Set<String> responsibilityIds) {
607            PerformanceLogger performanceLogger = null;
608            if ( LOG.isInfoEnabled() ) {
609                    performanceLogger = new PerformanceLogger();
610            }
611            Collection documentsAffected = getRouteHeaderService().findPendingByResponsibilityIds(responsibilityIds);
612            String cacheWaitValue = CoreFrameworkServiceLocator.getParameterService().getParameterValueAsString(KewApiConstants.KEW_NAMESPACE, KRADConstants.DetailTypes.RULE_DETAIL_TYPE, KewApiConstants.RULE_CACHE_REQUEUE_DELAY);
613            Long cacheWait = KewApiConstants.DEFAULT_CACHE_REQUEUE_WAIT_TIME;
614            if (!org.apache.commons.lang.StringUtils.isEmpty(cacheWaitValue)) {
615                try {
616                    cacheWait = Long.valueOf(cacheWaitValue);
617                } catch (NumberFormatException e) {
618                    LOG.warn("Cache wait time is not a valid number: " + cacheWaitValue);
619                }
620            }
621            if ( LOG.isInfoEnabled() ) {
622                    LOG.info("Scheduling requeue of " + documentsAffected.size() + " documents, affected by " + responsibilityIds.size()
623                        + " responsibility changes.  Installing a processing wait time of " + cacheWait
624                        + " milliseconds to avoid stale rule cache.");
625            }
626            for (Object aDocumentsAffected : documentsAffected)
627            {
628                String documentId = (String) aDocumentsAffected;
629    
630                 String applicationId = null;
631                 DocumentType documentType = KEWServiceLocator.getDocumentTypeService().findByDocumentId(documentId);
632                        
633                 if (documentType != null) {
634                    applicationId = documentType.getApplicationId();
635                 }
636    
637                if (applicationId == null)
638                {
639                    applicationId = CoreConfigHelper.getApplicationId();
640                }
641                if(documentType.getRegenerateActionRequestsOnChange().getPolicyValue()) {
642                    DocumentRefreshQueue documentRequeuer = KewApiServiceLocator.getDocumentRequeuerService(applicationId,
643                            documentId, cacheWait);
644                    documentRequeuer.refreshDocument(documentId);
645                }
646            }
647            if ( LOG.isInfoEnabled() ) {
648                    performanceLogger.log("Time to updateActionRequestsForResponsibilityChange");
649            }
650        }
651    
652        /**
653         * Deletes an action request and all of its action items following the graph down through the action request's
654         * children. This method should be invoked on a top-level action request.
655         */
656        public void deleteActionRequestGraph(ActionRequestValue actionRequest) {
657            deleteActionItems(actionRequest);
658            if (actionRequest.getActionTakenId() != null) {
659                ActionTakenValue actionTaken = getActionTakenService().findByActionTakenId(actionRequest.getActionTakenId());
660    
661                if(actionTaken != null){//iu patch
662                getActionTakenService().delete(actionTaken);
663                }//iu patch
664               
665            }
666            getActionRequestDAO().delete(actionRequest.getActionRequestId());
667            for (ActionRequestValue child: actionRequest.getChildrenRequests()) {
668                deleteActionRequestGraph(child);
669            }
670        }
671    
672        /**
673         * Deletes the action items for the action request
674         * @param actionRequest the action request whose action items to delete
675         */
676        private void deleteActionItems(ActionRequestValue actionRequest) {
677            List<ActionItem> actionItems = actionRequest.getActionItems();
678            if ( LOG.isDebugEnabled() ) {
679                    LOG.debug("deleting " + actionItems.size() + " action items for action request: " + actionRequest);
680            }
681            for (ActionItem actionItem: actionItems) {
682                    if ( LOG.isDebugEnabled() ) {
683                            LOG.debug("deleting action item: " + actionItem);
684                    }
685                getActionListService().deleteActionItem(actionItem);
686            }
687        }
688    
689    
690        public List<ActionRequestValue> findByDocumentIdIgnoreCurrentInd(String documentId) {
691            return getActionRequestDAO().findByDocumentIdIgnoreCurrentInd(documentId);
692        }
693    
694        public List<ActionRequestValue> findAllActionRequestsByDocumentId(String documentId) {
695            return getActionRequestDAO().findAllByDocId(documentId);
696        }
697    
698        public List<ActionRequestValue> findAllRootActionRequestsByDocumentId(String documentId) {
699            return getActionRequestDAO().findAllRootByDocId(documentId);
700        }
701    
702        public List<ActionRequestValue> findPendingByActionRequestedAndDocId(String actionRequestedCd, String documentId) {
703            return getActionRequestDAO().findPendingByActionRequestedAndDocId(actionRequestedCd, documentId);
704        }
705    
706        /**
707         * @see org.kuali.rice.kew.actionrequest.service.ActionRequestService#getPrincipalIdsWithPendingActionRequestByActionRequestedAndDocId(java.lang.String, java.lang.String)
708         */
709        public List<String> getPrincipalIdsWithPendingActionRequestByActionRequestedAndDocId(String actionRequestedCd, String documentId) {
710            List<String> principalIds = new ArrayList<String>();
711            List<ActionRequestValue> actionRequests = findPendingByActionRequestedAndDocId(actionRequestedCd, documentId);
712                    for(ActionRequestValue actionRequest: actionRequests){
713                            if(actionRequest.isUserRequest()){
714                                    principalIds.add(actionRequest.getPrincipalId());
715                            } else if(actionRequest.isGroupRequest()){
716                                    principalIds.addAll(
717                                                    KimApiServiceLocator.getGroupService().getMemberPrincipalIds(actionRequest.getGroupId()));
718                            }
719                    }
720            return principalIds;
721        }
722    
723        public List<ActionRequestValue> findPendingByDocIdAtOrBelowRouteLevel(String documentId, Integer routeLevel) {
724            return getActionRequestDAO().findPendingByDocIdAtOrBelowRouteLevel(documentId, routeLevel);
725        }
726    
727        public List<ActionRequestValue> findPendingRootRequestsByDocId(String documentId) {
728            return getRootRequests(findPendingByDoc(documentId));
729        }
730    
731        public List<ActionRequestValue> findPendingRootRequestsByDocIdAtRouteNode(String documentId, String nodeInstanceId) {
732            return getActionRequestDAO().findPendingRootRequestsByDocIdAtRouteNode(documentId, nodeInstanceId);
733        }
734    
735        public List<ActionRequestValue> findRootRequestsByDocIdAtRouteNode(String documentId, String nodeInstanceId) {
736            return getActionRequestDAO().findRootRequestsByDocIdAtRouteNode(documentId, nodeInstanceId);
737        }
738    
739        public List<ActionRequestValue> findPendingRootRequestsByDocIdAtOrBelowRouteLevel(String documentId, Integer routeLevel) {
740            return getActionRequestDAO().findPendingRootRequestsByDocIdAtOrBelowRouteLevel(documentId, routeLevel);
741        }
742    
743        public List<ActionRequestValue> findPendingRootRequestsByDocIdAtRouteLevel(String documentId, Integer routeLevel) {
744            return getActionRequestDAO().findPendingRootRequestsByDocIdAtRouteLevel(documentId, routeLevel);
745        }
746    
747        public List<ActionRequestValue> findPendingRootRequestsByDocumentType(String documentTypeId) {
748            return getActionRequestDAO().findPendingRootRequestsByDocumentType(documentTypeId);
749        }
750    
751        public void saveActionRequest(ActionRequestValue actionRequest) {
752            if (actionRequest.isGroupRequest()) {
753                 Group group = actionRequest.getGroup();
754                 if (group == null)  {
755                     throw new RiceRuntimeException("Attempted to save an action request with a non-existent group.");
756                 }
757                 if (!group.isActive() && actionRequest.getRouteHeader().getDocumentType().getFailOnInactiveGroup().getPolicyValue()) {
758                            throw new RiceRuntimeException("Attempted to save an action request with an inactive group.");
759                    }
760            }
761            getActionRequestDAO().saveActionRequest(actionRequest);
762        }
763    
764        public List<ActionRequestValue> findPendingByDoc(String documentId) {
765            return getActionRequestDAO().findAllPendingByDocId(documentId);
766        }
767    
768        public List<ActionRequestValue> findPendingByDocRequestCdRouteLevel(String documentId, String requestCode, Integer routeLevel) {
769            List<ActionRequestValue> requests = new ArrayList<ActionRequestValue>();
770            for (Object object : getActionRequestDAO().findAllPendingByDocId(documentId))
771            {
772                ActionRequestValue actionRequest = (ActionRequestValue) object;
773                if (ActionRequestValue.compareActionCode(actionRequest.getActionRequested(), requestCode, true) > 0)
774                {
775                    continue;
776                }
777                if (actionRequest.getRouteLevel().intValue() == routeLevel.intValue())
778                {
779                    requests.add(actionRequest);
780                }
781            }
782            return requests;
783        }
784    
785        public List<ActionRequestValue> findPendingByDocRequestCdNodeName(String documentId, String requestCode, String nodeName) {
786            List<ActionRequestValue> requests = new ArrayList<ActionRequestValue>();
787            for (Object object : getActionRequestDAO().findAllPendingByDocId(documentId))
788            {
789                ActionRequestValue actionRequest = (ActionRequestValue) object;
790                if (ActionRequestValue.compareActionCode(actionRequest.getActionRequested(), requestCode, true) > 0)
791                {
792                    continue;
793                }
794                if (actionRequest.getNodeInstance() != null && actionRequest.getNodeInstance().getName().equals(nodeName))
795                {
796                    requests.add(actionRequest);
797                }
798            }
799            return requests;
800        }
801    
802        public List findActivatedByGroup(String groupId) {
803            return getActionRequestDAO().findActivatedByGroup(groupId);
804        }
805    
806        private ActionListService getActionListService() {
807            return (ActionListService) KEWServiceLocator.getActionListService();
808        }
809    
810        private ActionTakenService getActionTakenService() {
811            return (ActionTakenService) KEWServiceLocator.getActionTakenService();
812        }
813    
814        public ActionRequestDAO getActionRequestDAO() {
815            return actionRequestDAO;
816        }
817    
818        public void setActionRequestDAO(ActionRequestDAO actionRequestDAO) {
819            this.actionRequestDAO = actionRequestDAO;
820        }
821    
822        private RouteHeaderService getRouteHeaderService() {
823            return (RouteHeaderService) KEWServiceLocator.getService(KEWServiceLocator.DOC_ROUTE_HEADER_SRV);
824        }
825    
826        public List<ActionRequestValue> findByStatusAndDocId(String statusCd, String documentId) {
827            return getActionRequestDAO().findByStatusAndDocId(statusCd, documentId);
828        }
829    
830        public void alterActionRequested(List actionRequests, String actionRequestCd) {
831            for (Object actionRequest1 : actionRequests)
832            {
833                ActionRequestValue actionRequest = (ActionRequestValue) actionRequest1;
834    
835                actionRequest.setActionRequested(actionRequestCd);
836                for (ActionItem item : actionRequest.getActionItems())
837                {
838                    item.setActionRequestCd(actionRequestCd);
839                }
840    
841                saveActionRequest(actionRequest);
842            }
843        }
844    
845        // TODO this still won't work in certain cases when checking from the root
846        public boolean isDuplicateRequest(ActionRequestValue actionRequest) {
847            List<ActionRequestValue> requests = findAllRootActionRequestsByDocumentId(actionRequest.getDocumentId());
848            for (ActionRequestValue existingRequest : requests) {
849                if (existingRequest.getStatus().equals(ActionRequestStatus.DONE.getCode())
850                        && existingRequest.getRouteLevel().equals(actionRequest.getRouteLevel())
851                        && ObjectUtils.equals(existingRequest.getPrincipalId(), actionRequest.getPrincipalId())
852                        && ObjectUtils.equals(existingRequest.getGroupId(), actionRequest.getGroupId())
853                        && ObjectUtils.equals(existingRequest.getRoleName(), actionRequest.getRoleName())
854                        && ObjectUtils.equals(existingRequest.getQualifiedRoleName(), actionRequest.getQualifiedRoleName())
855                        && existingRequest.getActionRequested().equals(actionRequest.getActionRequested())) {
856                    return true;
857                }
858            }
859            return false;
860        }
861    
862        public Recipient findDelegator(List actionRequests) {
863            Recipient delegator = null;
864            String requestCode = KewApiConstants.ACTION_REQUEST_FYI_REQ;
865            for (Object actionRequest1 : actionRequests)
866            {
867                ActionRequestValue actionRequest = (ActionRequestValue) actionRequest1;
868                ActionRequestValue delegatorRequest = findDelegatorRequest(actionRequest);
869                if (delegatorRequest != null)
870                {
871                    if (ActionRequestValue.compareActionCode(delegatorRequest.getActionRequested(), requestCode, true) >= 0)
872                    {
873                        delegator = delegatorRequest.getRecipient();
874                        requestCode = delegatorRequest.getActionRequested();
875                    }
876                }
877            }
878            return delegator;
879        }
880    
881        public Recipient findDelegator(ActionRequestValue actionRequest) {
882            ActionRequestValue delegatorRequest = findDelegatorRequest(actionRequest);
883            Recipient delegator = null;
884            if (delegatorRequest != null) {
885                delegator = delegatorRequest.getRecipient();
886            }
887            return delegator;
888        }
889    
890        public ActionRequestValue findDelegatorRequest(ActionRequestValue actionRequest) {
891            ActionRequestValue parentRequest = actionRequest.getParentActionRequest();
892            if (parentRequest != null && !(parentRequest.isUserRequest() || parentRequest.isGroupRequest())) {
893                parentRequest = findDelegatorRequest(parentRequest);
894            }
895            return parentRequest;
896        }
897    
898        public void deleteByDocumentId(String documentId) {
899            actionRequestDAO.deleteByDocumentId(documentId);
900        }
901    
902        public void deleteByActionRequestId(String actionRequestId) {
903            actionRequestDAO.delete(actionRequestId);
904        }
905    
906        public void validateActionRequest(ActionRequestValue actionRequest) {
907            LOG.debug("Enter validateActionRequest(..)");
908            List<WorkflowServiceErrorImpl> errors = new ArrayList<WorkflowServiceErrorImpl>();
909    
910            String actionRequestCd = actionRequest.getActionRequested();
911            if (actionRequestCd == null || actionRequestCd.trim().equals("")) {
912                errors.add(new WorkflowServiceErrorImpl("ActionRequest cd null.", "actionrequest.actionrequestcd.empty",
913                        actionRequest.getActionRequestId().toString()));
914            } else if (!KewApiConstants.ACTION_REQUEST_CD.containsKey(actionRequestCd)) {
915                errors.add(new WorkflowServiceErrorImpl("ActionRequest cd invalid.", "actionrequest.actionrequestcd.invalid",
916                        actionRequest.getActionRequestId().toString()));
917            }
918    
919            String documentId = actionRequest.getDocumentId();
920            if (documentId == null || StringUtils.isEmpty(documentId)) {
921                    errors.add(new WorkflowServiceErrorImpl("ActionRequest Document id empty.", "actionrequest.documentid.empty",
922                        actionRequest.getActionRequestId().toString()));
923            } else if (getRouteHeaderService().getRouteHeader(documentId) == null) {
924                errors.add(new WorkflowServiceErrorImpl("ActionRequest Document id invalid.",
925                        "actionrequest.documentid.invalid", actionRequest.getActionRequestId().toString()));
926            }
927    
928            String actionRequestStatus = actionRequest.getStatus();
929            if (actionRequestStatus == null || actionRequestStatus.trim().equals("")) {
930                errors.add(new WorkflowServiceErrorImpl("ActionRequest status null.", "actionrequest.actionrequeststatus.empty",
931                        actionRequest.getActionRequestId().toString()));
932            } else if (ActionRequestStatus.fromCode(actionRequestStatus) == null) {
933                errors.add(new WorkflowServiceErrorImpl("ActionRequest status invalid.",
934                        "actionrequest.actionrequeststatus.invalid", actionRequest.getActionRequestId().toString()));
935            }
936    
937            if (actionRequest.getResponsibilityId() == null) {
938                errors.add(new WorkflowServiceErrorImpl("ActionRequest responsibility id null.",
939                        "actionrequest.responsibilityid.empty", actionRequest.getActionRequestId().toString()));
940            }
941    
942            Integer priority = actionRequest.getPriority();
943            if (priority == null) {
944                errors.add(new WorkflowServiceErrorImpl("ActionRequest priority null.", "actionrequest.priority.empty",
945                        actionRequest.getActionRequestId().toString()));
946            }
947    
948            // if(actionRequest.getRouteMethodName() == null || actionRequest.getRouteMethodName().trim().equals("")){
949            // errors.add(new WorkflowServiceErrorImpl("ActionRequest route method name null.",
950            // "actionrequest.routemethodname.empty", actionRequest.getActionRequestId().toString()));
951            // }
952    
953            Integer routeLevel = actionRequest.getRouteLevel();
954            if (routeLevel == null) {
955                errors.add(new WorkflowServiceErrorImpl("ActionRequest route level null.", "actionrequest.routelevel.empty",
956                        actionRequest.getActionRequestId().toString()));
957            } else if (routeLevel < -1) {
958                errors.add(new WorkflowServiceErrorImpl("ActionRequest route level invalid.",
959                        "actionrequest.routelevel.invalid", actionRequest.getActionRequestId().toString()));
960            }
961    
962            Integer version = actionRequest.getDocVersion();
963            if (version == null) {
964                errors.add(new WorkflowServiceErrorImpl("ActionRequest doc version null.", "actionrequest.docversion.empty",
965                        actionRequest.getActionRequestId().toString()));
966            }
967    
968            if (actionRequest.getCreateDate() == null) {
969                errors.add(new WorkflowServiceErrorImpl("ActionRequest create date null.", "actionrequest.createdate.empty",
970                        actionRequest.getActionRequestId().toString()));
971            }
972    
973            String recipientType = actionRequest.getRecipientTypeCd();
974            if (recipientType != null && !recipientType.trim().equals("")) {
975                if (recipientType.equals(KewApiConstants.WORKGROUP)) {
976                    String workgroupId = actionRequest.getGroupId();
977                    if (workgroupId == null) {
978                        errors.add(new WorkflowServiceErrorImpl("ActionRequest workgroup null.",
979                                "actionrequest.workgroup.empty", actionRequest.getActionRequestId().toString()));
980                    } else if (KimApiServiceLocator.getGroupService().getGroup(workgroupId) == null) {
981                        errors.add(new WorkflowServiceErrorImpl("ActionRequest workgroup invalid.",
982                                "actionrequest.workgroup.invalid", actionRequest.getActionRequestId().toString()));
983                    }
984    
985                }
986                if (recipientType.equals(KewApiConstants.PERSON)) {
987                    String principalId = actionRequest.getPrincipalId();
988                    if (principalId == null || principalId.trim().equals("")) {
989                        errors.add(new WorkflowServiceErrorImpl("ActionRequest person id null.", "actionrequest.persosn.empty",
990                                actionRequest.getActionRequestId().toString()));
991                    } else {
992                            Principal principal = KimApiServiceLocator.getIdentityService().getPrincipal(principalId);
993                            if (principal == null) {
994                                    errors.add(new WorkflowServiceErrorImpl("ActionRequest person id invalid.",
995                                                    "actionrequest.personid.invalid", actionRequest.getActionRequestId().toString()));
996                            }
997                    }
998    
999                    if (recipientType.equals(KewApiConstants.ROLE)
1000                            && (actionRequest.getRoleName() == null || actionRequest.getRoleName().trim().equals(""))) {
1001                        errors.add(new WorkflowServiceErrorImpl("ActionRequest role name null.", "actionrequest.rolename.null",
1002                                actionRequest.getActionRequestId().toString()));
1003                    }
1004                }
1005                LOG.debug("Exit validateActionRequest(..) ");
1006                if (!errors.isEmpty()) {
1007                    throw new WorkflowServiceErrorException("ActionRequest Validation Error", errors);
1008                }
1009            }
1010        }
1011    
1012        public List getDelegateRequests(ActionRequestValue actionRequest) {
1013            List<ActionRequestValue> delegateRequests = new ArrayList<ActionRequestValue>();
1014            List requests = getTopLevelRequests(actionRequest);
1015            for (Object request : requests)
1016            {
1017                ActionRequestValue parentActionRequest = (ActionRequestValue) request;
1018                delegateRequests.addAll(parentActionRequest.getChildrenRequests());
1019            }
1020            return delegateRequests;
1021        }
1022    
1023        public List getTopLevelRequests(ActionRequestValue actionRequest) {
1024            List<ActionRequestValue> topLevelRequests = new ArrayList<ActionRequestValue>();
1025            if (actionRequest.isRoleRequest()) {
1026                topLevelRequests.addAll(actionRequest.getChildrenRequests());
1027            } else {
1028                topLevelRequests.add(actionRequest);
1029            }
1030            return topLevelRequests;
1031        }
1032    
1033        public boolean isValidActionRequestCode(String actionRequestCode) {
1034            return actionRequestCode != null && KewApiConstants.ACTION_REQUEST_CODES.containsKey(actionRequestCode);
1035        }
1036    
1037        public boolean doesPrincipalHaveRequest(String principalId, String documentId) {
1038            if (getActionRequestDAO().doesDocumentHaveUserRequest(principalId, documentId)) {
1039                return true;
1040            }
1041            // TODO since we only store the workgroup id for workgroup requests, if the user is in a workgroup that has a request
1042            // than we need get all the requests with workgroup ids and see if our user is in that group
1043            List<String> groupIds = getActionRequestDAO().getRequestGroupIds(documentId);
1044            for (String groupId : groupIds) {
1045                if (KimApiServiceLocator.getGroupService().isMemberOfGroup(principalId, groupId)) {
1046                    return true;
1047                }
1048            }
1049            return false;
1050        }
1051    
1052        @Override
1053        public ActionRequestValue getActionRequestForRole(String actionTakenId) {
1054            return getActionRequestDAO().getRoleActionRequestByActionTakenId(actionTakenId);
1055        }
1056    
1057    }