Coverage Report - org.kuali.rice.kns.service.impl.DocumentServiceImpl
 
Classes in this File Line Coverage Branch Coverage Complexity
DocumentServiceImpl
0%
0/403
0%
0/138
3.333
 
 1  
 /*
 2  
  * Copyright 2006-2011 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  
 
 17  
 package org.kuali.rice.kns.service.impl;
 18  
 
 19  
 import org.apache.commons.lang.StringUtils;
 20  
 import org.apache.commons.lang.time.StopWatch;
 21  
 import org.kuali.rice.core.api.datetime.DateTimeService;
 22  
 import org.kuali.rice.core.api.config.ConfigurationException;
 23  
 import org.kuali.rice.core.api.config.property.ConfigurationService;
 24  
 import org.kuali.rice.core.framework.persistence.jta.TransactionalNoValidationExceptionRollback;
 25  
 import org.kuali.rice.core.util.RiceKeyConstants;
 26  
 import org.kuali.rice.kew.exception.WorkflowException;
 27  
 import org.kuali.rice.kim.bo.Person;
 28  
 import org.kuali.rice.kim.service.KIMServiceLocator;
 29  
 import org.kuali.rice.kim.service.PersonService;
 30  
 import org.kuali.rice.kns.UserSession;
 31  
 import org.kuali.rice.kns.bo.AdHocRoutePerson;
 32  
 import org.kuali.rice.kns.bo.AdHocRouteRecipient;
 33  
 import org.kuali.rice.kns.bo.AdHocRouteWorkgroup;
 34  
 import org.kuali.rice.kns.bo.DocumentHeader;
 35  
 import org.kuali.rice.kns.bo.Note;
 36  
 import org.kuali.rice.kns.bo.PersistableBusinessObject;
 37  
 import org.kuali.rice.kns.dao.DocumentDao;
 38  
 import org.kuali.rice.kns.datadictionary.exception.UnknownDocumentTypeException;
 39  
 import org.kuali.rice.kns.document.Document;
 40  
 import org.kuali.rice.kns.document.MaintenanceDocument;
 41  
 import org.kuali.rice.kns.document.MaintenanceDocumentBase;
 42  
 import org.kuali.rice.kns.document.authorization.DocumentAuthorizer;
 43  
 import org.kuali.rice.kns.document.authorization.DocumentPresentationController;
 44  
 import org.kuali.rice.kns.exception.DocumentAuthorizationException;
 45  
 import org.kuali.rice.kns.exception.ValidationException;
 46  
 import org.kuali.rice.kns.rule.event.ApproveDocumentEvent;
 47  
 import org.kuali.rice.kns.rule.event.BlanketApproveDocumentEvent;
 48  
 import org.kuali.rice.kns.rule.event.KualiDocumentEvent;
 49  
 import org.kuali.rice.kns.rule.event.RouteDocumentEvent;
 50  
 import org.kuali.rice.kns.rule.event.SaveDocumentEvent;
 51  
 import org.kuali.rice.kns.rule.event.SaveEvent;
 52  
 import org.kuali.rice.kns.service.BusinessObjectService;
 53  
 import org.kuali.rice.kns.service.DataDictionaryService;
 54  
 import org.kuali.rice.kns.service.DocumentHeaderService;
 55  
 import org.kuali.rice.kns.service.DocumentHelperService;
 56  
 import org.kuali.rice.kns.service.DocumentService;
 57  
 import org.kuali.rice.kns.service.KNSServiceLocator;
 58  
 import org.kuali.rice.kns.service.KNSServiceLocatorInternal;
 59  
 import org.kuali.rice.kns.service.KNSServiceLocatorWeb;
 60  
 import org.kuali.rice.kns.service.NoteService;
 61  
 import org.kuali.rice.kns.util.GlobalVariables;
 62  
 import org.kuali.rice.kns.util.KNSConstants;
 63  
 import org.kuali.rice.kns.util.ObjectUtils;
 64  
 import org.kuali.rice.kns.workflow.service.KualiWorkflowDocument;
 65  
 import org.kuali.rice.kns.workflow.service.WorkflowDocumentService;
 66  
 import org.springframework.dao.OptimisticLockingFailureException;
 67  
 
 68  
 import java.lang.reflect.Constructor;
 69  
 import java.lang.reflect.InvocationTargetException;
 70  
 import java.text.MessageFormat;
 71  
 import java.util.ArrayList;
 72  
 import java.util.HashMap;
 73  
 import java.util.List;
 74  
 import java.util.Map;
 75  
 
 76  
 
 77  
 
 78  
 /**
 79  
  * This class is the service implementation for the Document structure. It contains all of the document level type of processing and
 80  
  * calling back into documents for various centralization of functionality. This is the default, Kuali delivered implementation
 81  
  * which utilizes Workflow.
 82  
  */
 83  
 @TransactionalNoValidationExceptionRollback
 84  0
 public class DocumentServiceImpl implements DocumentService {
 85  0
     private static org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(DocumentServiceImpl.class);
 86  
 
 87  
     private DateTimeService dateTimeService;
 88  
 
 89  
     private NoteService noteService;
 90  
 
 91  
     protected WorkflowDocumentService workflowDocumentService;
 92  
 
 93  
     protected BusinessObjectService businessObjectService;
 94  
 
 95  
     /**
 96  
      * Don't access directly, use synchronized getter and setter to access
 97  
      */
 98  
     private DocumentDao documentDao;
 99  
 
 100  
     private DataDictionaryService dataDictionaryService;
 101  
 
 102  
     private DocumentHeaderService documentHeaderService;
 103  
 
 104  
     private PersonService personService;
 105  
 
 106  
     private DocumentHelperService documentHelperService;
 107  
 
 108  
     private ConfigurationService kualiConfigurationService;
 109  
 
 110  
     /**
 111  
      * @see org.kuali.rice.kns.service.DocumentService#saveDocument(org.kuali.rice.kns.document.Document)
 112  
      */
 113  
     @Override
 114  
         public Document saveDocument(Document document) throws WorkflowException, ValidationException {
 115  0
         return saveDocument(document, SaveDocumentEvent.class);
 116  
     }
 117  
 
 118  
     @Override
 119  
         public Document saveDocument(Document document, Class<? extends KualiDocumentEvent> kualiDocumentEventClass) throws WorkflowException, ValidationException {
 120  0
         checkForNulls(document);
 121  0
         if (kualiDocumentEventClass == null) {
 122  0
             throw new IllegalArgumentException("invalid (null) kualiDocumentEventClass");
 123  
         }
 124  
         // if event is not an instance of a SaveDocumentEvent or a SaveOnlyDocumentEvent
 125  0
         if (!SaveEvent.class.isAssignableFrom(kualiDocumentEventClass)) {
 126  0
             throw new ConfigurationException("The KualiDocumentEvent class '" + kualiDocumentEventClass.getName() + "' does not implement the class '" + SaveEvent.class.getName() + "'");
 127  
         }
 128  
 //        if (!getDocumentActionFlags(document).getCanSave()) {
 129  
 //            throw buildAuthorizationException("save", document);
 130  
 //        }
 131  0
         document.prepareForSave();
 132  0
         Document savedDocument = validateAndPersistDocumentAndSaveAdHocRoutingRecipients(document, generateKualiDocumentEvent(document, kualiDocumentEventClass));
 133  0
         prepareWorkflowDocument(savedDocument);
 134  0
         getWorkflowDocumentService().save(savedDocument.getDocumentHeader().getWorkflowDocument(), null);
 135  
         
 136  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),savedDocument.getDocumentHeader().getWorkflowDocument());
 137  
 
 138  0
         return savedDocument;
 139  
     }
 140  
 
 141  
     private KualiDocumentEvent generateKualiDocumentEvent(Document document, Class<? extends KualiDocumentEvent> eventClass) throws ConfigurationException {
 142  0
             String potentialErrorMessage = "Found error trying to generate Kuali Document Event using event class '" + 
 143  
             eventClass.getName() + "' for document " + document.getDocumentNumber();
 144  
             
 145  
             try {
 146  0
                     Constructor<?> usableConstructor = null;
 147  0
                     List<Object> paramList = new ArrayList<Object>();
 148  0
                     for (Constructor<?> currentConstructor : eventClass.getConstructors()) {
 149  0
                             for (Class<?> parameterClass : currentConstructor.getParameterTypes()) {
 150  0
                                     if (Document.class.isAssignableFrom(parameterClass)) {
 151  0
                                             usableConstructor = currentConstructor;
 152  0
                                             paramList.add(document);
 153  
                                     } else {
 154  0
                                             paramList.add(null);
 155  
                                     }
 156  
                             }
 157  0
                             if (ObjectUtils.isNotNull(usableConstructor)) {
 158  0
                                     break;
 159  
                             }
 160  
                     }
 161  0
                     if (usableConstructor == null) {
 162  0
                             throw new RuntimeException("Cannot find a constructor for class '" + eventClass.getName() + "' that takes in a document parameter");
 163  
                     }
 164  0
                     return (KualiDocumentEvent) usableConstructor.newInstance(paramList.toArray());
 165  0
             } catch (SecurityException e) {
 166  0
                     throw new ConfigurationException(potentialErrorMessage, e);
 167  0
             } catch (IllegalArgumentException e) {
 168  0
                     throw new ConfigurationException(potentialErrorMessage, e);
 169  0
             } catch (InstantiationException e) {
 170  0
                     throw new ConfigurationException(potentialErrorMessage, e);
 171  0
             } catch (IllegalAccessException e) {
 172  0
                     throw new ConfigurationException(potentialErrorMessage, e);
 173  0
             } catch (InvocationTargetException e) {
 174  0
                     throw new ConfigurationException(potentialErrorMessage, e);
 175  
             }
 176  
     }
 177  
 
 178  
     /**
 179  
      * @see org.kuali.rice.kns.service.DocumentService#routeDocument(org.kuali.rice.kns.document.Document, java.lang.String, java.util.List)
 180  
      */
 181  
     @Override
 182  
         public Document routeDocument(Document document, String annotation, List<AdHocRouteRecipient> adHocRecipients) throws ValidationException, WorkflowException {
 183  0
         checkForNulls(document);
 184  
         //if (!getDocumentActionFlags(document).getCanRoute()) {
 185  
         //    throw buildAuthorizationException("route", document);
 186  
         //}
 187  0
         document.prepareForSave();
 188  0
         Document savedDocument = validateAndPersistDocument(document, new RouteDocumentEvent(document));
 189  0
         prepareWorkflowDocument(savedDocument);
 190  0
         getWorkflowDocumentService().route(savedDocument.getDocumentHeader().getWorkflowDocument(), annotation, adHocRecipients);
 191  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),savedDocument.getDocumentHeader().getWorkflowDocument());
 192  0
         removeAdHocPersonsAndWorkgroups(savedDocument);
 193  0
         return savedDocument;
 194  
     }
 195  
 
 196  
     /**
 197  
      * @see org.kuali.rice.kns.service.DocumentService#approveDocument(org.kuali.rice.kns.document.Document, java.lang.String,
 198  
      *      java.util.List)
 199  
      */
 200  
     @Override
 201  
         public Document approveDocument(Document document, String annotation, List<AdHocRouteRecipient> adHocRecipients) throws ValidationException, WorkflowException {
 202  0
         checkForNulls(document);
 203  
         //if (!getDocumentActionFlags(document).getCanApprove()) {
 204  
         //    throw buildAuthorizationException("approve", document);
 205  
         //}
 206  0
         document.prepareForSave();
 207  0
         Document savedDocument = validateAndPersistDocument(document, new ApproveDocumentEvent(document));
 208  0
         prepareWorkflowDocument(savedDocument);
 209  0
         getWorkflowDocumentService().approve(savedDocument.getDocumentHeader().getWorkflowDocument(), annotation, adHocRecipients);
 210  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),savedDocument.getDocumentHeader().getWorkflowDocument());
 211  0
         removeAdHocPersonsAndWorkgroups(savedDocument);
 212  0
         return savedDocument;
 213  
     }
 214  
 
 215  
 
 216  
     /**
 217  
      * @see org.kuali.rice.kns.service.DocumentService#superUserApproveDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 218  
      */
 219  
     @Override
 220  
         public Document superUserApproveDocument(Document document, String annotation) throws WorkflowException {
 221  0
         getDocumentDao().save(document);
 222  0
         prepareWorkflowDocument(document);
 223  0
         getWorkflowDocumentService().superUserApprove(document.getDocumentHeader().getWorkflowDocument(), annotation);
 224  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 225  0
         removeAdHocPersonsAndWorkgroups(document);
 226  0
         return document;
 227  
     }
 228  
 
 229  
     /**
 230  
      * @see org.kuali.rice.kns.service.DocumentService#superUserCancelDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 231  
      */
 232  
     @Override
 233  
         public Document superUserCancelDocument(Document document, String annotation) throws WorkflowException {
 234  0
         getDocumentDao().save(document);
 235  0
         prepareWorkflowDocument(document);
 236  0
         getWorkflowDocumentService().superUserCancel(document.getDocumentHeader().getWorkflowDocument(), annotation);
 237  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 238  0
         removeAdHocPersonsAndWorkgroups(document);
 239  0
         return document;
 240  
     }
 241  
 
 242  
     /**
 243  
      * @see org.kuali.rice.kns.service.DocumentService#superUserCancelDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 244  
      */
 245  
     @Override
 246  
         public Document superUserDisapproveDocument(Document document, String annotation) throws WorkflowException {
 247  0
         getDocumentDao().save(document);
 248  0
         prepareWorkflowDocument(document);
 249  0
         getWorkflowDocumentService().superUserDisapprove(document.getDocumentHeader().getWorkflowDocument(), annotation);
 250  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 251  0
         removeAdHocPersonsAndWorkgroups(document);
 252  0
         return document;
 253  
     }
 254  
 
 255  
     /**
 256  
      * @see org.kuali.rice.kns.service.DocumentService#disapproveDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 257  
      */
 258  
     @Override
 259  
         public Document disapproveDocument(Document document, String annotation) throws Exception {
 260  0
         checkForNulls(document);
 261  
 
 262  0
         Note note = createNoteFromDocument(document,annotation);
 263  0
         document.addNote(note);
 264  
 
 265  
         //SAVE THE NOTE
 266  
         //Note: This save logic is replicated here and in KualiDocumentAction, when to save (based on doc state) should be moved
 267  
         //      into a doc service method
 268  0
         getNoteService().save(note);
 269  
 
 270  0
         prepareWorkflowDocument(document);
 271  0
         getWorkflowDocumentService().disapprove(document.getDocumentHeader().getWorkflowDocument(), annotation);
 272  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 273  0
         removeAdHocPersonsAndWorkgroups(document);
 274  0
         return document;
 275  
     }
 276  
 
 277  
     /**
 278  
      * @see org.kuali.rice.kns.service.DocumentService#cancelDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 279  
      */
 280  
     @Override
 281  
         public Document cancelDocument(Document document, String annotation) throws WorkflowException {
 282  0
         checkForNulls(document);
 283  
         //if (!getDocumentActionFlags(document).getCanCancel()) {
 284  
         //    throw buildAuthorizationException("cancel", document);
 285  
         //}
 286  0
         if (document instanceof MaintenanceDocument) {
 287  0
                 MaintenanceDocument maintDoc = ((MaintenanceDocument) document);
 288  0
                 if (maintDoc.getOldMaintainableObject() != null) {
 289  0
                         maintDoc.getOldMaintainableObject().getBusinessObject().refresh();
 290  
                 }
 291  0
                        maintDoc.getNewMaintainableObject().getBusinessObject().refresh();
 292  
         }
 293  0
         prepareWorkflowDocument(document);
 294  0
         getWorkflowDocumentService().cancel(document.getDocumentHeader().getWorkflowDocument(), annotation);
 295  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 296  
         //getBusinessObjectService().delete(document.getAdHocRoutePersons());
 297  
         //getBusinessObjectService().delete(document.getAdHocRouteWorkgroups());
 298  0
         removeAdHocPersonsAndWorkgroups(document);
 299  0
         return document;
 300  
     }
 301  
 
 302  
     /**
 303  
      * @see org.kuali.rice.kns.service.DocumentService#acknowledgeDocument(org.kuali.rice.kns.document.Document, java.lang.String,
 304  
      *      java.util.List)
 305  
      */
 306  
     @Override
 307  
         public Document acknowledgeDocument(Document document, String annotation, List<AdHocRouteRecipient> adHocRecipients) throws WorkflowException {
 308  0
         checkForNulls(document);
 309  
         //if (!getDocumentActionFlags(document).getCanAcknowledge()) {
 310  
         //    throw buildAuthorizationException("acknowledge", document);
 311  
         //}
 312  0
         prepareWorkflowDocument(document);
 313  0
         getWorkflowDocumentService().acknowledge(document.getDocumentHeader().getWorkflowDocument(), annotation, adHocRecipients);
 314  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 315  0
         removeAdHocPersonsAndWorkgroups(document);
 316  0
         return document;
 317  
     }
 318  
 
 319  
     /**
 320  
      * @see org.kuali.rice.kns.service.DocumentService#blanketApproveDocument(org.kuali.rice.kns.document.Document, java.lang.String,
 321  
      *      java.util.List)
 322  
      */
 323  
     @Override
 324  
         public Document blanketApproveDocument(Document document, String annotation, List<AdHocRouteRecipient> adHocRecipients) throws ValidationException, WorkflowException {
 325  0
         checkForNulls(document);
 326  
         //if (!getDocumentActionFlags(document).getCanBlanketApprove()) {
 327  
         //    throw buildAuthorizationException("blanket approve", document);
 328  
         //}
 329  0
         document.prepareForSave();
 330  0
         Document savedDocument = validateAndPersistDocument(document, new BlanketApproveDocumentEvent(document));
 331  0
         prepareWorkflowDocument(savedDocument);
 332  0
         getWorkflowDocumentService().blanketApprove(savedDocument.getDocumentHeader().getWorkflowDocument(), annotation, adHocRecipients);
 333  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),savedDocument.getDocumentHeader().getWorkflowDocument());
 334  0
         removeAdHocPersonsAndWorkgroups(savedDocument);
 335  0
         return savedDocument;
 336  
     }
 337  
 
 338  
     /**
 339  
      * @see org.kuali.rice.kns.service.DocumentService#clearDocumentFyi(org.kuali.rice.kns.document.Document, java.util.List)
 340  
      */
 341  
     @Override
 342  
         public Document clearDocumentFyi(Document document, List<AdHocRouteRecipient> adHocRecipients) throws WorkflowException {
 343  0
         checkForNulls(document);
 344  
         // populate document content so searchable attributes will be indexed properly
 345  0
         document.populateDocumentForRouting();
 346  0
         getWorkflowDocumentService().clearFyi(document.getDocumentHeader().getWorkflowDocument(), adHocRecipients);
 347  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 348  0
         removeAdHocPersonsAndWorkgroups(document);
 349  0
         return document;
 350  
     }
 351  
 
 352  
     protected void checkForNulls(Document document) {
 353  0
         if (document == null) {
 354  0
             throw new IllegalArgumentException("invalid (null) document");
 355  
         }
 356  0
         if (document.getDocumentNumber() == null) {
 357  0
             throw new IllegalStateException("invalid (null) documentHeaderId");
 358  
         }
 359  0
     }
 360  
 
 361  
     private Document validateAndPersistDocumentAndSaveAdHocRoutingRecipients(Document document, KualiDocumentEvent event) {
 362  
         /*
 363  
          * Using this method to wrap validateAndPersistDocument to keep everything in one transaction. This avoids modifying the
 364  
          * signature on validateAndPersistDocument method
 365  
          */
 366  0
         List<AdHocRouteRecipient> adHocRoutingRecipients = new ArrayList<AdHocRouteRecipient>();
 367  0
         adHocRoutingRecipients.addAll(document.getAdHocRoutePersons());
 368  0
         adHocRoutingRecipients.addAll(document.getAdHocRouteWorkgroups());
 369  
 
 370  0
         for (AdHocRouteRecipient recipient : adHocRoutingRecipients) {
 371  0
                         recipient.setdocumentNumber(document.getDocumentNumber());
 372  
                 }
 373  0
         Map<String, String> criteria = new HashMap<String, String>();
 374  0
         criteria.put("documentNumber", document.getDocumentNumber());
 375  0
         getBusinessObjectService().deleteMatching(AdHocRouteRecipient.class, criteria);
 376  
 
 377  0
         getBusinessObjectService().save(adHocRoutingRecipients);
 378  0
         return validateAndPersistDocument(document, event);
 379  
     }
 380  
 
 381  
     /**
 382  
      * @see org.kuali.rice.kns.service.DocumentService#documentExists(java.lang.String)
 383  
      */
 384  
     @Override
 385  
         public boolean documentExists(String documentHeaderId) {
 386  
         // validate parameters
 387  0
         if (StringUtils.isBlank(documentHeaderId)) {
 388  0
             throw new IllegalArgumentException("invalid (blank) documentHeaderId");
 389  
         }
 390  
 
 391  0
             boolean internalUserSession = false;
 392  
             try {
 393  
                     // KFSMI-2543 - allowed method to run without a user session so it can be used
 394  
                     // by workflow processes
 395  0
                 if (GlobalVariables.getUserSession() == null) {
 396  0
                         internalUserSession = true;
 397  0
                         GlobalVariables.setUserSession(new UserSession(KNSConstants.SYSTEM_USER));
 398  0
                         GlobalVariables.clear();
 399  
                 }
 400  
 
 401  
                 // look for workflowDocumentHeader, since that supposedly won't break the transaction
 402  0
                 if (getWorkflowDocumentService().workflowDocumentExists(documentHeaderId)) {
 403  
                     // look for docHeaderId, since that fails without breaking the transaction
 404  0
                     return getDocumentHeaderService().getDocumentHeaderById(documentHeaderId) != null;
 405  
                 }
 406  
 
 407  0
                 return false;
 408  
             } finally {
 409  
                     // if a user session was established for this call, clear it our
 410  0
                     if ( internalUserSession ) {
 411  0
                             GlobalVariables.clear();
 412  0
                             GlobalVariables.setUserSession(null);
 413  
                     }
 414  
             }
 415  
     }
 416  
 
 417  
     /**
 418  
      * Creates a new document by class.
 419  
      *
 420  
      * @see org.kuali.rice.kns.service.DocumentService#getNewDocument(java.lang.Class)
 421  
      */
 422  
     @Override
 423  
         public Document getNewDocument(Class<? extends Document> documentClass) throws WorkflowException {
 424  0
         if (documentClass == null) {
 425  0
             throw new IllegalArgumentException("invalid (null) documentClass");
 426  
         }
 427  0
         if (!Document.class.isAssignableFrom(documentClass)) {
 428  0
             throw new IllegalArgumentException("invalid (non-Document) documentClass");
 429  
         }
 430  
 
 431  0
         String documentTypeName = getDataDictionaryService().getDocumentTypeNameByClass(documentClass);
 432  0
         if (StringUtils.isBlank(documentTypeName)) {
 433  0
             throw new UnknownDocumentTypeException("unable to get documentTypeName for unknown documentClass '" + documentClass.getName() + "'");
 434  
         }
 435  0
         return getNewDocument(documentTypeName);
 436  
     }
 437  
 
 438  
 
 439  
     /**
 440  
      * Creates a new document by document type name.
 441  
      *
 442  
      * @see org.kuali.rice.kns.service.DocumentService#getNewDocument(java.lang.String)
 443  
      */
 444  
     @Override
 445  
         public Document getNewDocument(String documentTypeName) throws WorkflowException {
 446  
 
 447  
         // argument validation
 448  0
         String watchName = "DocumentServiceImpl.getNewDocument";
 449  0
         StopWatch watch = new StopWatch();
 450  0
         watch.start();
 451  0
         if (LOG.isDebugEnabled()) {
 452  0
             LOG.debug(watchName + ": started");
 453  
         }
 454  0
         if (StringUtils.isBlank(documentTypeName)) {
 455  0
             throw new IllegalArgumentException("invalid (blank) documentTypeName");
 456  
         }
 457  0
         if (GlobalVariables.getUserSession() == null) {
 458  0
             throw new IllegalStateException("GlobalVariables must be populated with a valid UserSession before a new document can be created");
 459  
         }
 460  
 
 461  
         // get the class for this docTypeName
 462  0
         Class<? extends Document> documentClass = getDocumentClassByTypeName(documentTypeName);
 463  
 
 464  
         // get the current user
 465  0
         Person currentUser = GlobalVariables.getUserSession().getPerson();
 466  
 
 467  
         // get the authorization
 468  0
         DocumentAuthorizer documentAuthorizer = getDocumentHelperService().getDocumentAuthorizer(documentTypeName);
 469  0
         DocumentPresentationController documentPresentationController = getDocumentHelperService().getDocumentPresentationController(documentTypeName);
 470  
         // make sure this person is authorized to initiate
 471  0
         LOG.debug("calling canInitiate from getNewDocument()");
 472  0
         if (!documentPresentationController.canInitiate(documentTypeName) || !documentAuthorizer.canInitiate(documentTypeName, currentUser)) {
 473  0
                 throw new DocumentAuthorizationException(currentUser.getPrincipalName(), "initiate", documentTypeName);
 474  
         }
 475  
 
 476  
         // initiate new workflow entry, get the workflow doc
 477  0
         KualiWorkflowDocument workflowDocument = getWorkflowDocumentService().createWorkflowDocument(documentTypeName, GlobalVariables.getUserSession().getPerson());
 478  0
         KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),workflowDocument);
 479  
 
 480  
         // create a new document header object
 481  0
         DocumentHeader documentHeader = null;
 482  
         try {
 483  
             // create a new document header object
 484  0
             Class<? extends DocumentHeader> documentHeaderClass = getDocumentHeaderService().getDocumentHeaderBaseClass();
 485  0
             documentHeader = documentHeaderClass.newInstance();
 486  0
             documentHeader.setWorkflowDocument(workflowDocument);
 487  0
             documentHeader.setDocumentNumber(workflowDocument.getRouteHeaderId().toString());
 488  
             // status and notes are initialized correctly in the constructor
 489  
         }
 490  0
         catch (IllegalAccessException e) {
 491  0
             throw new RuntimeException("Error instantiating DocumentHeader", e);
 492  
         }
 493  0
         catch (InstantiationException e) {
 494  0
             throw new RuntimeException("Error instantiating DocumentHeader", e);
 495  0
         }
 496  
 
 497  
         // build Document of specified type
 498  0
         Document document = null;
 499  
         try {
 500  
             // all maintenance documents have same class
 501  0
             if (MaintenanceDocumentBase.class.isAssignableFrom(documentClass)) {
 502  0
                 Class<?>[] defaultConstructor = new Class[]{String.class};
 503  0
                 Constructor<? extends Document> cons = documentClass.getConstructor(defaultConstructor);
 504  0
                 if (ObjectUtils.isNull(cons)) {
 505  0
                     throw new ConfigurationException("Could not find constructor with document type name parameter needed for Maintenance Document Base class");
 506  
                 }
 507  0
                 document = cons.newInstance(documentTypeName);
 508  0
             } else {
 509  
                 // non-maintenance document
 510  0
                 document = documentClass.newInstance();
 511  
             }
 512  0
         } catch (IllegalAccessException e) {
 513  0
             throw new RuntimeException("Error instantiating Document", e);
 514  0
         } catch (InstantiationException e) {
 515  0
             throw new RuntimeException("Error instantiating Document", e);
 516  0
         } catch (SecurityException e) {
 517  0
             throw new RuntimeException("Error instantiating Maintenance Document", e);
 518  0
         } catch (NoSuchMethodException e) {
 519  0
             throw new RuntimeException("Error instantiating Maintenance Document: No constructor with String parameter found", e);
 520  0
         } catch (IllegalArgumentException e) {
 521  0
             throw new RuntimeException("Error instantiating Maintenance Document", e);
 522  0
         } catch (InvocationTargetException e) {
 523  0
             throw new RuntimeException("Error instantiating Maintenance Document", e);
 524  0
         }
 525  
 
 526  0
         document.setDocumentHeader(documentHeader);
 527  0
         document.setDocumentNumber(documentHeader.getDocumentNumber());
 528  
 
 529  0
         watch.stop();
 530  0
         if (LOG.isDebugEnabled()) {
 531  0
             LOG.debug(watchName + ": " + watch.toString());        
 532  
         }
 533  0
         return document;
 534  
     }
 535  
 
 536  
     /**
 537  
      * This is temporary until workflow 2.0 and reads from a table to get documents whose status has changed to A (approved - no
 538  
      * outstanding approval actions requested)
 539  
      *
 540  
      * @param documentHeaderId
 541  
      * @throws WorkflowException
 542  
      * @return Document
 543  
      */
 544  
     @Override
 545  
         public Document getByDocumentHeaderId(String documentHeaderId) throws WorkflowException {
 546  0
         if (documentHeaderId == null) {
 547  0
             throw new IllegalArgumentException("invalid (null) documentHeaderId");
 548  
         }
 549  0
             boolean internalUserSession = false;
 550  
             try {
 551  
                     // KFSMI-2543 - allowed method to run without a user session so it can be used
 552  
                     // by workflow processes
 553  0
                 if (GlobalVariables.getUserSession() == null) {
 554  0
                         internalUserSession = true;
 555  0
                         GlobalVariables.setUserSession(new UserSession(KNSConstants.SYSTEM_USER));
 556  0
                         GlobalVariables.clear();
 557  
                 }
 558  
 
 559  0
                 KualiWorkflowDocument workflowDocument = null;
 560  
 
 561  0
                 if ( LOG.isDebugEnabled() ) {
 562  0
                         LOG.debug("Retrieving doc id: " + documentHeaderId + " from workflow service.");
 563  
                 }
 564  0
                 workflowDocument = getWorkflowDocumentService().createWorkflowDocument(Long.valueOf(documentHeaderId), GlobalVariables.getUserSession().getPerson());
 565  0
                 KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),workflowDocument);
 566  
 
 567  0
                 Class<? extends Document> documentClass = getDocumentClassByTypeName(workflowDocument.getDocumentType());
 568  
 
 569  
                 // retrieve the Document
 570  0
                 Document document = getDocumentDao().findByDocumentHeaderId(documentClass, documentHeaderId);
 571  
                 
 572  0
                 return postProcessDocument(documentHeaderId, workflowDocument, document);
 573  
             } finally {
 574  
                     // if a user session was established for this call, clear it out
 575  0
                     if ( internalUserSession ) {
 576  0
                             GlobalVariables.clear();
 577  0
                             GlobalVariables.setUserSession(null);
 578  
                     }
 579  
             }
 580  
     }
 581  
 
 582  
         /**
 583  
          * @see org.kuali.rice.kns.service.DocumentService#getByDocumentHeaderIdSessionless(java.lang.String)
 584  
          */
 585  
         @Override
 586  
         public Document getByDocumentHeaderIdSessionless(String documentHeaderId)
 587  
                         throws WorkflowException {
 588  0
         if (documentHeaderId == null) {
 589  0
             throw new IllegalArgumentException("invalid (null) documentHeaderId");
 590  
         }
 591  
 
 592  0
         KualiWorkflowDocument workflowDocument = null;
 593  
 
 594  0
         if ( LOG.isDebugEnabled() ) {
 595  0
             LOG.debug("Retrieving doc id: " + documentHeaderId + " from workflow service.");
 596  
         }
 597  
 
 598  0
         Person person = getPersonService().getPersonByPrincipalName(KNSConstants.SYSTEM_USER);
 599  0
         workflowDocument = workflowDocumentService.createWorkflowDocument(Long.valueOf(documentHeaderId), person);
 600  
 
 601  0
         Class<? extends Document> documentClass = getDocumentClassByTypeName(workflowDocument.getDocumentType());
 602  
 
 603  
         // retrieve the Document
 604  0
         Document document = getDocumentDao().findByDocumentHeaderId(documentClass, documentHeaderId);
 605  
         
 606  0
         return postProcessDocument(documentHeaderId, workflowDocument, document);
 607  
         }
 608  
 
 609  
     private Class<? extends Document> getDocumentClassByTypeName(String documentTypeName) {
 610  0
         if (StringUtils.isBlank(documentTypeName)) {
 611  0
             throw new IllegalArgumentException("invalid (blank) documentTypeName");
 612  
         }
 613  
 
 614  0
         Class<? extends Document> clazz = getDataDictionaryService().getDocumentClassByTypeName(documentTypeName);
 615  0
         if (clazz == null) {
 616  0
             throw new UnknownDocumentTypeException("unable to get class for unknown documentTypeName '" + documentTypeName + "'");
 617  
         }
 618  0
         return clazz;
 619  
     }
 620  
     
 621  
     /**
 622  
      * Loads the Notes for the note target on this Document.
 623  
      * 
 624  
      * @param document the document for which to load the notes
 625  
      */
 626  
     protected void loadNotes(Document document) {
 627  0
             if (isNoteTargetReady(document)) {
 628  0
                     List<Note> notes = getNoteService().getByRemoteObjectId(document.getNoteTarget().getObjectId());
 629  
                     // KULRNE-5692 - force a refresh of the attachments
 630  
             // they are not (non-updateable) references and don't seem to update properly upon load
 631  0
                     for (Note note : notes) {
 632  0
                             note.refreshReferenceObject("attachment");
 633  
                     }
 634  0
                     document.setNotes(notes);
 635  
             }
 636  0
     }
 637  
 
 638  
     /**
 639  
      * Performs required post-processing for every document from the documentDao
 640  
      *
 641  
      * @param documentHeaderId
 642  
      * @param workflowDocument
 643  
      * @param document
 644  
      */
 645  
     private Document postProcessDocument(String documentHeaderId, KualiWorkflowDocument workflowDocument, Document document) {
 646  0
         if (document != null) {
 647  0
             document.getDocumentHeader().setWorkflowDocument(workflowDocument);
 648  0
             document.processAfterRetrieve();
 649  0
             loadNotes(document);
 650  
         }
 651  0
         return document;
 652  
     }
 653  
 
 654  
 
 655  
     /**
 656  
      * The default implementation - this retrieves all documents by a list of documentHeader for a given class.
 657  
      *
 658  
      * @see org.kuali.rice.kns.service.DocumentService#getDocumentsByListOfDocumentHeaderIds(java.lang.Class, java.util.List)
 659  
      */
 660  
     @Override
 661  
         public List<Document> getDocumentsByListOfDocumentHeaderIds(Class<? extends Document> documentClass, List<String> documentHeaderIds) throws WorkflowException {
 662  
         // validate documentHeaderIdList and contents
 663  0
         if (documentHeaderIds == null) {
 664  0
             throw new IllegalArgumentException("invalid (null) documentHeaderId list");
 665  
         }
 666  0
         int index = 0;
 667  0
         for (String documentHeaderId : documentHeaderIds) {
 668  0
             if (StringUtils.isBlank(documentHeaderId)) {
 669  0
                 throw new IllegalArgumentException("invalid (blank) documentHeaderId at list index " + index);
 670  
             }
 671  0
             index++;
 672  
         }
 673  
 
 674  0
             boolean internalUserSession = false;
 675  
             try {
 676  
                     // KFSMI-2543 - allowed method to run without a user session so it can be used
 677  
                     // by workflow processes
 678  0
                 if (GlobalVariables.getUserSession() == null) {
 679  0
                         internalUserSession = true;
 680  0
                         GlobalVariables.setUserSession(new UserSession(KNSConstants.SYSTEM_USER));
 681  0
                         GlobalVariables.clear();
 682  
                 }
 683  
 
 684  
                 // retrieve all documents that match the document header ids
 685  0
                 List<? extends Document> rawDocuments = getDocumentDao().findByDocumentHeaderIds(documentClass, documentHeaderIds);
 686  
 
 687  
                 // post-process them
 688  0
                 List<Document> documents = new ArrayList<Document>();
 689  0
                 for (Document document : rawDocuments) {
 690  0
                     KualiWorkflowDocument workflowDocument = getWorkflowDocumentService().createWorkflowDocument(Long.valueOf(document.getDocumentNumber()), GlobalVariables.getUserSession().getPerson());
 691  
 
 692  0
                     document = postProcessDocument(document.getDocumentNumber(), workflowDocument, document);
 693  0
                     documents.add(document);
 694  0
                 }
 695  0
                 return documents;
 696  
             } finally {
 697  
                     // if a user session was established for this call, clear it our
 698  0
                     if ( internalUserSession ) {
 699  0
                             GlobalVariables.clear();
 700  0
                             GlobalVariables.setUserSession(null);
 701  
                     }
 702  
             }
 703  
     }
 704  
 
 705  
     /* Helper Methods */
 706  
 
 707  
     /**
 708  
      * Validates and persists a document.
 709  
      *
 710  
      * @see org.kuali.rice.kns.service.DocumentService#validateAndPersistDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 711  
      */
 712  
     public Document validateAndPersistDocument(Document document, KualiDocumentEvent event) throws ValidationException {
 713  0
         if (document == null) {
 714  0
             LOG.error("document passed to validateAndPersist was null");
 715  0
             throw new IllegalArgumentException("invalid (null) document");
 716  
         }
 717  0
         if ( LOG.isDebugEnabled() ) {
 718  0
                 LOG.debug("validating and preparing to persist document " + document.getDocumentNumber());
 719  
         }
 720  
 
 721  0
         document.validateBusinessRules(event);
 722  0
         document.prepareForSave(event);
 723  
 
 724  
         // save the document
 725  0
         Document savedDocument = null;
 726  
         try {
 727  0
                 if ( LOG.isInfoEnabled() ) {
 728  0
                         LOG.info("storing document " + document.getDocumentNumber());
 729  
                 }
 730  0
             savedDocument = getDocumentDao().save(document);
 731  
         }
 732  0
         catch (OptimisticLockingFailureException e) {
 733  0
             LOG.error("exception encountered on store of document " + e.getMessage());
 734  0
             throw e;
 735  0
         }
 736  
         
 737  0
         boolean notesSaved = saveDocumentNotes(document);
 738  0
         if (!notesSaved) {
 739  0
                 if (LOG.isInfoEnabled()) {
 740  0
                         LOG.info("Notes not saved during validateAndPersistDocument, likely means that note save needs to be deferred because note target is not ready.");
 741  
                 }
 742  
         }
 743  
 
 744  0
         savedDocument.postProcessSave(event);
 745  
 
 746  0
         return savedDocument;
 747  
     }
 748  
 
 749  
 
 750  
     /**
 751  
      * Sets the title and app document id in the flex document
 752  
      *
 753  
      * @param document
 754  
      * @throws WorkflowException
 755  
      */
 756  
     @Override
 757  
         public void prepareWorkflowDocument(Document document) throws WorkflowException {
 758  
         // populate document content so searchable attributes will be indexed properly
 759  0
         document.populateDocumentForRouting();
 760  
 
 761  
         // make sure we push the document title into the workflowDocument
 762  0
         populateDocumentTitle(document);
 763  
 
 764  
         // make sure we push the application document id into the workflowDocument
 765  0
         populateApplicationDocumentId(document);
 766  0
     }
 767  
 
 768  
     /**
 769  
      * This method will grab the generated document title from the document and add it to the workflowDocument so that it gets pushed into
 770  
      * workflow when routed.
 771  
      *
 772  
      * @param document
 773  
      * @throws WorkflowException
 774  
      */
 775  
     private void populateDocumentTitle(Document document) throws WorkflowException {
 776  0
         String documentTitle = document.getDocumentTitle();
 777  0
         if (StringUtils.isNotBlank(documentTitle)) {
 778  0
             document.getDocumentHeader().getWorkflowDocument().setTitle(documentTitle);
 779  
         }
 780  0
     }
 781  
 
 782  
     /**
 783  
      * This method will grab the organization document number from the document and add it to the workflowDocument so that it gets pushed
 784  
      * into workflow when routed.
 785  
      *
 786  
      * @param document
 787  
      */
 788  
     private void populateApplicationDocumentId(Document document) {
 789  0
         String organizationDocumentNumber = document.getDocumentHeader().getOrganizationDocumentNumber();
 790  0
         if (StringUtils.isNotBlank(organizationDocumentNumber)) {
 791  0
             document.getDocumentHeader().getWorkflowDocument().setAppDocId(organizationDocumentNumber);
 792  
         }
 793  0
     }
 794  
 
 795  
     /**
 796  
      * This is to allow for updates of document statuses and other related requirements for updates outside of the initial save and
 797  
      * route
 798  
      */
 799  
     @Override
 800  
         public Document updateDocument(Document document) {
 801  0
         checkForNulls(document);
 802  0
         return getDocumentDao().save(document);
 803  
     }
 804  
 
 805  
     /**
 806  
      *
 807  
      * @see org.kuali.rice.kns.service.DocumentService#createNoteFromDocument(org.kuali.rice.kns.document.Document, java.lang.String)
 808  
      */
 809  
     @Override
 810  
         public Note createNoteFromDocument(Document document, String text) {
 811  0
         Note note = new Note();
 812  
 
 813  0
         note.setNotePostedTimestamp(getDateTimeService().getCurrentTimestamp());
 814  0
         note.setVersionNumber(Long.valueOf(1));
 815  0
         note.setNoteText(text);
 816  0
         note.setNoteTypeCode(document.getNoteType().getCode());
 817  
 
 818  0
         PersistableBusinessObject bo = document.getNoteTarget();
 819  
         // TODO gah! this is awful
 820  0
         Person kualiUser = GlobalVariables.getUserSession().getPerson();
 821  0
         if (kualiUser == null) {
 822  0
                 throw new IllegalStateException("Current UserSession has a null Person.");
 823  
         }
 824  0
         return bo == null ? null : getNoteService().createNote(note, bo, kualiUser.getPrincipalId());
 825  
     }
 826  
     
 827  
     
 828  
         /**
 829  
          * @see org.kuali.rice.kns.service.DocumentService#saveDocumentNotes(org.kuali.rice.kns.document.Document)
 830  
          */
 831  
     @Override
 832  
         public boolean saveDocumentNotes(Document document) {
 833  0
                 if (isNoteTargetReady(document)) {
 834  0
                         List<Note> notes = document.getNotes();
 835  0
                         for (Note note : document.getNotes()) {
 836  0
                                 linkNoteRemoteObjectId(note, document.getNoteTarget());
 837  
                         }
 838  0
                         getNoteService().saveNoteList(notes);
 839  0
                         return true;
 840  
                 }
 841  0
                 return false;
 842  
         }
 843  
         
 844  
         /**
 845  
          * Determines if the given document's note target is ready for notes to be
 846  
          * attached and persisted against it.  This method verifies that the document's
 847  
          * note target is non-null as well as checking that it has a non-empty object id. 
 848  
          *
 849  
          * @param the document on which to check for note target readiness
 850  
          * @return true if the note target is ready, false otherwise
 851  
          */
 852  
         protected boolean isNoteTargetReady(Document document) {
 853  0
                 PersistableBusinessObject noteTarget = document.getNoteTarget();
 854  0
                 if (noteTarget == null || StringUtils.isBlank(noteTarget.getObjectId())) {
 855  0
                         return false;
 856  
                 }
 857  0
                 return true;
 858  
         }
 859  
         
 860  
         private void linkNoteRemoteObjectId(Note note, PersistableBusinessObject noteTarget) {
 861  0
             String objectId = noteTarget.getObjectId();
 862  0
             if (StringUtils.isBlank(objectId)) {
 863  0
                     throw new IllegalStateException("Attempted to link a Note with a PersistableBusinessObject with no object id");
 864  
             }
 865  0
             note.setRemoteObjectIdentifier(noteTarget.getObjectId());
 866  0
     }
 867  
 
 868  
     /**
 869  
          * This overridden method ...
 870  
          *
 871  
          * @see org.kuali.rice.kns.service.DocumentService#sendAdHocRequests(org.kuali.rice.kns.document.Document, java.util.List)
 872  
          */
 873  
         @Override
 874  
         public void sendAdHocRequests(Document document, String annotation, List<AdHocRouteRecipient> adHocRecipients) throws WorkflowException{
 875  0
                 prepareWorkflowDocument(document);
 876  0
                 getWorkflowDocumentService().sendWorkflowNotification(document.getDocumentHeader().getWorkflowDocument(),
 877  
                         annotation, adHocRecipients);
 878  0
                 KNSServiceLocatorWeb.getSessionDocumentService().addDocumentToUserSession(GlobalVariables.getUserSession(),document.getDocumentHeader().getWorkflowDocument());
 879  
                 //getBusinessObjectService().delete(document.getAdHocRoutePersons());
 880  
                 //getBusinessObjectService().delete(document.getAdHocRouteWorkgroups());
 881  0
                 removeAdHocPersonsAndWorkgroups(document);
 882  0
         }
 883  
 
 884  
         /**
 885  
      * spring injected date time service
 886  
      *
 887  
      * @param dateTimeService
 888  
      */
 889  
     public synchronized void setDateTimeService(DateTimeService dateTimeService) {
 890  0
         this.dateTimeService = dateTimeService;
 891  0
     }
 892  
 
 893  
     /**
 894  
      * Gets the DateTimeService, lazily initializing if necessary
 895  
      * @return the DateTimeService
 896  
      */
 897  
     private synchronized DateTimeService getDateTimeService() {
 898  0
         if (this.dateTimeService == null) {
 899  0
             this.dateTimeService = KNSServiceLocator.getDateTimeService();
 900  
         }
 901  0
         return this.dateTimeService;
 902  
     }
 903  
 
 904  
     /**
 905  
      * Sets the noteService attribute value.
 906  
      * @param noteService The noteService to set.
 907  
      */
 908  
     public synchronized void setNoteService(NoteService noteService) {
 909  0
         this.noteService = noteService;
 910  0
     }
 911  
 
 912  
     /**
 913  
      * Gets the NoteService, lazily initializing if necessary
 914  
      * @return the NoteService
 915  
      */
 916  
     protected synchronized NoteService getNoteService() {
 917  0
         if (this.noteService == null) {
 918  0
             this.noteService = KNSServiceLocator.getNoteService();
 919  
         }
 920  0
         return this.noteService;
 921  
     }
 922  
 
 923  
     /**
 924  
      * Sets the businessObjectService attribute value.
 925  
      *
 926  
      * @param businessObjectService The businessObjectService to set.
 927  
      */
 928  
     public synchronized void setBusinessObjectService(BusinessObjectService businessObjectService) {
 929  0
         this.businessObjectService = businessObjectService;
 930  0
     }
 931  
 
 932  
     /**
 933  
      * Gets the {@link BusinessObjectService}, lazily initializing if necessary
 934  
      * @return the {@link BusinessObjectService}
 935  
      */
 936  
     protected synchronized BusinessObjectService getBusinessObjectService() {
 937  0
         if (this.businessObjectService == null) {
 938  0
             this.businessObjectService = KNSServiceLocator.getBusinessObjectService();
 939  
         }
 940  0
         return this.businessObjectService;
 941  
     }
 942  
 
 943  
     /**
 944  
      * Sets the workflowDocumentService attribute value.
 945  
      *
 946  
      * @param workflowDocumentService The workflowDocumentService to set.
 947  
      */
 948  
     public synchronized void setWorkflowDocumentService(WorkflowDocumentService workflowDocumentService) {
 949  0
         this.workflowDocumentService = workflowDocumentService;
 950  0
     }
 951  
 
 952  
     /**
 953  
      * Gets the {@link WorkflowDocumentService}, lazily initializing if necessary
 954  
      * @return the {@link WorkflowDocumentService}
 955  
      */
 956  
     protected synchronized WorkflowDocumentService getWorkflowDocumentService() {
 957  0
         if (this.workflowDocumentService == null) {
 958  0
             this.workflowDocumentService = KNSServiceLocatorWeb.getWorkflowDocumentService();
 959  
         }
 960  0
         return this.workflowDocumentService;
 961  
     }
 962  
 
 963  
     /**
 964  
      * Sets the documentDao attribute value.
 965  
      *
 966  
      * @param documentDao The documentDao to set.
 967  
      */
 968  
     public synchronized void setDocumentDao(DocumentDao documentDao) {
 969  0
         this.documentDao = documentDao;
 970  0
     }
 971  
 
 972  
     /**
 973  
      * Gets the {@link DocumentDao}, lazily initializing if necessary
 974  
      * @return the {@link DocumentDao}
 975  
      */
 976  
     protected synchronized DocumentDao getDocumentDao() {
 977  0
         if (this.documentDao == null) {
 978  0
             this.documentDao = KNSServiceLocatorInternal.getDocumentDao();
 979  
         }
 980  0
         return documentDao;
 981  
     }
 982  
 
 983  
     /**
 984  
      * Sets the dataDictionaryService attribute value.
 985  
      *
 986  
      * @param dataDictionaryService
 987  
      */
 988  
     public synchronized void setDataDictionaryService(DataDictionaryService dataDictionaryService) {
 989  0
         this.dataDictionaryService = dataDictionaryService;
 990  0
     }
 991  
 
 992  
     /**
 993  
      * Gets the {@link DataDictionaryService}, lazily initializing if necessary
 994  
      * @return the {@link DataDictionaryService}
 995  
      */
 996  
     protected synchronized DataDictionaryService getDataDictionaryService() {
 997  0
         if (this.dataDictionaryService == null) {
 998  0
             this.dataDictionaryService = KNSServiceLocatorWeb.getDataDictionaryService();
 999  
         }
 1000  0
         return this.dataDictionaryService;
 1001  
     }
 1002  
 
 1003  
     /**
 1004  
      * @param documentHeaderService the documentHeaderService to set
 1005  
      */
 1006  
     public synchronized void setDocumentHeaderService(DocumentHeaderService documentHeaderService) {
 1007  0
         this.documentHeaderService = documentHeaderService;
 1008  0
     }
 1009  
 
 1010  
     /**
 1011  
      * Gets the {@link DocumentHeaderService}, lazily initializing if necessary
 1012  
      * @return the {@link DocumentHeaderService}
 1013  
      */
 1014  
     protected synchronized DocumentHeaderService getDocumentHeaderService() {
 1015  0
         if (this.documentHeaderService == null) {
 1016  0
             this.documentHeaderService = KNSServiceLocatorWeb.getDocumentHeaderService();
 1017  
         }
 1018  0
         return this.documentHeaderService;
 1019  
     }
 1020  
 
 1021  
     /**
 1022  
          * @param personService the personService to set
 1023  
          */
 1024  
         public PersonService getPersonService() {
 1025  0
                 if (personService == null) {
 1026  0
                         personService = KIMServiceLocator.getPersonService();
 1027  
                 }
 1028  0
                 return personService;
 1029  
         }
 1030  
 
 1031  
     /**
 1032  
      * @return the documentHelperService
 1033  
      */
 1034  
     public DocumentHelperService getDocumentHelperService() {
 1035  0
         if (documentHelperService == null) {
 1036  0
             this.documentHelperService = KNSServiceLocatorWeb.getDocumentHelperService();
 1037  
         }
 1038  0
         return this.documentHelperService;
 1039  
     }
 1040  
 
 1041  
     /**
 1042  
      * @param documentHelperService the documentHelperService to set
 1043  
      */
 1044  
     public void setDocumentHelperService(DocumentHelperService documentHelperService) {
 1045  0
         this.documentHelperService = documentHelperService;
 1046  0
     }
 1047  
     
 1048  
     private void removeAdHocPersonsAndWorkgroups(Document document){
 1049  0
             List<AdHocRoutePerson> adHocRoutePersons = new ArrayList<AdHocRoutePerson>();
 1050  0
             List<AdHocRouteWorkgroup> adHocRouteWorkgroups = new ArrayList<AdHocRouteWorkgroup>();
 1051  0
             getBusinessObjectService().delete(document.getAdHocRoutePersons());
 1052  0
             getBusinessObjectService().delete(document.getAdHocRouteWorkgroups());
 1053  0
             document.setAdHocRoutePersons(adHocRoutePersons);
 1054  0
             document.setAdHocRouteWorkgroups(adHocRouteWorkgroups);
 1055  0
     }
 1056  
 
 1057  
     /**
 1058  
      * @see org.kuali.rice.kns.service.DocumentService
 1059  
      */
 1060  
     @Override
 1061  
     public void sendNoteRouteNotification(Document document, Note note, Person sender) throws WorkflowException {
 1062  0
         AdHocRouteRecipient routeRecipient = note.getAdHocRouteRecipient();
 1063  
 
 1064  
         // build notification request
 1065  0
         Person requestedUser = this.getPersonService().getPersonByPrincipalName(routeRecipient.getId());
 1066  0
         String senderName = sender.getFirstName() + " " + sender.getLastName();
 1067  0
         String requestedName = requestedUser.getFirstName() + " " + requestedUser.getLastName();
 1068  
 
 1069  0
         String notificationText = kualiConfigurationService.getPropertyString(RiceKeyConstants.MESSAGE_NOTE_NOTIFICATION_ANNOTATION);
 1070  0
         if (StringUtils.isBlank(notificationText)) {
 1071  0
             throw new RuntimeException("No annotation message found for note notification. Message needs added to application resources with key:" + RiceKeyConstants.MESSAGE_NOTE_NOTIFICATION_ANNOTATION);
 1072  
         }
 1073  0
         notificationText = MessageFormat.format(notificationText, new Object[]{senderName, requestedName, note.getNoteText()});
 1074  
 
 1075  0
         List<AdHocRouteRecipient> routeRecipients = new ArrayList<AdHocRouteRecipient>();
 1076  0
         routeRecipients.add(routeRecipient);
 1077  
 
 1078  0
         workflowDocumentService.sendWorkflowNotification(document.getDocumentHeader().getWorkflowDocument(), notificationText, routeRecipients, KNSConstants.NOTE_WORKFLOW_NOTIFICATION_REQUEST_LABEL);
 1079  
 
 1080  
         // clear recipient allowing an notification to be sent to another person
 1081  0
         note.setAdHocRouteRecipient(new AdHocRoutePerson());
 1082  0
     }
 1083  
 
 1084  
     /**
 1085  
      * @param kualiConfigurationService the kualiConfigurationService to set
 1086  
      */
 1087  
     public void setKualiConfigurationService(ConfigurationService kualiConfigurationService) {
 1088  0
         this.kualiConfigurationService = kualiConfigurationService;
 1089  0
     }
 1090  
 }