Coverage Report - org.kuali.rice.kew.routeheader.DocumentRouteHeaderValue
 
Classes in this File Line Coverage Branch Coverage Complexity
DocumentRouteHeaderValue
0%
0/343
0%
0/124
1.786
 
 1  
 /*
 2  
  * Copyright 2005-2007 The Kuali Foundation
 3  
  *
 4  
  * Licensed under the Educational Community License, Version 2.0 (the "License");
 5  
  * you may not use this file except in compliance with the License.
 6  
  * You may obtain a copy of the License at
 7  
  *
 8  
  * http://www.opensource.org/licenses/ecl2.php
 9  
  *
 10  
  * Unless required by applicable law or agreed to in writing, software
 11  
  * distributed under the License is distributed on an "AS IS" BASIS,
 12  
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  
  * See the License for the specific language governing permissions and
 14  
  * limitations under the License.
 15  
  */
 16  
 package org.kuali.rice.kew.routeheader;
 17  
 
 18  
 import org.apache.commons.lang.ObjectUtils;
 19  
 import org.apache.commons.lang.StringUtils;
 20  
 import org.apache.log4j.Logger;
 21  
 import org.hibernate.annotations.Fetch;
 22  
 import org.hibernate.annotations.FetchMode;
 23  
 import org.hibernate.annotations.GenericGenerator;
 24  
 import org.hibernate.annotations.Parameter;
 25  
 import org.kuali.rice.core.util.KeyValue;
 26  
 import org.kuali.rice.kew.actionitem.ActionItem;
 27  
 import org.kuali.rice.kew.actionlist.CustomActionListAttribute;
 28  
 import org.kuali.rice.kew.actionlist.DefaultCustomActionListAttribute;
 29  
 import org.kuali.rice.kew.actionrequest.ActionRequestFactory;
 30  
 import org.kuali.rice.kew.actionrequest.ActionRequestValue;
 31  
 import org.kuali.rice.kew.actiontaken.ActionTakenValue;
 32  
 import org.kuali.rice.kew.docsearch.SearchableAttributeValue;
 33  
 import org.kuali.rice.kew.doctype.ApplicationDocumentStatus;
 34  
 import org.kuali.rice.kew.doctype.DocumentTypePolicy;
 35  
 import org.kuali.rice.kew.doctype.bo.DocumentType;
 36  
 import org.kuali.rice.kew.dto.DTOConverter;
 37  
 import org.kuali.rice.kew.dto.RouteHeaderDTO;
 38  
 import org.kuali.rice.kew.engine.CompatUtils;
 39  
 import org.kuali.rice.kew.engine.node.Branch;
 40  
 import org.kuali.rice.kew.engine.node.BranchState;
 41  
 import org.kuali.rice.kew.engine.node.RouteNode;
 42  
 import org.kuali.rice.kew.engine.node.RouteNodeInstance;
 43  
 import org.kuali.rice.kew.exception.InvalidActionTakenException;
 44  
 import org.kuali.rice.kew.exception.ResourceUnavailableException;
 45  
 import org.kuali.rice.kew.exception.WorkflowException;
 46  
 import org.kuali.rice.kew.exception.WorkflowRuntimeException;
 47  
 import org.kuali.rice.kew.mail.CustomEmailAttribute;
 48  
 import org.kuali.rice.kew.mail.CustomEmailAttributeImpl;
 49  
 import org.kuali.rice.kew.notes.CustomNoteAttribute;
 50  
 import org.kuali.rice.kew.notes.CustomNoteAttributeImpl;
 51  
 import org.kuali.rice.kew.notes.Note;
 52  
 import org.kuali.rice.kew.service.KEWServiceLocator;
 53  
 import org.kuali.rice.kew.util.CodeTranslator;
 54  
 import org.kuali.rice.kew.util.KEWConstants;
 55  
 import org.kuali.rice.kim.bo.entity.KimPrincipal;
 56  
 import org.kuali.rice.kns.bo.PersistableBusinessObjectBase;
 57  
 
 58  
 import javax.persistence.*;
 59  
 import java.sql.Timestamp;
 60  
 import java.util.*;
 61  
 
 62  
 
 63  
 
 64  
 /**
 65  
  * A document within KEW.  A document effectively represents a process that moves through
 66  
  * the workflow engine.  It is created from a particular {@link DocumentType} and follows
 67  
  * the route path defined by that DocumentType.
 68  
  *
 69  
  * <p>During a document's lifecycle it progresses through a series of statuses, starting
 70  
  * with INITIATED and moving to one of the terminal states (such as FINAL, CANCELED, etc).
 71  
  * The list of status on a document are defined in the {@link KEWConstants} class and
 72  
  * include the constants starting with "ROUTE_HEADER_" and ending with "_CD".
 73  
  *
 74  
  * <p>Associated with the document is the document content.  The document content is XML
 75  
  * which represents the content of that document.  This XML content is typically used
 76  
  * to make routing decisions for the document.
 77  
  *
 78  
  * <p>A document has associated with it a set of {@link ActionRequestValue} object and
 79  
  * {@link ActionTakenValue} objects.  Action Requests represent requests for user
 80  
  * action (such as Approve, Acknowledge, etc).  Action Takens represent action that
 81  
  * users have performed on the document, such as approvals or cancelling of the document.
 82  
  *
 83  
  * <p>The instantiated route path of a document is defined by it's graph of
 84  
  * {@link RouteNodeInstance} objects.  The path starts at the initial node of the document
 85  
  * and progresses from there following the next nodes of each node instance.  The current
 86  
  * active nodes on the document are defined by the "active" flag on the node instance
 87  
  * where are not marked as "complete".
 88  
  *
 89  
  * @see DocumentType
 90  
  * @see ActionRequestValue
 91  
  * @see ActionItem
 92  
  * @see ActionTakenValue
 93  
  * @see RouteNodeInstance
 94  
  * @see KEWConstants
 95  
  *
 96  
  * @author Kuali Rice Team (rice.collab@kuali.org)
 97  
  */
 98  
 @Entity
 99  
 @Table(name="KREW_DOC_HDR_T")
 100  
 //@Sequence(name="KREW_DOC_HDR_S", property="routeHeaderId")
 101  
 @NamedQueries({
 102  
         @NamedQuery(name="DocumentRouteHeaderValue.FindByRouteHeaderId", query="select d from DocumentRouteHeaderValue as d where d.routeHeaderId = :routeHeaderId"),
 103  
         @NamedQuery(name="DocumentRouteHeaderValue.QuickLinks.FindWatchedDocumentsByInitiatorWorkflowId", query="SELECT NEW org.kuali.rice.kew.quicklinks.WatchedDocument(routeHeaderId, docRouteStatus, docTitle) FROM DocumentRouteHeaderValue WHERE initiatorWorkflowId = :initiatorWorkflowId AND docRouteStatus IN ('"+ KEWConstants.ROUTE_HEADER_ENROUTE_CD +"','"+ KEWConstants.ROUTE_HEADER_EXCEPTION_CD +"') ORDER BY createDate DESC"),
 104  
         @NamedQuery(name="DocumentRouteHeaderValue.GetAppDocId", query="SELECT d.appDocId from DocumentRouteHeaderValue as d where d.routeHeaderId = :routeHeaderId")
 105  
 })
 106  
 public class DocumentRouteHeaderValue extends PersistableBusinessObjectBase {
 107  
     private static final long serialVersionUID = -4700736340527913220L;
 108  0
     private static final Logger LOG = Logger.getLogger(DocumentRouteHeaderValue.class);
 109  
 
 110  
     public static final String CURRENT_ROUTE_NODE_NAME_DELIMITER = ", ";
 111  
     
 112  
     @Column(name="DOC_TYP_ID")
 113  
         private java.lang.Long documentTypeId;
 114  
     @Column(name="DOC_HDR_STAT_CD")
 115  
         private java.lang.String docRouteStatus;
 116  
     @Column(name="RTE_LVL")
 117  
         private java.lang.Integer docRouteLevel;
 118  
         @Column(name="STAT_MDFN_DT")
 119  
         private java.sql.Timestamp statusModDate;
 120  
         @Column(name="CRTE_DT")
 121  
         private java.sql.Timestamp createDate;
 122  
         @Column(name="APRV_DT")
 123  
         private java.sql.Timestamp approvedDate;
 124  
         @Column(name="FNL_DT")
 125  
         private java.sql.Timestamp finalizedDate;
 126  
     @Transient
 127  
     private DocumentRouteHeaderValueContent documentContent;
 128  
     @Column(name="TTL")
 129  
         private java.lang.String docTitle;
 130  
     @Column(name="APP_DOC_ID")
 131  
         private java.lang.String appDocId;
 132  0
     @Column(name="DOC_VER_NBR")
 133  
     private java.lang.Integer docVersion = new Integer(KEWConstants.DOCUMENT_VERSION_NODAL);
 134  
     @Column(name="INITR_PRNCPL_ID")
 135  
         private java.lang.String initiatorWorkflowId;
 136  
     @Column(name="RTE_PRNCPL_ID")
 137  
         private java.lang.String routedByUserWorkflowId;
 138  
         @Column(name="RTE_STAT_MDFN_DT")
 139  
         private java.sql.Timestamp routeStatusDate;
 140  
         @Column(name="RTE_LVL_MDFN_DT")
 141  
         private java.sql.Timestamp routeLevelDate;
 142  
     @Column(name="APP_DOC_STAT")
 143  
         private java.lang.String appDocStatus;
 144  
         @Column(name="APP_DOC_STAT_MDFN_DT")
 145  
         private java.sql.Timestamp appDocStatusDate;
 146  
         
 147  
     @Id
 148  
     @GeneratedValue(generator="KREW_DOC_HDR_S")
 149  
         @GenericGenerator(name="KREW_DOC_HDR_S",strategy="org.hibernate.id.enhanced.SequenceStyleGenerator",parameters={
 150  
                         @Parameter(name="sequence_name",value="KREW_DOC_HDR_S"),
 151  
                         @Parameter(name="value_column",value="id")
 152  
         })
 153  
         @Column(name="DOC_HDR_ID")
 154  
         private java.lang.Long routeHeaderId;
 155  
     
 156  
     //@OneToMany(fetch=FetchType.EAGER, cascade=CascadeType.REMOVE, mappedBy="routeHeader")
 157  
     //@Fetch(value = FetchMode.SELECT)
 158  
     //private List<ActionRequestValue> actionRequests = new ArrayList<ActionRequestValue>();
 159  
     
 160  
     //@OneToMany(fetch=FetchType.EAGER, cascade=CascadeType.REMOVE, mappedBy="routeHeader")
 161  
     //@OrderBy("actionDate ASC")
 162  
     //@Fetch(value = FetchMode.SELECT)
 163  
     //private List<ActionTakenValue> actionsTaken = new ArrayList<ActionTakenValue>();
 164  
     
 165  
     //@OneToMany(fetch=FetchType.EAGER, cascade=CascadeType.REMOVE, mappedBy="routeHeader")
 166  
     //@Fetch(value = FetchMode.SELECT)
 167  
     //private List<ActionItem> actionItems = new ArrayList<ActionItem>();
 168  
     
 169  
     /**
 170  
      * The appDocStatusHistory keeps a list of Application Document Status transitions
 171  
      * for the document.  It tracks the previous status, the new status, and a timestamp of the 
 172  
      * transition for each status transition.
 173  
      */
 174  0
     @OneToMany(fetch=FetchType.EAGER, cascade={CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE}, mappedBy="routeHeaderId")
 175  
     //@JoinColumn(referencedColumnName="DOC_HDR_ID")
 176  
     @OrderBy("statusTransitionId ASC")
 177  
     @Fetch(value = FetchMode.SELECT)
 178  
     private List<DocumentStatusTransition> appDocStatusHistory = new ArrayList<DocumentStatusTransition>();
 179  
     
 180  0
     @OneToMany(fetch=FetchType.LAZY, cascade={CascadeType.PERSIST, CascadeType.REMOVE})
 181  
     @JoinColumn(name="DOC_HDR_ID")
 182  
     @OrderBy("noteId ASC")
 183  
     private List<Note> notes = new ArrayList<Note>();
 184  
     
 185  0
     @Transient
 186  
     private List<SearchableAttributeValue> searchableAttributeValues = new ArrayList<SearchableAttributeValue>();
 187  0
     @Transient
 188  
     private Collection queueItems = new ArrayList();
 189  0
     @Transient
 190  
     private boolean routingReport = false;
 191  
     @Transient
 192  
     private List<ActionRequestValue> simulatedActionRequests;
 193  
     
 194  
     private static final boolean FINAL_STATE = true;
 195  
     protected static final HashMap<String,String> legalActions;
 196  
     protected static final HashMap<String,String> stateTransitionMap;
 197  
 
 198  
     /* New Workflow 2.1 Field */
 199  0
     @ManyToMany(fetch=FetchType.EAGER, cascade=CascadeType.REMOVE)
 200  
     @JoinTable(name = "KREW_INIT_RTE_NODE_INSTN_T", joinColumns = @JoinColumn(name = "DOC_HDR_ID"), inverseJoinColumns = @JoinColumn(name = "RTE_NODE_INSTN_ID")) 
 201  
     @Fetch(value = FetchMode.SELECT)
 202  
     private List<RouteNodeInstance> initialRouteNodeInstances = new ArrayList<RouteNodeInstance>();
 203  
 
 204  
     static {
 205  0
         stateTransitionMap = new HashMap<String,String>();
 206  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_INITIATED_CD, KEWConstants.ROUTE_HEADER_SAVED_CD + KEWConstants.ROUTE_HEADER_ENROUTE_CD + KEWConstants.ROUTE_HEADER_CANCEL_CD);
 207  
 
 208  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_SAVED_CD, KEWConstants.ROUTE_HEADER_SAVED_CD + KEWConstants.ROUTE_HEADER_ENROUTE_CD + KEWConstants.ROUTE_HEADER_CANCEL_CD + KEWConstants.ROUTE_HEADER_APPROVED_CD);
 209  
 
 210  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_ENROUTE_CD, KEWConstants.ROUTE_HEADER_DISAPPROVED_CD +
 211  
                 KEWConstants.ROUTE_HEADER_CANCEL_CD + KEWConstants.ROUTE_HEADER_APPROVED_CD + KEWConstants.ROUTE_HEADER_EXCEPTION_CD + KEWConstants.ROUTE_HEADER_SAVED_CD);
 212  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_DISAPPROVED_CD, "");
 213  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_CANCEL_CD, "");
 214  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_FINAL_CD, "");
 215  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_EXCEPTION_CD, KEWConstants.ROUTE_HEADER_EXCEPTION_CD + KEWConstants.ROUTE_HEADER_ENROUTE_CD + KEWConstants.ROUTE_HEADER_CANCEL_CD + KEWConstants.ROUTE_HEADER_APPROVED_CD + KEWConstants.ROUTE_HEADER_DISAPPROVED_CD + KEWConstants.ROUTE_HEADER_SAVED_CD);
 216  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_APPROVED_CD, KEWConstants.ROUTE_HEADER_APPROVED_CD + KEWConstants.ROUTE_HEADER_PROCESSED_CD + KEWConstants.ROUTE_HEADER_EXCEPTION_CD);
 217  0
         stateTransitionMap.put(KEWConstants.ROUTE_HEADER_PROCESSED_CD, KEWConstants.ROUTE_HEADER_FINAL_CD + KEWConstants.ROUTE_HEADER_PROCESSED_CD);
 218  
 
 219  0
         legalActions = new HashMap<String,String>();
 220  0
         legalActions.put(KEWConstants.ROUTE_HEADER_INITIATED_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_SAVED_CD + KEWConstants.ACTION_TAKEN_COMPLETED_CD + KEWConstants.ACTION_TAKEN_ROUTED_CD + KEWConstants.ACTION_TAKEN_CANCELED_CD + KEWConstants.ACTION_TAKEN_ADHOC_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD + KEWConstants.ACTION_TAKEN_BLANKET_APPROVE_CD + KEWConstants.ACTION_TAKEN_MOVE_CD);
 221  0
         legalActions.put(KEWConstants.ROUTE_HEADER_SAVED_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_SAVED_CD + KEWConstants.ACTION_TAKEN_COMPLETED_CD + KEWConstants.ACTION_TAKEN_ROUTED_CD + KEWConstants.ACTION_TAKEN_APPROVED_CD + KEWConstants.ACTION_TAKEN_CANCELED_CD + KEWConstants.ACTION_TAKEN_ADHOC_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD + KEWConstants.ACTION_TAKEN_BLANKET_APPROVE_CD + KEWConstants.ACTION_TAKEN_MOVE_CD);
 222  
         /* ACTION_TAKEN_ROUTED_CD not included in enroute state
 223  
          * ACTION_TAKEN_SAVED_CD removed as of version 2.4
 224  
          */
 225  0
         legalActions.put(KEWConstants.ROUTE_HEADER_ENROUTE_CD, /*KEWConstants.ACTION_TAKEN_SAVED_CD + KEWConstants.ACTION_TAKEN_ROUTED_CD + */KEWConstants.ACTION_TAKEN_APPROVED_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ADHOC_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD + KEWConstants.ACTION_TAKEN_BLANKET_APPROVE_CD + KEWConstants.ACTION_TAKEN_CANCELED_CD + KEWConstants.ACTION_TAKEN_COMPLETED_CD + KEWConstants.ACTION_TAKEN_DENIED_CD + KEWConstants.ACTION_TAKEN_SU_APPROVED_CD + KEWConstants.ACTION_TAKEN_SU_CANCELED_CD + KEWConstants.ACTION_TAKEN_SU_DISAPPROVED_CD + KEWConstants.ACTION_TAKEN_SU_ROUTE_LEVEL_APPROVED_CD + KEWConstants.ACTION_TAKEN_RETURNED_TO_PREVIOUS_CD + KEWConstants.ACTION_TAKEN_SU_RETURNED_TO_PREVIOUS_CD + KEWConstants.ACTION_TAKEN_MOVE_CD);
 226  
         /* ACTION_TAKEN_ROUTED_CD not included in exception state
 227  
          * ACTION_TAKEN_SAVED_CD removed as of version 2.4.2
 228  
          */
 229  0
         legalActions.put(KEWConstants.ROUTE_HEADER_EXCEPTION_CD, /*KEWConstants.ACTION_TAKEN_SAVED_CD + */KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD + KEWConstants.ACTION_TAKEN_APPROVED_CD + KEWConstants.ACTION_TAKEN_BLANKET_APPROVE_CD + KEWConstants.ACTION_TAKEN_CANCELED_CD + KEWConstants.ACTION_TAKEN_COMPLETED_CD + KEWConstants.ACTION_TAKEN_DENIED_CD + KEWConstants.ACTION_TAKEN_SU_APPROVED_CD + KEWConstants.ACTION_TAKEN_SU_CANCELED_CD + KEWConstants.ACTION_TAKEN_SU_DISAPPROVED_CD + KEWConstants.ACTION_TAKEN_SU_ROUTE_LEVEL_APPROVED_CD + KEWConstants.ACTION_TAKEN_RETURNED_TO_PREVIOUS_CD + KEWConstants.ACTION_TAKEN_SU_RETURNED_TO_PREVIOUS_CD + KEWConstants.ACTION_TAKEN_MOVE_CD);
 230  0
         legalActions.put(KEWConstants.ROUTE_HEADER_FINAL_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD);
 231  0
         legalActions.put(KEWConstants.ROUTE_HEADER_CANCEL_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD);
 232  0
         legalActions.put(KEWConstants.ROUTE_HEADER_DISAPPROVED_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD);
 233  0
         legalActions.put(KEWConstants.ROUTE_HEADER_APPROVED_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD);
 234  0
         legalActions.put(KEWConstants.ROUTE_HEADER_PROCESSED_CD, KEWConstants.ACTION_TAKEN_FYI_CD + KEWConstants.ACTION_TAKEN_ACKNOWLEDGED_CD + KEWConstants.ACTION_TAKEN_ADHOC_REVOKED_CD);
 235  0
     }
 236  
 
 237  0
     public DocumentRouteHeaderValue() {
 238  0
     }
 239  
 
 240  
     public KimPrincipal getInitiatorPrincipal() {
 241  
             // if we are running a simulation, there will be no initiator
 242  0
             if (getInitiatorWorkflowId() == null) {
 243  0
                     return null;
 244  
             }
 245  0
             return KEWServiceLocator.getIdentityHelperService().getPrincipal(getInitiatorWorkflowId());
 246  
     }
 247  
     
 248  
     public KimPrincipal getRoutedByPrincipal()
 249  
     {
 250  0
             if (getRoutedByUserWorkflowId() == null) {
 251  0
                     return null;
 252  
             }
 253  0
             return KEWServiceLocator.getIdentityHelperService().getPrincipal(getRoutedByUserWorkflowId());
 254  
            }
 255  
         
 256  
     public String getInitiatorDisplayName() {
 257  0
         return KEWServiceLocator.getIdentityHelperService().getPerson(getInitiatorWorkflowId()).getName();
 258  
     }
 259  
 
 260  
     public String getRoutedByDisplayName() {
 261  0
             return KEWServiceLocator.getIdentityHelperService().getPerson(getRoutedByUserWorkflowId()).getName();
 262  
     }
 263  
         
 264  
     public String getCurrentRouteLevelName() {
 265  0
         String name = "Not Found";
 266  
         // TODO the isRouteLevelDocument junk can be ripped out
 267  0
             if(routingReport){
 268  0
                     name = "Routing Report";
 269  0
             } else if (CompatUtils.isRouteLevelDocument(this)) {
 270  0
             int routeLevelInt = getDocRouteLevel().intValue();
 271  0
             LOG.info("Getting current route level name for a Route level document: " + routeLevelInt+CURRENT_ROUTE_NODE_NAME_DELIMITER+routeHeaderId);
 272  0
             List routeLevelNodes = CompatUtils.getRouteLevelCompatibleNodeList(getDocumentType());
 273  0
             LOG.info("Route level compatible node list has " + routeLevelNodes.size() + " nodes");
 274  0
             if (routeLevelInt < routeLevelNodes.size()) {
 275  0
                 name = ((RouteNode)routeLevelNodes.get(routeLevelInt)).getRouteNodeName();
 276  
             }
 277  0
         } else {
 278  0
                 name = "";
 279  0
                 for (Iterator<String> iterator = getCurrentNodeNames().iterator(); iterator.hasNext();) {
 280  0
                         String nodeName = iterator.next();
 281  0
                         name += nodeName + (iterator.hasNext() ? CURRENT_ROUTE_NODE_NAME_DELIMITER : "");
 282  0
                 }
 283  
         }
 284  0
         return name;
 285  
     }
 286  
     
 287  
     public List<String> getCurrentNodeNames() {
 288  0
             List<String> currentNodeNames = new ArrayList<String>();
 289  0
             Collection<RouteNodeInstance> nodeInstances = KEWServiceLocator.getRouteNodeService().getActiveNodeInstances(getRouteHeaderId());
 290  0
         if (nodeInstances.isEmpty()) {
 291  0
             nodeInstances = KEWServiceLocator.getRouteNodeService().getTerminalNodeInstances(getRouteHeaderId());
 292  
         }
 293  0
         for (RouteNodeInstance nodeInstance : nodeInstances) {
 294  0
             currentNodeNames.add(nodeInstance.getRouteNode().getRouteNodeName());
 295  
         }
 296  0
         return currentNodeNames;
 297  
     }
 298  
 
 299  
     public String getRouteStatusLabel() {
 300  0
         return CodeTranslator.getRouteStatusLabel(getDocRouteStatus());
 301  
     }
 302  
 
 303  
     public String getDocRouteStatusLabel() {
 304  0
         return CodeTranslator.getRouteStatusLabel(getDocRouteStatus());
 305  
     }
 306  
     /**
 307  
      * 
 308  
      * This method returns the Document Status Policy for the document type associated with this Route Header.
 309  
      * The Document Status Policy denotes whether the KEW Route Status, or the Application Document Status,
 310  
      * or both are to be displayed.
 311  
      * 
 312  
      * @return
 313  
      */
 314  
     public String getDocStatusPolicy() {
 315  0
             return getDocumentType().getDocumentStatusPolicy().getPolicyStringValue();
 316  
     }
 317  
     
 318  
     public Collection getQueueItems() {
 319  0
         return queueItems;
 320  
     }
 321  
 
 322  
     public void setQueueItems(Collection queueItems) {
 323  0
         this.queueItems = queueItems;
 324  0
     }
 325  
 
 326  
     public List<ActionItem> getActionItems() {
 327  0
         return (List<ActionItem>) KEWServiceLocator.getActionListService().findByRouteHeaderId(routeHeaderId);
 328  
     }
 329  
 
 330  
     public List<ActionTakenValue> getActionsTaken() {
 331  0
         return (List<ActionTakenValue>) KEWServiceLocator.getActionTakenService().findByRouteHeaderId(routeHeaderId);
 332  
     }
 333  
 
 334  
     public List<ActionRequestValue> getActionRequests() {
 335  0
             if (this.simulatedActionRequests == null || this.simulatedActionRequests.isEmpty()) {
 336  0
                     return KEWServiceLocator.getActionRequestService().findAllActionRequestsByRouteHeaderId(routeHeaderId);
 337  
             } else {
 338  0
                     return this.simulatedActionRequests;
 339  
             }
 340  
     }
 341  
 
 342  
     public List<ActionRequestValue> getSimulatedActionRequests() {
 343  0
             if (this.simulatedActionRequests == null) {
 344  0
                     this.simulatedActionRequests = new ArrayList<ActionRequestValue>();
 345  
             }
 346  0
                 return this.simulatedActionRequests;
 347  
         }
 348  
 
 349  
         public void setSimulatedActionRequests(List<ActionRequestValue> simulatedActionRequests) {
 350  0
                 this.simulatedActionRequests = simulatedActionRequests;
 351  0
         }
 352  
 
 353  
         public DocumentType getDocumentType() {
 354  0
             return KEWServiceLocator.getDocumentTypeService().findById(getDocumentTypeId());
 355  
     }
 356  
 
 357  
     public java.lang.String getAppDocId() {
 358  0
         return appDocId;
 359  
     }
 360  
 
 361  
     public void setAppDocId(java.lang.String appDocId) {
 362  0
         this.appDocId = appDocId;
 363  0
     }
 364  
 
 365  
     public java.sql.Timestamp getApprovedDate() {
 366  0
         return approvedDate;
 367  
     }
 368  
 
 369  
     public void setApprovedDate(java.sql.Timestamp approvedDate) {
 370  0
         this.approvedDate = approvedDate;
 371  0
     }
 372  
 
 373  
     public java.sql.Timestamp getCreateDate() {
 374  0
         return createDate;
 375  
     }
 376  
 
 377  
     public void setCreateDate(java.sql.Timestamp createDate) {
 378  0
         this.createDate = createDate;
 379  0
     }
 380  
 
 381  
     public java.lang.String getDocContent() {
 382  0
             return getDocumentContent().getDocumentContent();
 383  
     }
 384  
 
 385  
     public void setDocContent(java.lang.String docContent) {
 386  0
             DocumentRouteHeaderValueContent content = getDocumentContent();
 387  0
             content.setDocumentContent(docContent);
 388  0
     }
 389  
 
 390  
     public java.lang.Integer getDocRouteLevel() {
 391  0
         return docRouteLevel;
 392  
     }
 393  
 
 394  
     public void setDocRouteLevel(java.lang.Integer docRouteLevel) {
 395  0
         this.docRouteLevel = docRouteLevel;
 396  0
     }
 397  
 
 398  
     public java.lang.String getDocRouteStatus() {
 399  0
         return docRouteStatus;
 400  
     }
 401  
 
 402  
     public void setDocRouteStatus(java.lang.String docRouteStatus) {
 403  0
         this.docRouteStatus = docRouteStatus;
 404  0
     }
 405  
 
 406  
     public java.lang.String getDocTitle() {
 407  0
         return docTitle;
 408  
     }
 409  
 
 410  
     public void setDocTitle(java.lang.String docTitle) {
 411  0
         this.docTitle = docTitle;
 412  0
     }
 413  
 
 414  
     public java.lang.Long getDocumentTypeId() {
 415  0
         return documentTypeId;
 416  
     }
 417  
 
 418  
     public void setDocumentTypeId(java.lang.Long docTypeId) {
 419  0
         this.documentTypeId = docTypeId;
 420  0
     }
 421  
 
 422  
     public java.lang.Integer getDocVersion() {
 423  0
         return docVersion;
 424  
     }
 425  
 
 426  
     public void setDocVersion(java.lang.Integer docVersion) {
 427  0
         this.docVersion = docVersion;
 428  0
     }
 429  
 
 430  
     public java.sql.Timestamp getFinalizedDate() {
 431  0
         return finalizedDate;
 432  
     }
 433  
 
 434  
     public void setFinalizedDate(java.sql.Timestamp finalizedDate) {
 435  0
         this.finalizedDate = finalizedDate;
 436  0
     }
 437  
 
 438  
     public java.lang.String getInitiatorWorkflowId() {
 439  0
         return initiatorWorkflowId;
 440  
     }
 441  
 
 442  
     public void setInitiatorWorkflowId(java.lang.String initiatorWorkflowId) {
 443  0
         this.initiatorWorkflowId = initiatorWorkflowId;
 444  0
     }
 445  
 
 446  
     public java.lang.String getRoutedByUserWorkflowId() {
 447  0
         if ( (isEnroute()) && (StringUtils.isBlank(routedByUserWorkflowId)) ) {
 448  0
             return initiatorWorkflowId;
 449  
         }
 450  0
         return routedByUserWorkflowId;
 451  
     }
 452  
 
 453  
     public void setRoutedByUserWorkflowId(java.lang.String routedByUserWorkflowId) {
 454  0
         this.routedByUserWorkflowId = routedByUserWorkflowId;
 455  0
     }
 456  
 
 457  
     public java.lang.Long getRouteHeaderId() {
 458  0
         return routeHeaderId;
 459  
     }
 460  
 
 461  
     public void setRouteHeaderId(java.lang.Long routeHeaderId) {
 462  0
         this.routeHeaderId = routeHeaderId;
 463  0
     }
 464  
 
 465  
     public java.sql.Timestamp getRouteLevelDate() {
 466  0
         return routeLevelDate;
 467  
     }
 468  
 
 469  
     public void setRouteLevelDate(java.sql.Timestamp routeLevelDate) {
 470  0
         this.routeLevelDate = routeLevelDate;
 471  0
     }
 472  
 
 473  
     public java.sql.Timestamp getRouteStatusDate() {
 474  0
         return routeStatusDate;
 475  
     }
 476  
 
 477  
     public void setRouteStatusDate(java.sql.Timestamp routeStatusDate) {
 478  0
         this.routeStatusDate = routeStatusDate;
 479  0
     }
 480  
 
 481  
     public java.sql.Timestamp getStatusModDate() {
 482  0
         return statusModDate;
 483  
     }
 484  
 
 485  
     public void setStatusModDate(java.sql.Timestamp statusModDate) {
 486  0
         this.statusModDate = statusModDate;
 487  0
     }
 488  
 
 489  
     /**
 490  
      * 
 491  
      * This method returns the Application Document Status.
 492  
      * This status is an alternative to the Route Status that may be used for a document.
 493  
      * It is configurable per document type.
 494  
      * 
 495  
      * @see ApplicationDocumentStatus
 496  
      * @see DocumentTypePolicy
 497  
      * 
 498  
      * @return
 499  
      */
 500  
     public java.lang.String getAppDocStatus() {
 501  0
             if (appDocStatus == null || "".equals(appDocStatus)){
 502  0
                     return KEWConstants.UNKNOWN_STATUS;
 503  
             }
 504  0
         return appDocStatus;
 505  
     }
 506  
 
 507  
     public void setAppDocStatus(java.lang.String appDocStatus){
 508  0
         this.appDocStatus = appDocStatus;
 509  0
     }
 510  
     
 511  
     /**
 512  
      * 
 513  
      * This method returns a combination of the route status label and the app doc status.
 514  
      * 
 515  
      * @return
 516  
      */
 517  
     public String getCombinedStatus(){
 518  0
             String routeStatus = getRouteStatusLabel();
 519  0
             String appStatus = getAppDocStatus();
 520  0
             if (routeStatus != null && routeStatus.length()>0){
 521  0
                     if (appStatus.length() > 0){
 522  0
                             routeStatus += ", "+appStatus;
 523  
                     }
 524  
             } else {
 525  0
                     return appStatus;
 526  
             }
 527  0
             return routeStatus;
 528  
     }
 529  
 
 530  
     /**
 531  
      * 
 532  
      * This method sets the appDocStatus.
 533  
      * It firsts validates the new value against the defined acceptable values, if defined.
 534  
      * It also updates the AppDocStatus date, and saves the status transition information
 535  
      * 
 536  
      * @param appDocStatus
 537  
      * @throws WorkflowRuntimeException
 538  
      */
 539  
     public void updateAppDocStatus(java.lang.String appDocStatus) throws WorkflowRuntimeException{
 540  
                //validate against allowable values if defined
 541  0
             if (appDocStatus != null && appDocStatus.length() > 0 && !appDocStatus.equalsIgnoreCase(this.appDocStatus)){                    
 542  0
                     DocumentType documentType = KEWServiceLocator.getDocumentTypeService().findById(this.getDocumentTypeId());            
 543  0
                     if (documentType.getValidApplicationStatuses() != null  && documentType.getValidApplicationStatuses().size() > 0){
 544  0
                             Iterator<ApplicationDocumentStatus> iter = documentType.getValidApplicationStatuses().iterator();
 545  0
                             boolean statusValidated = false;
 546  0
                             while (iter.hasNext())
 547  
                             {
 548  0
                                     ApplicationDocumentStatus myAppDocStat = iter.next();
 549  0
                                     if (appDocStatus.compareToIgnoreCase(myAppDocStat.getStatusName()) == 0)
 550  
                                     {
 551  0
                                             statusValidated = true;
 552  0
                                             break;
 553  
                                     }
 554  0
                             }
 555  0
                             if (!statusValidated){
 556  0
                                     WorkflowRuntimeException xpee = new WorkflowRuntimeException("AppDocStatus value " +  appDocStatus + " not allowable."); 
 557  0
                                     LOG.error("Error validating nextAppDocStatus name: " +  appDocStatus + " against acceptable values.", xpee);
 558  0
                                     throw xpee; 
 559  
                             }
 560  
                     }
 561  
 
 562  
                     // set the status value
 563  0
                     String oldStatus = this.appDocStatus;
 564  0
                     this.appDocStatus = appDocStatus;
 565  
 
 566  
                     // update the timestamp
 567  0
                     setAppDocStatusDate(new Timestamp(System.currentTimeMillis()));
 568  
 
 569  
                     // save the status transition
 570  0
                     this.appDocStatusHistory.add(new DocumentStatusTransition(routeHeaderId, oldStatus, appDocStatus));
 571  
             }
 572  
 
 573  0
     }
 574  
     
 575  
     
 576  
     public java.sql.Timestamp getAppDocStatusDate() {
 577  0
         return appDocStatusDate;
 578  
     }
 579  
 
 580  
     public void setAppDocStatusDate(java.sql.Timestamp appDocStatusDate) {
 581  0
         this.appDocStatusDate = appDocStatusDate;
 582  0
     }
 583  
 
 584  
     public Object copy(boolean preserveKeys) {
 585  0
         throw new UnsupportedOperationException("The copy method is deprecated and unimplemented!");
 586  
     }
 587  
 
 588  
     /**
 589  
      * @return True if the document is in the state of Initiated
 590  
      */
 591  
     public boolean isStateInitiated() {
 592  0
         return KEWConstants.ROUTE_HEADER_INITIATED_CD.equals(docRouteStatus);
 593  
     }
 594  
 
 595  
     /**
 596  
      * @return True if the document is in the state of Saved
 597  
      */
 598  
     public boolean isStateSaved() {
 599  0
         return KEWConstants.ROUTE_HEADER_SAVED_CD.equals(docRouteStatus);
 600  
     }
 601  
 
 602  
     /**
 603  
      * @return true if the document has ever been inte enroute state
 604  
      */
 605  
     public boolean isRouted() {
 606  0
         return !(isStateInitiated() || isStateSaved());
 607  
     }
 608  
 
 609  
     public boolean isInException() {
 610  0
         return KEWConstants.ROUTE_HEADER_EXCEPTION_CD.equals(docRouteStatus);
 611  
     }
 612  
 
 613  
     public boolean isDisaproved() {
 614  0
         return KEWConstants.ROUTE_HEADER_DISAPPROVED_CD.equals(docRouteStatus);
 615  
     }
 616  
 
 617  
     public boolean isCanceled() {
 618  0
         return KEWConstants.ROUTE_HEADER_CANCEL_CD.equals(docRouteStatus);
 619  
     }
 620  
 
 621  
     public boolean isFinal() {
 622  0
         return KEWConstants.ROUTE_HEADER_FINAL_CD.equals(docRouteStatus);
 623  
     }
 624  
 
 625  
     public boolean isEnroute() {
 626  0
             return KEWConstants.ROUTE_HEADER_ENROUTE_CD.equals(docRouteStatus);
 627  
     }
 628  
 
 629  
     /**
 630  
      * @return true if the document is in the processed state
 631  
      */
 632  
     public boolean isProcessed() {
 633  0
         return KEWConstants.ROUTE_HEADER_PROCESSED_CD.equals(docRouteStatus);
 634  
     }
 635  
 
 636  
     /**
 637  
      * @return true if the document is in the approved state
 638  
      */
 639  
     public boolean isApproved() {
 640  0
         return KEWConstants.ROUTE_HEADER_APPROVED_CD.equals(docRouteStatus);
 641  
     }
 642  
 
 643  
     public boolean isRoutable() {
 644  0
         return KEWConstants.ROUTE_HEADER_ENROUTE_CD.equals(docRouteStatus) ||
 645  
                         //KEWConstants.ROUTE_HEADER_EXCEPTION_CD.equals(docRouteStatus) ||
 646  
                         KEWConstants.ROUTE_HEADER_SAVED_CD.equals(docRouteStatus) ||
 647  
                         KEWConstants.ROUTE_HEADER_APPROVED_CD.equals(docRouteStatus) ||
 648  
                         KEWConstants.ROUTE_HEADER_PROCESSED_CD.equals(docRouteStatus);
 649  
     }
 650  
 
 651  
     /**
 652  
      * Return true if the given action code is valid for this document's current state.
 653  
      *
 654  
      * @param actionCd
 655  
      *            The action code to be tested.
 656  
      * @return True if the action code is valid for the document's status.
 657  
      */
 658  
     public boolean isValidActionToTake(String actionCd) {
 659  0
         String actions = (String) legalActions.get(docRouteStatus);
 660  0
         if (!actions.contains(actionCd)) {
 661  0
             return false;
 662  
         } else {
 663  0
             return true;
 664  
         }
 665  
     }
 666  
 
 667  
     public boolean isValidStatusChange(String newStatus) {
 668  0
             return ((String) stateTransitionMap.get(getDocRouteStatus())).contains(newStatus);
 669  
     }
 670  
 
 671  
     public void setRouteStatus(String newStatus, boolean finalState) throws InvalidActionTakenException {
 672  0
         if (newStatus != getDocRouteStatus()) {
 673  
             // only modify the status mod date if the status actually changed
 674  0
             setRouteStatusDate(new Timestamp(System.currentTimeMillis()));
 675  
         }
 676  0
         if (((String) stateTransitionMap.get(getDocRouteStatus())).contains(newStatus)) {
 677  0
             LOG.debug("changing status");
 678  0
             setDocRouteStatus(newStatus);
 679  
         } else {
 680  0
             LOG.debug("unable to change status");
 681  0
             throw new InvalidActionTakenException("Document status " + CodeTranslator.getRouteStatusLabel(getDocRouteStatus()) + " cannot transition to status " + CodeTranslator.getRouteStatusLabel(newStatus));
 682  
         }
 683  0
         setStatusModDate(new Timestamp(System.currentTimeMillis()));
 684  0
         if (finalState) {
 685  0
             LOG.debug("setting final timeStamp");
 686  0
             setFinalizedDate(new Timestamp(System.currentTimeMillis()));
 687  
         }
 688  0
     }
 689  
 
 690  
     /**
 691  
      * Mark the document as being processed.
 692  
      *
 693  
      * @throws ResourceUnavailableException
 694  
      * @throws InvalidActionTakenException
 695  
      */
 696  
     public void markDocumentProcessed() throws InvalidActionTakenException {
 697  0
         LOG.debug(this + " marked processed");
 698  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_PROCESSED_CD, !FINAL_STATE);
 699  0
     }
 700  
 
 701  
     /**
 702  
      * Mark document cancled.
 703  
      *
 704  
      * @throws ResourceUnavailableException
 705  
      * @throws InvalidActionTakenException
 706  
      */
 707  
     public void markDocumentCanceled() throws InvalidActionTakenException {
 708  0
         LOG.debug(this + " marked canceled");
 709  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_CANCEL_CD, FINAL_STATE);
 710  0
     }
 711  
 
 712  
     /**
 713  
      * Mark document disapproved
 714  
      *
 715  
      * @throws ResourceUnavailableException
 716  
      * @throws InvalidActionTakenException
 717  
      */
 718  
     public void markDocumentDisapproved() throws InvalidActionTakenException {
 719  0
         LOG.debug(this + " marked disapproved");
 720  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_DISAPPROVED_CD, FINAL_STATE);
 721  0
     }
 722  
 
 723  
     /**
 724  
      * Mark document saved
 725  
      *
 726  
      * @throws ResourceUnavailableException
 727  
      * @throws InvalidActionTakenException
 728  
      */
 729  
     public void markDocumentSaved() throws InvalidActionTakenException {
 730  0
         LOG.debug(this + " marked saved");
 731  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_SAVED_CD, !FINAL_STATE);
 732  0
     }
 733  
 
 734  
     /**
 735  
      * Mark the document as being approved.
 736  
      *
 737  
      * @throws ResourceUnavailableException
 738  
      * @throws InvalidActionTakenException
 739  
      */
 740  
     public void markDocumentApproved() throws InvalidActionTakenException {
 741  0
         LOG.debug(this + " marked approved");
 742  0
         setApprovedDate(new Timestamp(System.currentTimeMillis()));
 743  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_APPROVED_CD, !FINAL_STATE);
 744  0
     }
 745  
 
 746  
     /**
 747  
      * Mark the document as being in the exception state.
 748  
      *
 749  
      * @throws ResourceUnavailableException
 750  
      * @throws InvalidActionTakenException
 751  
      */
 752  
     public void markDocumentInException() throws InvalidActionTakenException {
 753  0
         LOG.debug(this + " marked in exception");
 754  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_EXCEPTION_CD, !FINAL_STATE);
 755  0
     }
 756  
 
 757  
     /**
 758  
      * Mark the document as being actively routed.
 759  
      *
 760  
      * @throws ResourceUnavailableException
 761  
      * @throws InvalidActionTakenException
 762  
      */
 763  
     public void markDocumentEnroute() throws InvalidActionTakenException {
 764  0
         LOG.debug(this + " marked enroute");
 765  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_ENROUTE_CD, !FINAL_STATE);
 766  0
     }
 767  
 
 768  
     /**
 769  
      * Mark document finalized.
 770  
      *
 771  
      * @throws ResourceUnavailableException
 772  
      * @throws InvalidActionTakenException
 773  
      */
 774  
     public void markDocumentFinalized() throws InvalidActionTakenException {
 775  0
         LOG.debug(this + " marked finalized");
 776  0
         setRouteStatus(KEWConstants.ROUTE_HEADER_FINAL_CD, FINAL_STATE);
 777  0
     }
 778  
 
 779  
     /**
 780  
      * This method takes data from a VO and sets it on this route header
 781  
      * @param routeHeaderVO
 782  
      * @throws WorkflowException
 783  
      */
 784  
     public void setRouteHeaderData(RouteHeaderDTO routeHeaderVO) throws WorkflowException {
 785  0
         if (!ObjectUtils.equals(getDocTitle(), routeHeaderVO.getDocTitle())) {
 786  0
                 KEWServiceLocator.getActionListService().updateActionItemsForTitleChange(getRouteHeaderId(), routeHeaderVO.getDocTitle());
 787  
         }
 788  0
         setDocTitle(routeHeaderVO.getDocTitle());
 789  0
         setAppDocId(routeHeaderVO.getAppDocId());
 790  0
         setStatusModDate(new Timestamp(System.currentTimeMillis()));
 791  0
         updateAppDocStatus(routeHeaderVO.getAppDocStatus());
 792  
 
 793  
         /* set the variables from the routeHeaderVO */
 794  0
         List<KeyValue> variables = routeHeaderVO.getVariables();
 795  0
         for (KeyValue kvp : variables) {
 796  0
             setVariable(kvp.getKey(), kvp.getValue());
 797  
         }
 798  0
     }
 799  
 
 800  
     /**
 801  
      * Convenience method that returns the branch of the first (and presumably only?) initial node
 802  
      * @return the branch of the first (and presumably only?) initial node
 803  
      */
 804  
     public Branch getRootBranch() {
 805  0
         if (!this.initialRouteNodeInstances.isEmpty()) {
 806  0
             return ((RouteNodeInstance) getInitialRouteNodeInstance(0)).getBranch();
 807  
         } 
 808  0
         return null;
 809  
     }
 810  
 
 811  
     /**
 812  
      * Looks up a variable (embodied in a "BranchState" key/value pair) in the
 813  
      * branch state table.
 814  
      */
 815  
     private BranchState findVariable(String name) {
 816  0
         Branch rootBranch = getRootBranch();
 817  0
         if (rootBranch != null) {
 818  0
             List<BranchState> branchState = rootBranch.getBranchState();
 819  0
             Iterator<BranchState> it = branchState.iterator();
 820  0
             while (it.hasNext()) {
 821  0
                 BranchState state = it.next();
 822  0
                 if (ObjectUtils.equals(state.getKey(), BranchState.VARIABLE_PREFIX + name)) {
 823  0
                     return state;
 824  
                 }
 825  0
             }
 826  
         }
 827  0
         return null;
 828  
     }
 829  
 
 830  
     /**
 831  
      * Gets a variable
 832  
      * @param name variable name
 833  
      * @return variable value, or null if variable is not defined
 834  
      */
 835  
     public String getVariable(String name) {
 836  0
         BranchState state = findVariable(name);
 837  0
         if (state == null) {
 838  0
             LOG.debug("Variable not found: '" + name + "'");
 839  0
             return null;
 840  
         }
 841  0
         return state.getValue();
 842  
     }
 843  
 
 844  
     public void removeVariableThatContains(String name) {
 845  0
         List<BranchState> statesToRemove = new ArrayList<BranchState>();
 846  0
         for (BranchState state : this.getRootBranchState()) {
 847  0
             if (state.getKey().contains(name)) {
 848  0
                 statesToRemove.add(state);
 849  
             }
 850  
         }
 851  0
         this.getRootBranchState().removeAll(statesToRemove);
 852  0
     }
 853  
     
 854  
     /**
 855  
      * Sets a variable
 856  
      * @param name variable name
 857  
      * @param value variable value, or null if variable should be removed
 858  
      */
 859  
     public void setVariable(String name, String value) {
 860  0
         BranchState state = findVariable(name);
 861  0
         Branch rootBranch = getRootBranch();
 862  0
         if (rootBranch != null) {
 863  0
             List<BranchState> branchState = rootBranch.getBranchState();
 864  0
             if (state == null) {
 865  0
                 if (value == null) {
 866  0
                     LOG.debug("set non existent variable '" + name + "' to null value");
 867  0
                     return;
 868  
                 }
 869  0
                 LOG.debug("Adding branch state: '" + name + "'='" + value + "'");
 870  0
                 state = new BranchState();
 871  0
                 state.setBranch(rootBranch);
 872  0
                 state.setKey(BranchState.VARIABLE_PREFIX + name);
 873  0
                 state.setValue(value);
 874  0
                 rootBranch.addBranchState(state);
 875  
             } else {
 876  0
                 if (value == null) {
 877  0
                     LOG.debug("Removing value: " + state.getKey() + "=" + state.getValue());
 878  0
                     branchState.remove(state);
 879  
                 } else {
 880  0
                     LOG.debug("Setting value of variable '" + name + "' to '" + value + "'");
 881  0
                     state.setValue(value);
 882  
                 }
 883  
             }
 884  
         }
 885  0
     }
 886  
     
 887  
     public List<BranchState> getRootBranchState() {
 888  0
         if (this.getRootBranch() != null) {
 889  0
             return this.getRootBranch().getBranchState();
 890  
         }
 891  0
         return null;
 892  
     }
 893  
 
 894  
     public CustomActionListAttribute getCustomActionListAttribute() throws WorkflowException {
 895  0
         CustomActionListAttribute customActionListAttribute = null;
 896  0
         if (this.getDocumentType() != null) {
 897  0
                 customActionListAttribute = this.getDocumentType().getCustomActionListAttribute();
 898  0
                 if (customActionListAttribute != null) {
 899  0
                         return customActionListAttribute;
 900  
                 }
 901  
         }
 902  0
         customActionListAttribute = new DefaultCustomActionListAttribute();
 903  0
         return customActionListAttribute;
 904  
     }
 905  
 
 906  
     public CustomEmailAttribute getCustomEmailAttribute() throws WorkflowException {
 907  0
         CustomEmailAttribute customEmailAttribute = null;
 908  
         try {
 909  0
             if (this.getDocumentType() != null) {
 910  0
                 customEmailAttribute = this.getDocumentType().getCustomEmailAttribute();
 911  0
                 if (customEmailAttribute != null) {
 912  0
                     customEmailAttribute.setRouteHeaderVO(DTOConverter.convertRouteHeader(this, null));
 913  0
                     return customEmailAttribute;
 914  
                 }
 915  
             }
 916  0
         } catch (Exception e) {
 917  0
             LOG.debug("Error in retrieving custom email attribute", e);
 918  0
         }
 919  0
         customEmailAttribute = new CustomEmailAttributeImpl();
 920  0
         customEmailAttribute.setRouteHeaderVO(DTOConverter.convertRouteHeader(this, null));
 921  0
         return customEmailAttribute;
 922  
     }
 923  
 
 924  
     public CustomNoteAttribute getCustomNoteAttribute() throws WorkflowException
 925  
     {
 926  0
         CustomNoteAttribute customNoteAttribute = null;
 927  
         try {
 928  0
             if (this.getDocumentType() != null) {
 929  0
                 customNoteAttribute = this.getDocumentType().getCustomNoteAttribute();
 930  0
                 if (customNoteAttribute != null) {
 931  0
                     customNoteAttribute.setRouteHeaderVO(DTOConverter.convertRouteHeader(this, null));
 932  0
                     return customNoteAttribute;
 933  
                 }
 934  
             }
 935  0
         } catch (Exception e) {
 936  0
             LOG.debug("Error in retrieving custom note attribute", e);
 937  0
         }
 938  0
         customNoteAttribute = new CustomNoteAttributeImpl();
 939  0
         customNoteAttribute.setRouteHeaderVO(DTOConverter.convertRouteHeader(this, null));
 940  0
         return customNoteAttribute;
 941  
     }
 942  
 
 943  
     public ActionRequestValue getDocActionRequest(int index) {
 944  0
             List<ActionRequestValue> actionRequests = getActionRequests();
 945  0
         while (actionRequests.size() <= index) {
 946  0
                 ActionRequestValue actionRequest = new ActionRequestFactory(this).createBlankActionRequest();
 947  0
                 actionRequest.setNodeInstance(new RouteNodeInstance());
 948  0
             actionRequests.add(actionRequest);
 949  0
         }
 950  0
         return (ActionRequestValue) actionRequests.get(index);
 951  
     }
 952  
 
 953  
     public ActionTakenValue getDocActionTaken(int index) {
 954  0
             List<ActionTakenValue> actionsTaken = getActionsTaken();
 955  0
         while (actionsTaken.size() <= index) {
 956  0
             actionsTaken.add(new ActionTakenValue());
 957  
         }
 958  0
         return (ActionTakenValue) actionsTaken.get(index);
 959  
     }
 960  
 
 961  
     public ActionItem getDocActionItem(int index) {
 962  0
             List<ActionItem> actionItems = getActionItems();
 963  0
         while (actionItems.size() <= index) {
 964  0
             actionItems.add(new ActionItem());
 965  
         }
 966  0
         return (ActionItem) actionItems.get(index);
 967  
     }
 968  
 
 969  
     private RouteNodeInstance getInitialRouteNodeInstance(int index) {
 970  0
             if (initialRouteNodeInstances.size() >= index) {
 971  0
                 return (RouteNodeInstance) initialRouteNodeInstances.get(index);
 972  
             } 
 973  0
             return null;
 974  
     }
 975  
 
 976  
 //        /**
 977  
 //         * @param searchableAttributeValues The searchableAttributeValues to set.
 978  
 //         */
 979  
 //        public void setSearchableAttributeValues(List<SearchableAttributeValue> searchableAttributeValues) {
 980  
 //                this.searchableAttributeValues = searchableAttributeValues;
 981  
 //        }
 982  
 //
 983  
 //        /**
 984  
 //         * @return Returns the searchableAttributeValues.
 985  
 //         */
 986  
 //        public List<SearchableAttributeValue> getSearchableAttributeValues() {
 987  
 //                return searchableAttributeValues;
 988  
 //        }
 989  
 
 990  
         public boolean isRoutingReport() {
 991  0
                 return routingReport;
 992  
         }
 993  
 
 994  
         public void setRoutingReport(boolean routingReport) {
 995  0
                 this.routingReport = routingReport;
 996  0
         }
 997  
 
 998  
     public List<RouteNodeInstance> getInitialRouteNodeInstances() {
 999  0
         return initialRouteNodeInstances;
 1000  
     }
 1001  
 
 1002  
     public void setInitialRouteNodeInstances(List<RouteNodeInstance> initialRouteNodeInstances) {
 1003  0
         this.initialRouteNodeInstances = initialRouteNodeInstances;
 1004  0
     }
 1005  
 
 1006  
         public List<Note> getNotes() {
 1007  0
                 return notes;
 1008  
         }
 1009  
 
 1010  
         public void setNotes(List<Note> notes) {
 1011  0
                 this.notes = notes;
 1012  0
         }
 1013  
 
 1014  
         public DocumentRouteHeaderValueContent getDocumentContent() {
 1015  0
                 if (documentContent == null) {
 1016  0
                         documentContent = KEWServiceLocator.getRouteHeaderService().getContent(getRouteHeaderId());
 1017  
                 }
 1018  0
                 return documentContent;
 1019  
         }
 1020  
 
 1021  
         public void setDocumentContent(DocumentRouteHeaderValueContent documentContent) {
 1022  0
                 this.documentContent = documentContent;
 1023  0
         }
 1024  
 
 1025  
         public List<DocumentStatusTransition> getAppDocStatusHistory() {
 1026  0
                 return this.appDocStatusHistory;
 1027  
         }
 1028  
 
 1029  
         public void setAppDocStatusHistory(
 1030  
                         List<DocumentStatusTransition> appDocStatusHistory) {
 1031  0
                 this.appDocStatusHistory = appDocStatusHistory;
 1032  0
         }
 1033  
 }