Coverage Report - org.kuali.rice.kew.routelog.web.RouteLogAction
 
Classes in this File Line Coverage Branch Coverage Complexity
RouteLogAction
0%
0/130
0%
0/62
3.895
RouteLogAction$RouteNodeInstanceFabricator
0%
0/80
0%
0/32
3.895
 
 1  
 /*
 2  
  * Copyright 2005-2007 The Kuali Foundation
 3  
  *
 4  
  *
 5  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 6  
  * you may not use this file except in compliance with the License.
 7  
  * You may obtain a copy of the License at
 8  
  *
 9  
  * http://www.opensource.org/licenses/ecl2.php
 10  
  *
 11  
  * Unless required by applicable law or agreed to in writing, software
 12  
  * distributed under the License is distributed on an "AS IS" BASIS,
 13  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 14  
  * See the License for the specific language governing permissions and
 15  
  * limitations under the License.
 16  
  */
 17  
 package org.kuali.rice.kew.routelog.web;
 18  
 
 19  
 import org.apache.struts.action.ActionForm;
 20  
 import org.apache.struts.action.ActionForward;
 21  
 import org.apache.struts.action.ActionMapping;
 22  
 import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader;
 23  
 import org.kuali.rice.core.api.resourceloader.GlobalResourceLoader;
 24  
 import org.kuali.rice.kew.actionrequest.ActionRequestValue;
 25  
 import org.kuali.rice.kew.actionrequest.service.ActionRequestService;
 26  
 import org.kuali.rice.kew.actiontaken.ActionTakenValue;
 27  
 import org.kuali.rice.kew.doctype.SecuritySession;
 28  
 import org.kuali.rice.kew.doctype.service.DocumentSecurityService;
 29  
 import org.kuali.rice.kew.dto.*;
 30  
 import org.kuali.rice.kew.dto.DTOConverter.RouteNodeInstanceLoader;
 31  
 import org.kuali.rice.kew.engine.node.Branch;
 32  
 import org.kuali.rice.kew.engine.node.NodeState;
 33  
 import org.kuali.rice.kew.engine.node.RouteNode;
 34  
 import org.kuali.rice.kew.engine.node.RouteNodeInstance;
 35  
 import org.kuali.rice.kew.engine.node.service.RouteNodeService;
 36  
 import org.kuali.rice.kew.exception.InvalidActionTakenException;
 37  
 import org.kuali.rice.kew.exception.WorkflowRuntimeException;
 38  
 import org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue;
 39  
 import org.kuali.rice.kew.service.KEWServiceLocator;
 40  
 import org.kuali.rice.kew.service.WorkflowUtility;
 41  
 import org.kuali.rice.kew.util.KEWConstants;
 42  
 import org.kuali.rice.kew.util.Utilities;
 43  
 import org.kuali.rice.kew.web.KewKualiAction;
 44  
 import org.kuali.rice.kns.UserSession;
 45  
 import org.kuali.rice.kns.util.GlobalVariables;
 46  
 
 47  
 import javax.servlet.http.HttpServletRequest;
 48  
 import javax.servlet.http.HttpServletResponse;
 49  
 import javax.xml.namespace.QName;
 50  
 import java.util.*;
 51  
 
 52  
 
 53  
 /**
 54  
  * A Struts Action used to display the routelog.
 55  
  *
 56  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 57  
  */
 58  0
 public class RouteLogAction extends KewKualiAction {
 59  
 
 60  0
     private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(RouteLogAction.class);
 61  0
     private static Comparator<ActionRequestValue> ROUTE_LOG_ACTION_REQUEST_SORTER = new Utilities.RouteLogActionRequestSorter();
 62  
     
 63  
     @Override
 64  
         public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws Exception {
 65  
 
 66  0
         RouteLogForm rlForm = (RouteLogForm) form;
 67  0
         Long routeHeaderId = null;
 68  0
         if (! org.apache.commons.lang.StringUtils.isEmpty(rlForm.getRouteHeaderId())) {
 69  0
             routeHeaderId = new Long(rlForm.getRouteHeaderId());
 70  0
         } else if (! org.apache.commons.lang.StringUtils.isEmpty(rlForm.getDocId())) {
 71  0
             routeHeaderId = new Long(rlForm.getDocId());
 72  
         } else {
 73  0
                 throw new WorkflowRuntimeException("No paramater provided to fetch document");
 74  
         }
 75  
 
 76  0
         DocumentRouteHeaderValue routeHeader = KEWServiceLocator.getRouteHeaderService().getRouteHeader(routeHeaderId);
 77  
 
 78  0
         DocumentSecurityService security = KEWServiceLocator.getDocumentSecurityService();
 79  0
         if (!security.routeLogAuthorized(getUserSession(), routeHeader, new SecuritySession(GlobalVariables.getUserSession()))) {
 80  0
           return mapping.findForward("NotAuthorized");
 81  
         }
 82  
         
 83  0
         fixActionRequestsPositions(routeHeader);
 84  0
         populateRouteLogFormActionRequests(rlForm, routeHeader);
 85  
 
 86  0
         rlForm.setLookFuture(routeHeader.getDocumentType().getLookIntoFuturePolicy().getPolicyValue().booleanValue());
 87  
 
 88  0
         if (rlForm.isShowFuture()) {
 89  
             try {
 90  0
                 populateRouteLogFutureRequests(rlForm, routeHeader);
 91  0
             } catch (Exception e) {
 92  0
                 String errorMsg = "Unable to determine Future Action Requests";
 93  0
                 LOG.info(errorMsg,e);
 94  0
                 rlForm.setShowFutureError(errorMsg);
 95  0
             }
 96  
         }
 97  0
         request.setAttribute("routeHeader", routeHeader);
 98  
         
 99  
                 // check whether action message logging should be enabled, user must
 100  
                 // have KIM permission for doc type 
 101  0
         boolean isAuthorizedToAddRouteLogMessage = KEWServiceLocator.getDocumentTypePermissionService()
 102  
                                 .canAddRouteLogMessage(GlobalVariables.getUserSession().getPrincipalId(), routeHeader);
 103  0
                 if (isAuthorizedToAddRouteLogMessage) {
 104  0
                         rlForm.setEnableLogAction(true);
 105  
                 } else {
 106  0
                         rlForm.setEnableLogAction(false);
 107  
                 }
 108  
         
 109  0
         return super.execute(mapping, rlForm, request, response);
 110  
     }
 111  
 
 112  
     @SuppressWarnings("unchecked")
 113  
         public void populateRouteLogFormActionRequests(RouteLogForm rlForm, DocumentRouteHeaderValue routeHeader) {
 114  0
         List<ActionRequestValue> rootRequests = getActionRequestService().getRootRequests(routeHeader.getActionRequests());
 115  0
         Collections.sort(rootRequests, ROUTE_LOG_ACTION_REQUEST_SORTER);
 116  0
         rootRequests = switchActionRequestPositionsIfPrimaryDelegatesPresent(rootRequests);
 117  0
         int arCount = 0;
 118  0
         for ( ActionRequestValue actionRequest : rootRequests ) {
 119  0
             if (actionRequest.isPending()) {
 120  0
                 arCount++;
 121  
 
 122  0
                 if (KEWConstants.ACTION_REQUEST_INITIALIZED.equals(actionRequest.getStatus())) {
 123  0
                     actionRequest.setDisplayStatus("PENDING");
 124  0
                 } else if (KEWConstants.ACTION_REQUEST_ACTIVATED.equals(actionRequest.getStatus())) {
 125  0
                     actionRequest.setDisplayStatus("IN ACTION LIST");
 126  
                 }
 127  
             }
 128  
         }
 129  0
         rlForm.setRootRequests(rootRequests);
 130  0
         rlForm.setPendingActionRequestCount(arCount);
 131  0
     }
 132  
 
 133  
     @SuppressWarnings("unchecked")
 134  
         private ActionRequestValue switchActionRequestPositionIfPrimaryDelegatePresent( ActionRequestValue actionRequest ) {
 135  
             
 136  
             /**
 137  
              * KULRICE-4756 - The main goal here is to fix the regression of what happened in Rice 1.0.2 with the display
 138  
              * of primary delegate requests.  The delegate is displayed at the top-most level correctly on action requests
 139  
              * that are "rooted" at a "role" request.
 140  
              * 
 141  
              * If they are rooted at a principal or group request, then the display of the primary delegator at the top-most
 142  
              * level does not happen (instead it shows the delegator and you have to expand the request to see the primary
 143  
              * delegate).
 144  
              * 
 145  
              * Ultimately, the KAI group and Rice BA need to come up with a specification for how the Route Log should
 146  
              * display delegate information.  For now, will fix this so that in the non "role" case, it will put the
 147  
              * primary delegate as the outermost request *except* in the case where there is more than one primary delegate.
 148  
              */
 149  
             
 150  0
             if (!actionRequest.isRoleRequest()) {
 151  0
                     List<ActionRequestValue> primaryDelegateRequests = actionRequest.getPrimaryDelegateRequests();
 152  
                     // only display primary delegate request at top if there is only *one* primary delegate request
 153  0
                     if ( primaryDelegateRequests.size() != 1) {
 154  0
                             return actionRequest;
 155  
                     }
 156  0
                     ActionRequestValue primaryDelegateRequest = primaryDelegateRequests.get(0);
 157  0
                     actionRequest.getChildrenRequests().remove(primaryDelegateRequest);
 158  0
                     primaryDelegateRequest.setChildrenRequests(actionRequest.getChildrenRequests());
 159  0
                     primaryDelegateRequest.setParentActionRequest(actionRequest.getParentActionRequest());
 160  0
                     primaryDelegateRequest.setParentActionRequestId(actionRequest.getParentActionRequestId());
 161  
                     
 162  0
                     actionRequest.setChildrenRequests( new ArrayList<ActionRequestValue>(0) );
 163  0
                     actionRequest.setParentActionRequest(primaryDelegateRequest);
 164  0
                     actionRequest.setParentActionRequestId(primaryDelegateRequest.getActionRequestId());
 165  
                     
 166  0
                     primaryDelegateRequest.getChildrenRequests().add(0, actionRequest);
 167  
                     
 168  0
                     for (ActionRequestValue delegateRequest : primaryDelegateRequest.getChildrenRequests()) {
 169  0
                             delegateRequest.setParentActionRequest(primaryDelegateRequest);
 170  0
                             delegateRequest.setParentActionRequestId(primaryDelegateRequest.getActionRequestId());
 171  
                     }
 172  
                     
 173  0
                     return primaryDelegateRequest;
 174  
             }
 175  
             
 176  0
             return actionRequest;
 177  
     }
 178  
 
 179  
     private List<ActionRequestValue> switchActionRequestPositionsIfPrimaryDelegatesPresent( Collection<ActionRequestValue> actionRequests ) {
 180  0
             List<ActionRequestValue> results = new ArrayList<ActionRequestValue>( actionRequests.size() );
 181  0
             for ( ActionRequestValue actionRequest : actionRequests ) {
 182  0
                         results.add( switchActionRequestPositionIfPrimaryDelegatePresent(actionRequest) );
 183  
             }
 184  0
             return results;
 185  
     }
 186  
     
 187  
     @SuppressWarnings("unchecked")
 188  
     private void fixActionRequestsPositions(DocumentRouteHeaderValue routeHeader) {
 189  0
         for (ActionTakenValue actionTaken : routeHeader.getActionsTaken()) {
 190  0
             Collections.sort((List<ActionRequestValue>) actionTaken.getActionRequests(), ROUTE_LOG_ACTION_REQUEST_SORTER);
 191  0
             actionTaken.setActionRequests( actionTaken.getActionRequests() );
 192  
         }
 193  0
     }
 194  
     
 195  
     /**
 196  
      * executes a simulation of the future routing, and sets the futureRootRequests and futureActionRequestCount
 197  
      * properties on the provided RouteLogForm.
 198  
      * 
 199  
      * @param rlForm the RouteLogForm --used in a write-only fashion.
 200  
      * @param document the DocumentRouteHeaderValue for the document whose future routing is being simulated.
 201  
      * @throws Exception
 202  
      */
 203  
     public void populateRouteLogFutureRequests(RouteLogForm rlForm, DocumentRouteHeaderValue document) throws Exception {
 204  
 
 205  0
         ReportCriteriaDTO reportCriteria = new ReportCriteriaDTO(document.getRouteHeaderId());
 206  0
         String serviceNamespace = document.getDocumentType().getServiceNamespace();
 207  0
         WorkflowUtility workflowUtility = 
 208  
                 (WorkflowUtility) GlobalResourceLoader.getService(new QName(serviceNamespace, "WorkflowUtilityService"));
 209  
 
 210  
         // gather the IDs for action requests that predate the simulation
 211  0
                 Set<Long> preexistingActionRequestIds = getActionRequestIds(document);
 212  
         
 213  
                 // run the simulation via WorkflowUtility
 214  0
         DocumentDetailDTO documentDetail = workflowUtility.routingReport(reportCriteria);
 215  
 
 216  
         // fabricate our ActionRequestValueS from the results
 217  0
         List<ActionRequestValue> futureActionRequests = 
 218  
                 reconstituteActionRequestValues(documentDetail, preexistingActionRequestIds);
 219  
 
 220  0
         Collections.sort(futureActionRequests, ROUTE_LOG_ACTION_REQUEST_SORTER);
 221  
         
 222  0
         futureActionRequests = switchActionRequestPositionsIfPrimaryDelegatesPresent(futureActionRequests);
 223  
         
 224  0
         int pendingActionRequestCount = 0;
 225  0
         for (ActionRequestValue actionRequest: futureActionRequests) {
 226  0
             if (actionRequest.isPending()) {
 227  0
                 pendingActionRequestCount++;
 228  
 
 229  0
                 if (KEWConstants.ACTION_REQUEST_INITIALIZED.equals(actionRequest.getStatus())) {
 230  0
                     actionRequest.setDisplayStatus("PENDING");
 231  0
                 } else if (KEWConstants.ACTION_REQUEST_ACTIVATED.equals(actionRequest.getStatus())) {
 232  0
                     actionRequest.setDisplayStatus("IN ACTION LIST");
 233  
                 }
 234  
             }
 235  
         }
 236  
 
 237  0
         rlForm.setFutureRootRequests(futureActionRequests);
 238  0
         rlForm.setFutureActionRequestCount(pendingActionRequestCount);
 239  0
     }
 240  
 
 241  
 
 242  
         /**
 243  
          * This utility method returns a Set of LongS containing the IDs for the ActionRequestValueS associated with 
 244  
          * this DocumentRouteHeaderValue. 
 245  
          */
 246  
         @SuppressWarnings("unchecked")
 247  
         private Set<Long> getActionRequestIds(DocumentRouteHeaderValue document) {
 248  0
                 Set<Long> actionRequestIds = new HashSet<Long>();
 249  
 
 250  0
                 List<ActionRequestValue> actionRequests = 
 251  
                         KEWServiceLocator.getActionRequestService().findAllActionRequestsByRouteHeaderId(document.getRouteHeaderId());
 252  
                 
 253  0
                 if (actionRequests != null) {
 254  0
                         for (ActionRequestValue actionRequest : actionRequests) {
 255  0
                                 if (actionRequest.getActionRequestId() != null) {
 256  0
                                         actionRequestIds.add(actionRequest.getActionRequestId());
 257  
                                 }
 258  
                         }
 259  
                 }
 260  0
                 return actionRequestIds;
 261  
         }
 262  
 
 263  
         /**
 264  
          * This method creates ActionRequestValue objects from the DocumentDetailDTO output from 
 265  
          * {@link WorkflowUtility#routingReport(ReportCriteriaDTO)}Report()
 266  
          * 
 267  
          * @param documentDetail contains the DTOs from which the ActionRequestValues are reconstituted
 268  
          * @param preexistingActionRequestIds this is a Set of ActionRequest IDs that will not be reconstituted
 269  
          * @return the ActionRequestValueS that have been created
 270  
          */
 271  
         private List<ActionRequestValue> reconstituteActionRequestValues(DocumentDetailDTO documentDetail,
 272  
                         Set<Long> preexistingActionRequestIds) {
 273  
 
 274  0
         RouteNodeInstanceFabricator routeNodeInstanceFabricator = 
 275  
                     new RouteNodeInstanceFabricator(KEWServiceLocator.getRouteNodeService());
 276  
 
 277  0
         if (documentDetail.getNodeInstances() != null && documentDetail.getNodeInstances().length > 0) {
 278  0
                 for (RouteNodeInstanceDTO routeNodeInstanceVO : documentDetail.getNodeInstances()) {
 279  0
                         routeNodeInstanceFabricator.importRouteNodeInstanceDTO(routeNodeInstanceVO);
 280  
                 }
 281  
                 }
 282  
         
 283  0
         ActionRequestDTO[] actionRequestVOs = documentDetail.getActionRequests();
 284  0
         List<ActionRequestValue> futureActionRequests = new ArrayList<ActionRequestValue>();
 285  0
         if (actionRequestVOs != null) {
 286  0
                         for (ActionRequestDTO actionRequestVO : actionRequestVOs) {
 287  0
                                 if (actionRequestVO != null) {
 288  0
                                         if (!preexistingActionRequestIds.contains(actionRequestVO.getActionRequestId())) {
 289  0
                                                 ActionRequestValue converted = DTOConverter.convertActionRequestDTO(actionRequestVO, routeNodeInstanceFabricator);
 290  0
                                                 futureActionRequests.add(converted);
 291  
                                         }
 292  
                                 }
 293  
                         }
 294  
                 }
 295  0
                 return futureActionRequests;
 296  
         }
 297  
     
 298  
     private ActionRequestService getActionRequestService() {
 299  0
         return (ActionRequestService) KEWServiceLocator.getService(KEWServiceLocator.ACTION_REQUEST_SRV);
 300  
     }
 301  
     
 302  
     private UserSession getUserSession() {
 303  0
         return GlobalVariables.getUserSession();
 304  
     }
 305  
     
 306  
     /**
 307  
      * Creates dummy RouteNodeInstances based on imported data from RouteNodeInstanceDTOs.
 308  
      * It is then able to vend those RouteNodeInstanceS back by their IDs.
 309  
      * 
 310  
      * @author Kuali Rice Team (rice.collab@kuali.org)
 311  
      *
 312  
      */
 313  0
     private static class RouteNodeInstanceFabricator implements RouteNodeInstanceLoader {
 314  
 
 315  0
             private Map<Long,Branch> branches = new HashMap<Long, Branch>();;
 316  0
             private Map<Long,RouteNodeInstance> routeNodeInstances = new HashMap<Long, RouteNodeInstance>();
 317  0
             private Map<Long,RouteNode> routeNodes = new HashMap<Long, RouteNode>();
 318  0
             private Map<Long,NodeState> nodeStates = new HashMap<Long, NodeState>();
 319  
 
 320  
             private RouteNodeService routeNodeService;
 321  
             
 322  
             /**
 323  
                  * This constructs a FutureRouteNodeInstanceFabricator, which will generate bogus
 324  
                  * RouteNodeInstances for SimulationEngine results
 325  
                  * 
 326  
                  */
 327  0
                 public RouteNodeInstanceFabricator(RouteNodeService routeNodeService) {
 328  0
                         this.routeNodeService = routeNodeService;
 329  0
                 }
 330  
 
 331  
                 /**
 332  
                  * 
 333  
                  * This method looks at the given RouteNodeInstanceDTO and imports it (and all it's ancestors)
 334  
                  * as dummy RouteNodeInstanceS
 335  
                  * 
 336  
                  * @param nodeInstanceDTO
 337  
                  */
 338  
                 public void importRouteNodeInstanceDTO(RouteNodeInstanceDTO nodeInstanceDTO) {
 339  0
                         _importRouteNodeInstanceDTO(nodeInstanceDTO);
 340  0
                 }
 341  
                 
 342  
                 /**
 343  
                  * helper method for {@link #importRouteNodeInstanceDTO(RouteNodeInstanceDTO)} which does all
 344  
                  * the work.  The public method just wraps this one but hides the returned RouteNodeInstance,
 345  
                  * which is used for the recursive call to populate the nextNodeInstanceS inside our 
 346  
                  * RouteNodeInstanceS.
 347  
                  * 
 348  
                  * @param nodeInstanceDTO
 349  
                  * @return
 350  
                  */
 351  
             private RouteNodeInstance _importRouteNodeInstanceDTO(RouteNodeInstanceDTO nodeInstanceDTO) {
 352  0
                     if (nodeInstanceDTO == null) {
 353  0
                             return null;
 354  
                     }
 355  0
                     RouteNodeInstance nodeInstance = new RouteNodeInstance();
 356  0
                     nodeInstance.setActive(nodeInstanceDTO.isActive());
 357  
 
 358  0
                     nodeInstance.setComplete(nodeInstanceDTO.isComplete());
 359  0
                     nodeInstance.setDocumentId(nodeInstanceDTO.getDocumentId());
 360  0
                     nodeInstance.setInitial(nodeInstanceDTO.isInitial());
 361  
 
 362  0
                     Branch branch = getBranch(nodeInstanceDTO.getBranchId());
 363  0
                     nodeInstance.setBranch(branch);
 364  
 
 365  0
                     if (nodeInstanceDTO.getRouteNodeId() != null) {
 366  0
                             RouteNode routeNode = routeNodeService.findRouteNodeById(nodeInstanceDTO.getRouteNodeId());
 367  
 
 368  0
                             if (routeNode == null) {
 369  0
                                     routeNode = getRouteNode(nodeInstanceDTO.getRouteNodeId());
 370  0
                                     routeNode.setNodeType(nodeInstanceDTO.getName());
 371  
                             }
 372  
 
 373  0
                             nodeInstance.setRouteNode(routeNode);
 374  
 
 375  0
                             if (routeNode.getBranch() != null) {
 376  0
                                 branch.setName(routeNode.getBranch().getName());
 377  
                         } 
 378  
                     }
 379  
 
 380  0
                     RouteNodeInstance process = getRouteNodeInstance(nodeInstanceDTO.getProcessId());
 381  0
                     nodeInstance.setProcess(process);
 382  
 
 383  0
                     nodeInstance.setRouteNodeInstanceId(nodeInstanceDTO.getRouteNodeInstanceId());
 384  0
                     DTOConverter.convertState(null);
 385  
 
 386  0
                     List<NodeState> nodeState = new ArrayList<NodeState>();
 387  0
                     if (nodeInstanceDTO.getState() != null) {
 388  0
                                 for (StateDTO stateDTO : nodeInstanceDTO.getState()) {
 389  0
                                         NodeState state = getNodeState(stateDTO.getStateId());
 390  0
                                         if (state != null) {
 391  0
                                                 state.setKey(stateDTO.getKey());
 392  0
                                                 state.setValue(stateDTO.getValue());
 393  0
                                                 state.setStateId(stateDTO.getStateId());
 394  0
                                                 state.setNodeInstance(nodeInstance);
 395  0
                                                 nodeState.add(state);
 396  
                                         }
 397  
                                 }
 398  
                         }
 399  0
                     nodeInstance.setState(nodeState);
 400  
 
 401  0
                     List<RouteNodeInstance> nextNodeInstances = new ArrayList<RouteNodeInstance>();
 402  0
                     nodeInstance.setNextNodeInstances(nextNodeInstances);
 403  
 
 404  0
                     for (RouteNodeInstanceDTO nextNodeInstanceVO : nodeInstanceDTO.getNextNodes()) {
 405  
                             // recurse to populate nextNodeInstances
 406  0
                             nextNodeInstances.add(_importRouteNodeInstanceDTO(nextNodeInstanceVO));
 407  
                     }
 408  
 
 409  0
                     routeNodeInstances.put(nodeInstance.getRouteNodeInstanceId(), nodeInstance);
 410  0
                     return nodeInstance;
 411  
             }
 412  
             
 413  
                 /**
 414  
                  * This method returns a dummy RouteNodeInstance for the given ID, or null if it hasn't
 415  
                  * imported from a RouteNodeInstanceDTO with that ID
 416  
                  * 
 417  
                  * @see org.kuali.rice.kew.dto.DTOConverter.RouteNodeInstanceLoader#load(java.lang.Long)
 418  
                  */
 419  
                 @Override
 420  
                 public RouteNodeInstance load(Long routeNodeInstanceID) {
 421  0
                         return routeNodeInstances.get(routeNodeInstanceID);
 422  
                 }
 423  
 
 424  
 
 425  
             /**
 426  
              * This method creates bogus BranchES as needed
 427  
              * 
 428  
              * @param branchId
 429  
              * @return
 430  
              */
 431  
             private Branch getBranch(Long branchId) {
 432  0
                     Branch result = null;
 433  
 
 434  0
                     if (branchId != null) {
 435  
                             // if branch doesn't exist, create it
 436  0
                             if (!branches.containsKey(branchId)) {
 437  0
                                     result = new Branch();
 438  0
                                     result.setBranchId(branchId);
 439  0
                                     branches.put(branchId, result);
 440  
                             } else {
 441  0
                                     result = branches.get(branchId);
 442  
                             }
 443  
                     }
 444  0
                     return result;
 445  
             }
 446  
 
 447  
             /**
 448  
              * This method creates bogus RouteNodeS as needed
 449  
              * 
 450  
              * @param routeNodeId
 451  
              * @return
 452  
              */
 453  
             private RouteNode getRouteNode(Long routeNodeId) {
 454  0
                     RouteNode result = null;
 455  
 
 456  0
                     if (routeNodeId != null) {
 457  
                             // if RouteNode doesn't exist, create it
 458  0
                             if (!routeNodes.containsKey(routeNodeId)) {
 459  0
                                     result = new RouteNode();
 460  0
                                     result.setRouteNodeId(routeNodeId);
 461  0
                                     routeNodes.put(routeNodeId, result);
 462  
                             } else {
 463  0
                                     result = routeNodes.get(routeNodeId);
 464  
                             }
 465  
                     }
 466  0
                     return result;
 467  
             }
 468  
 
 469  
             /**
 470  
              * This method creates bogus RouteNodeInstanceS as needed
 471  
              * 
 472  
              * @param routeNodeInstanceId
 473  
              * @return
 474  
              */
 475  
             public RouteNodeInstance getRouteNodeInstance(Long routeNodeInstanceId) {
 476  0
                     RouteNodeInstance result = null;
 477  
 
 478  0
                     if (routeNodeInstanceId != null) {
 479  
                             // if RouteNodeInstance doesn't exist, create it
 480  0
                             if (!routeNodeInstances.containsKey(routeNodeInstanceId)) {
 481  0
                                     result = new RouteNodeInstance();
 482  0
                                     result.setRouteNodeInstanceId(routeNodeInstanceId);
 483  0
                                     routeNodeInstances.put(routeNodeInstanceId, result);
 484  
                             } else {
 485  0
                                     result = routeNodeInstances.get(routeNodeInstanceId);
 486  
                             }
 487  
                     }
 488  0
                     return result;
 489  
             }
 490  
 
 491  
             /**
 492  
              * This method creates bogus NodeStateS as needed
 493  
              * 
 494  
              * @param nodeStateId
 495  
              * @return
 496  
              */
 497  
             private NodeState getNodeState(Long nodeStateId) {
 498  0
                     NodeState result = null;
 499  
 
 500  0
                     if (nodeStateId != null) {
 501  
                             // if NodeState doesn't exist, create it
 502  0
                             if (!nodeStates.containsKey(nodeStateId)) {
 503  0
                                     result = new NodeState();
 504  0
                                     result.setNodeStateId(nodeStateId);
 505  0
                                     nodeStates.put(nodeStateId, result);
 506  
                             } else {
 507  0
                                     result = nodeStates.get(nodeStateId);
 508  
                             }
 509  
                     }
 510  0
                     return result;
 511  
             }
 512  
 
 513  
     } // end inner class FutureRouteNodeInstanceFabricator
 514  
 
 515  
     /**
 516  
      * Logs a new message to the route log for the current document, then refreshes the action taken list to display
 517  
      * back the new message in the route log tab. User must have permission to log a message for the doc type and the
 518  
      * request must be coming from the route log tab display (not the route log page).
 519  
      */
 520  
         public ActionForward logActionMessageInRouteLog(ActionMapping mapping, ActionForm form, HttpServletRequest request,
 521  
                         HttpServletResponse response) throws Exception {
 522  0
                 RouteLogForm routeLogForm = (RouteLogForm) form;
 523  
 
 524  0
                 Long routeHeaderId = null;
 525  0
                 if (!org.apache.commons.lang.StringUtils.isEmpty(routeLogForm.getRouteHeaderId())) {
 526  0
                         routeHeaderId = new Long(routeLogForm.getRouteHeaderId());
 527  0
                 } else if (!org.apache.commons.lang.StringUtils.isEmpty(routeLogForm.getDocId())) {
 528  0
                         routeHeaderId = new Long(routeLogForm.getDocId());
 529  
                 } else {
 530  0
                         throw new WorkflowRuntimeException("No paramater provided to fetch document");
 531  
                 }
 532  
                 
 533  0
                 DocumentRouteHeaderValue routeHeader = KEWServiceLocator.getRouteHeaderService().getRouteHeader(routeHeaderId);
 534  
                 
 535  
                 // check user has permission to add a route log message
 536  0
                 boolean isAuthorizedToAddRouteLogMessage = KEWServiceLocator.getDocumentTypePermissionService()
 537  
                                 .canAddRouteLogMessage(GlobalVariables.getUserSession().getPrincipalId(), routeHeader);
 538  
 
 539  0
                 if (!isAuthorizedToAddRouteLogMessage) {
 540  0
                         throw new InvalidActionTakenException("Principal with name '"
 541  
                                         + GlobalVariables.getUserSession().getPrincipalName()
 542  
                                         + "' is not authorized to add route log messages for documents of type '"
 543  
                                         + routeHeader.getDocumentType().getName());
 544  
                 }
 545  
 
 546  0
                 LOG.info("Logging new action message for user " + GlobalVariables.getUserSession().getPrincipalName()
 547  
                                 + ", route header id " + routeHeader);
 548  0
                 KEWServiceLocator.getWorkflowDocumentService().logDocumentAction(
 549  
                                 GlobalVariables.getUserSession().getPrincipalId(), routeHeader,
 550  
                                 routeLogForm.getNewRouteLogActionMessage());
 551  
 
 552  0
                 routeLogForm.setNewRouteLogActionMessage("");
 553  
 
 554  
                 // retrieve routeHeader again to pull new action taken
 555  0
                 routeHeader = KEWServiceLocator.getRouteHeaderService().getRouteHeader(routeHeaderId, true);
 556  0
                 fixActionRequestsPositions(routeHeader);
 557  0
                 request.setAttribute("routeHeader", routeHeader);
 558  
 
 559  0
                 return mapping.findForward(getDefaultMapping());
 560  
         }
 561  
     
 562  
 }