Coverage Report - org.kuali.rice.kns.document.MaintenanceDocumentBase
 
Classes in this File Line Coverage Branch Coverage Complexity
MaintenanceDocumentBase
0%
0/351
0%
0/148
2.774
 
 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.kns.document;
 17  
 
 18  
 import java.io.FileNotFoundException;
 19  
 import java.io.IOException;
 20  
 import java.io.StringReader;
 21  
 import java.util.ArrayList;
 22  
 import java.util.Collections;
 23  
 import java.util.LinkedHashMap;
 24  
 import java.util.List;
 25  
 
 26  
 import javax.persistence.CascadeType;
 27  
 import javax.persistence.Column;
 28  
 import javax.persistence.Entity;
 29  
 import javax.persistence.FetchType;
 30  
 import javax.persistence.JoinColumn;
 31  
 import javax.persistence.ManyToOne;
 32  
 import javax.persistence.Table;
 33  
 import javax.persistence.Transient;
 34  
 import javax.xml.parsers.DocumentBuilder;
 35  
 import javax.xml.parsers.DocumentBuilderFactory;
 36  
 import javax.xml.parsers.ParserConfigurationException;
 37  
 
 38  
 import org.apache.commons.lang.StringUtils;
 39  
 import org.apache.ojb.broker.core.proxy.ProxyHelper;
 40  
 import org.apache.struts.upload.FormFile;
 41  
 import org.kuali.rice.core.util.RiceKeyConstants;
 42  
 import org.kuali.rice.kew.dto.DocumentRouteStatusChangeDTO;
 43  
 import org.kuali.rice.kim.bo.Person;
 44  
 import org.kuali.rice.kns.bo.BusinessObject;
 45  
 import org.kuali.rice.kns.bo.DocumentAttachment;
 46  
 import org.kuali.rice.kns.bo.DocumentHeader;
 47  
 import org.kuali.rice.kns.bo.GlobalBusinessObject;
 48  
 import org.kuali.rice.kns.bo.Note;
 49  
 import org.kuali.rice.kns.bo.PersistableAttachment;
 50  
 import org.kuali.rice.kns.bo.PersistableBusinessObject;
 51  
 import org.kuali.rice.kns.datadictionary.DocumentEntry;
 52  
 import org.kuali.rice.kns.datadictionary.WorkflowAttributes;
 53  
 import org.kuali.rice.kns.datadictionary.WorkflowProperties;
 54  
 import org.kuali.rice.kns.exception.PessimisticLockingException;
 55  
 import org.kuali.rice.kns.exception.ValidationException;
 56  
 import org.kuali.rice.kns.maintenance.Maintainable;
 57  
 import org.kuali.rice.kns.rule.event.KualiDocumentEvent;
 58  
 import org.kuali.rice.kns.rule.event.SaveDocumentEvent;
 59  
 import org.kuali.rice.kns.service.*;
 60  
 import org.kuali.rice.kns.util.GlobalVariables;
 61  
 import org.kuali.rice.kns.util.KNSConstants;
 62  
 import org.kuali.rice.kns.util.MaintenanceUtils;
 63  
 import org.kuali.rice.kns.util.NoteType;
 64  
 import org.kuali.rice.kns.util.ObjectUtils;
 65  
 import org.kuali.rice.kns.util.documentserializer.PropertySerializabilityEvaluator;
 66  
 import org.kuali.rice.kns.workflow.service.KualiWorkflowDocument;
 67  
 import org.w3c.dom.Document;
 68  
 import org.w3c.dom.Node;
 69  
 import org.w3c.dom.NodeList;
 70  
 import org.xml.sax.InputSource;
 71  
 import org.xml.sax.SAXException;
 72  
 
 73  
 
 74  
 /**
 75  
  * The maintenance xml structure will be: <maintainableDocumentContents maintainableImplClass="className">
 76  
  * <oldMaintainableObject>... </oldMaintainableObject> <newMaintainableObject>... </newMaintainableObject>
 77  
  * </maintainableDocumentContents> Maintenance Document
 78  
  */
 79  
 @Entity
 80  
 @Table(name = "KRNS_MAINT_DOC_T")
 81  
 public class MaintenanceDocumentBase extends DocumentBase implements MaintenanceDocument, SessionDocument {
 82  
     private static final long serialVersionUID = -505085142412593305L;
 83  0
     private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(MaintenanceDocumentBase.class);
 84  
     public static final String MAINTAINABLE_IMPL_CLASS = "maintainableImplClass";
 85  
     public static final String OLD_MAINTAINABLE_TAG_NAME = "oldMaintainableObject";
 86  
     public static final String NEW_MAINTAINABLE_TAG_NAME = "newMaintainableObject";
 87  
     public static final String MAINTENANCE_ACTION_TAG_NAME = "maintenanceAction";
 88  
     public static final String NOTES_TAG_NAME = "notes";
 89  
 
 90  
     @Transient
 91  
     transient private static MaintenanceDocumentDictionaryService maintenanceDocumentDictionaryService;
 92  
     @Transient
 93  
     transient private static MaintenanceDocumentService maintenanceDocumentService;
 94  
     @Transient
 95  
     transient private static DocumentHeaderService documentHeaderService;
 96  
     @Transient
 97  
     transient private static DocumentService documentService;
 98  
 
 99  
     @Transient
 100  
     protected Maintainable oldMaintainableObject;
 101  
     @Transient
 102  
     protected Maintainable newMaintainableObject;
 103  
 
 104  
     @Column(name = "DOC_CNTNT", length = 4096)
 105  
     protected String xmlDocumentContents;
 106  
     @Transient
 107  
     protected boolean fieldsClearedOnCopy;
 108  0
     @Transient
 109  
     protected boolean displayTopicFieldInNotes = false;
 110  
 
 111  
     @Transient
 112  
     protected transient FormFile fileAttachment;
 113  
     @Transient
 114  
     protected String attachmentPropertyName;
 115  
 
 116  
     @ManyToOne(fetch = FetchType.LAZY, cascade = {CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE})
 117  
     @JoinColumn(name = "DOC_HDR_ID", insertable = false, updatable = false)
 118  
     protected DocumentAttachment attachment;
 119  
 
 120  
     public FormFile getFileAttachment() {
 121  0
         return this.fileAttachment;
 122  
     }
 123  
 
 124  
     public void setFileAttachment(FormFile fileAttachment) {
 125  0
         this.fileAttachment = fileAttachment;
 126  0
     }
 127  
 
 128  
     public String getAttachmentPropertyName() {
 129  0
         return this.attachmentPropertyName;
 130  
     }
 131  
 
 132  
     public void setAttachmentPropertyName(String attachmentPropertyName) {
 133  0
         this.attachmentPropertyName = attachmentPropertyName;
 134  0
     }
 135  
 
 136  
     public MaintenanceDocumentBase() {
 137  0
         super();
 138  0
         fieldsClearedOnCopy = false;
 139  0
     }
 140  
 
 141  
     /**
 142  
      * Initializies the maintainables.
 143  
      */
 144  
     public MaintenanceDocumentBase(String documentTypeName) {
 145  0
         this();
 146  0
         Class clazz = getMaintenanceDocumentDictionaryService().getMaintainableClass(documentTypeName);
 147  
         try {
 148  0
             oldMaintainableObject = (Maintainable) clazz.newInstance();
 149  0
             newMaintainableObject = (Maintainable) clazz.newInstance();
 150  
 
 151  
             // initialize maintainable with a business object
 152  0
             Class boClazz = getMaintenanceDocumentDictionaryService().getBusinessObjectClass(documentTypeName);
 153  0
             oldMaintainableObject.setBusinessObject((PersistableBusinessObject) boClazz.newInstance());
 154  0
             oldMaintainableObject.setBoClass(boClazz);
 155  0
             newMaintainableObject.setBusinessObject((PersistableBusinessObject) boClazz.newInstance());
 156  0
             newMaintainableObject.setBoClass(boClazz);
 157  0
         } catch (InstantiationException e) {
 158  0
             LOG.error("Unable to initialize maintainables of type " + clazz.getName());
 159  0
             throw new RuntimeException("Unable to initialize maintainables of type " + clazz.getName());
 160  0
         } catch (IllegalAccessException e) {
 161  0
             LOG.error("Unable to initialize maintainables of type " + clazz.getName());
 162  0
             throw new RuntimeException("Unable to initialize maintainables of type " + clazz.getName());
 163  0
         }
 164  0
     }
 165  
 
 166  
     /**
 167  
      * Builds out the document title for maintenance documents - this will get loaded into the flex doc and passed into workflow. It
 168  
      * will be searchable.
 169  
      */
 170  
     @Override
 171  
     public String getDocumentTitle() {
 172  0
         String documentTitle = "";
 173  
 
 174  0
         documentTitle = newMaintainableObject.getDocumentTitle(this);
 175  0
         if (StringUtils.isNotBlank(documentTitle)) {
 176  
             // if doc title has been overridden by maintainable, use it
 177  0
             return documentTitle;
 178  
         }
 179  
 
 180  
         // TODO - build out with bo label once we get the data dictionary stuff in place
 181  
         // build out the right classname
 182  0
         String className = newMaintainableObject.getBusinessObject().getClass().getName();
 183  0
         String truncatedClassName = className.substring(className.lastIndexOf('.') + 1);
 184  0
         if (isOldBusinessObjectInDocument()) {
 185  0
             documentTitle = "Edit ";
 186  
         } else {
 187  0
             documentTitle = "New ";
 188  
         }
 189  0
         documentTitle += truncatedClassName + " - ";
 190  0
         documentTitle += this.getDocumentHeader().getDocumentDescription() + " ";
 191  0
         return documentTitle;
 192  
     }
 193  
 
 194  
     /**
 195  
      * @param xmlDocument
 196  
      * @return
 197  
      */
 198  
     protected boolean isOldMaintainableInDocument(Document xmlDocument) {
 199  0
         boolean isOldMaintainableInExistence = false;
 200  0
         if (xmlDocument.getElementsByTagName(OLD_MAINTAINABLE_TAG_NAME).getLength() > 0) {
 201  0
             isOldMaintainableInExistence = true;
 202  
         }
 203  0
         return isOldMaintainableInExistence;
 204  
     }
 205  
 
 206  
 
 207  
     /**
 208  
      * Checks old maintainable bo has key values
 209  
      */
 210  
     public boolean isOldBusinessObjectInDocument() {
 211  0
         boolean isOldBusinessObjectInExistence = false;
 212  0
         if (oldMaintainableObject == null || oldMaintainableObject.getBusinessObject() == null) {
 213  0
             isOldBusinessObjectInExistence = false;
 214  
         } else {
 215  0
             isOldBusinessObjectInExistence = oldMaintainableObject.isOldBusinessObjectInDocument();
 216  
         }
 217  0
         return isOldBusinessObjectInExistence;
 218  
     }
 219  
 
 220  
     /**
 221  
      * This method is a simplified-naming wrapper around isOldBusinessObjectInDocument(), so that the method name matches the
 222  
      * functionality.
 223  
      */
 224  
     public boolean isNew() {
 225  0
         return MaintenanceUtils.isMaintenanceDocumentCreatingNewRecord(newMaintainableObject.getMaintenanceAction());
 226  
     }
 227  
 
 228  
     /**
 229  
      * This method is a simplified-naming wrapper around isOldBusinessObjectInDocument(), so that the method name matches the
 230  
      * functionality.
 231  
      */
 232  
     public boolean isEdit() {
 233  0
         if (KNSConstants.MAINTENANCE_EDIT_ACTION.equalsIgnoreCase(newMaintainableObject.getMaintenanceAction())) {
 234  0
             return true;
 235  
         } else {
 236  0
             return false;
 237  
         }
 238  
         // return isOldBusinessObjectInDocument();
 239  
     }
 240  
 
 241  
     public boolean isNewWithExisting() {
 242  0
         if (KNSConstants.MAINTENANCE_NEWWITHEXISTING_ACTION.equalsIgnoreCase(newMaintainableObject.getMaintenanceAction())) {
 243  0
             return true;
 244  
         } else {
 245  0
             return false;
 246  
         }
 247  
     }
 248  
 
 249  
     public void populateMaintainablesFromXmlDocumentContents() {
 250  
         // get a hold of the parsed xml document, then read the classname,
 251  
         // then instantiate one to two instances depending on content
 252  
         // then populate those instances
 253  0
         if (!StringUtils.isEmpty(xmlDocumentContents)) {
 254  0
             DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
 255  
             try {
 256  0
                 DocumentBuilder builder = factory.newDocumentBuilder();
 257  0
                 Document xmlDocument = builder.parse(new InputSource(new StringReader(xmlDocumentContents)));
 258  0
                 String clazz = xmlDocument.getDocumentElement().getAttribute(MAINTAINABLE_IMPL_CLASS);
 259  0
                 if (isOldMaintainableInDocument(xmlDocument)) {
 260  0
                     oldMaintainableObject = (Maintainable) Class.forName(clazz).newInstance();
 261  0
                     PersistableBusinessObject bo = getBusinessObjectFromXML(OLD_MAINTAINABLE_TAG_NAME);
 262  
 
 263  0
                     String oldMaintenanceAction = getMaintenanceAction(xmlDocument, OLD_MAINTAINABLE_TAG_NAME);
 264  0
                     oldMaintainableObject.setMaintenanceAction(oldMaintenanceAction);
 265  
 
 266  0
                     oldMaintainableObject.setBusinessObject(bo);
 267  0
                     oldMaintainableObject.setBoClass(bo.getClass());
 268  
                 }
 269  0
                 newMaintainableObject = (Maintainable) Class.forName(clazz).newInstance();
 270  0
                 PersistableBusinessObject bo = getBusinessObjectFromXML(NEW_MAINTAINABLE_TAG_NAME);
 271  0
                 newMaintainableObject.setBusinessObject(bo);
 272  0
                 newMaintainableObject.setBoClass(bo.getClass());
 273  
 
 274  0
                 String newMaintenanceAction = getMaintenanceAction(xmlDocument, NEW_MAINTAINABLE_TAG_NAME);
 275  0
                 newMaintainableObject.setMaintenanceAction(newMaintenanceAction);
 276  
 
 277  0
                 if (newMaintainableObject.isBoNotesEnabled()) {
 278  0
                     List<Note> notes = getNotesFromXml(NOTES_TAG_NAME);
 279  0
                     setNotes(notes);
 280  
                 }
 281  
 
 282  0
             } catch (ParserConfigurationException e) {
 283  0
                 LOG.error("Error while parsing document contents", e);
 284  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 285  0
             } catch (SAXException e) {
 286  0
                 LOG.error("Error while parsing document contents", e);
 287  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 288  0
             } catch (IOException e) {
 289  0
                 LOG.error("Error while parsing document contents", e);
 290  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 291  0
             } catch (InstantiationException e) {
 292  0
                 LOG.error("Error while parsing document contents", e);
 293  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 294  0
             } catch (IllegalAccessException e) {
 295  0
                 LOG.error("Error while parsing document contents", e);
 296  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 297  0
             } catch (ClassNotFoundException e) {
 298  0
                 LOG.error("Error while parsing document contents", e);
 299  0
                 throw new RuntimeException("Could not load document contents from xml", e);
 300  0
             }
 301  
 
 302  
         }
 303  0
     }
 304  
 
 305  
     /**
 306  
      * This method is a lame containment of ugly DOM walking code. This is ONLY necessary because of the version conflicts between
 307  
      * Xalan.jar in 2.6.x and 2.7. As soon as we can upgrade to 2.7, this will be switched to using XPath, which is faster and much
 308  
      * easier on the eyes.
 309  
      *
 310  
      * @param xmlDocument
 311  
      * @param oldOrNewElementName - String oldMaintainableObject or newMaintainableObject
 312  
      * @return the value of the element, or null if none was there
 313  
      */
 314  
     protected String getMaintenanceAction(Document xmlDocument, String oldOrNewElementName) {
 315  
 
 316  0
         if (StringUtils.isBlank(oldOrNewElementName)) {
 317  0
             throw new IllegalArgumentException("oldOrNewElementName may not be blank, null, or empty-string.");
 318  
         }
 319  
 
 320  0
         String maintenanceAction = null;
 321  0
         NodeList rootChildren = xmlDocument.getDocumentElement().getChildNodes();
 322  0
         for (int i = 0; i < rootChildren.getLength(); i++) {
 323  0
             Node rootChild = rootChildren.item(i);
 324  0
             if (oldOrNewElementName.equalsIgnoreCase(rootChild.getNodeName())) {
 325  0
                 NodeList maintChildren = rootChild.getChildNodes();
 326  0
                 for (int j = 0; j < maintChildren.getLength(); j++) {
 327  0
                     Node maintChild = maintChildren.item(j);
 328  0
                     if (MAINTENANCE_ACTION_TAG_NAME.equalsIgnoreCase(maintChild.getNodeName())) {
 329  0
                         maintenanceAction = maintChild.getChildNodes().item(0).getNodeValue();
 330  
                     }
 331  
                 }
 332  
             }
 333  
         }
 334  0
         return maintenanceAction;
 335  
     }
 336  
 
 337  
     private List<Note> getNotesFromXml(String notesTagName) {
 338  0
         String notesXml = StringUtils.substringBetween(xmlDocumentContents, "<" + notesTagName + ">", "</" + notesTagName + ">");
 339  0
         if (StringUtils.isBlank(notesXml)) {
 340  0
             return Collections.emptyList();
 341  
         }
 342  0
         List<Note> notes = (List<Note>) KNSServiceLocator.getXmlObjectSerializerService().fromXml(notesXml);
 343  0
         if (notes == null) {
 344  0
             return Collections.emptyList();
 345  
         }
 346  0
         return notes;
 347  
     }
 348  
 
 349  
     /**
 350  
      * Retrieves substring of document contents from maintainable tag name. Then use xml service to translate xml into a business
 351  
      * object.
 352  
      */
 353  
     protected PersistableBusinessObject getBusinessObjectFromXML(String maintainableTagName) {
 354  0
         String maintXml = StringUtils.substringBetween(xmlDocumentContents, "<" + maintainableTagName + ">", "</" + maintainableTagName + ">");
 355  0
         PersistableBusinessObject businessObject = (PersistableBusinessObject) KNSServiceLocator.getXmlObjectSerializerService().fromXml(maintXml);
 356  0
         return businessObject;
 357  
     }
 358  
 
 359  
     /**
 360  
      * Populates the xml document contents from the maintainables.
 361  
      *
 362  
      * @see org.kuali.rice.kns.document.MaintenanceDocument#populateXmlDocumentContentsFromMaintainables()
 363  
      */
 364  
     public void populateXmlDocumentContentsFromMaintainables() {
 365  0
         StringBuilder docContentBuffer = new StringBuilder();
 366  0
         docContentBuffer.append("<maintainableDocumentContents maintainableImplClass=\"").append(newMaintainableObject.getClass().getName()).append("\">");
 367  
 
 368  
         // if business objects notes are enabled then we need to persist notes to the XML
 369  0
         if (getNewMaintainableObject().isBoNotesEnabled()) {
 370  0
             docContentBuffer.append("<" + NOTES_TAG_NAME + ">");
 371  
             // copy notes to a non-ojb Proxied ArrayList to get rid of the usage of those proxies
 372  
             // note: XmlObjectSerializerServiceImpl should be doing this for us but it does not
 373  
             // appear to be working (at least in this case) and the xml comes through
 374  
             // with the fully qualified ListProxyDefault class name from OJB embedded inside it.
 375  0
             List<Note> noteList = new ArrayList<Note>();
 376  0
             for (Note note : getNotes()) {
 377  0
                 noteList.add(note);
 378  
             }
 379  0
             docContentBuffer.append(KNSServiceLocator.getXmlObjectSerializerService().toXml(noteList));
 380  0
             docContentBuffer.append("</" + NOTES_TAG_NAME + ">");
 381  
         }
 382  0
         if (oldMaintainableObject != null && oldMaintainableObject.getBusinessObject() != null) {
 383  
             // TODO: refactor this out into a method
 384  
 
 385  
 
 386  0
             docContentBuffer.append("<" + OLD_MAINTAINABLE_TAG_NAME + ">");
 387  
 
 388  0
             PersistableBusinessObject oldBo = oldMaintainableObject.getBusinessObject();
 389  0
             ObjectUtils.materializeAllSubObjects(oldBo); // hack to resolve XStream not dealing well with Proxies
 390  0
             docContentBuffer.append(KNSServiceLocator.getBusinessObjectSerializerService().serializeBusinessObjectToXml(oldBo));
 391  
 
 392  
             // add the maintainable's maintenanceAction
 393  0
             docContentBuffer.append("<" + MAINTENANCE_ACTION_TAG_NAME + ">");
 394  0
             docContentBuffer.append(oldMaintainableObject.getMaintenanceAction());
 395  0
             docContentBuffer.append("</" + MAINTENANCE_ACTION_TAG_NAME + ">\n");
 396  
 
 397  0
             docContentBuffer.append("</" + OLD_MAINTAINABLE_TAG_NAME + ">");
 398  
         }
 399  0
         docContentBuffer.append("<" + NEW_MAINTAINABLE_TAG_NAME + ">");
 400  
 
 401  0
         PersistableBusinessObject newBo = newMaintainableObject.getBusinessObject();
 402  0
         ObjectUtils.materializeAllSubObjects(newBo); // hack to resolve XStream not dealing well with Proxies
 403  0
         docContentBuffer.append(KNSServiceLocator.getBusinessObjectSerializerService().serializeBusinessObjectToXml(newBo));
 404  
 
 405  
         // add the maintainable's maintenanceAction
 406  0
         docContentBuffer.append("<" + MAINTENANCE_ACTION_TAG_NAME + ">");
 407  0
         docContentBuffer.append(newMaintainableObject.getMaintenanceAction());
 408  0
         docContentBuffer.append("</" + MAINTENANCE_ACTION_TAG_NAME + ">\n");
 409  
 
 410  0
         docContentBuffer.append("</" + NEW_MAINTAINABLE_TAG_NAME + ">");
 411  0
         docContentBuffer.append("</maintainableDocumentContents>");
 412  0
         xmlDocumentContents = docContentBuffer.toString();
 413  0
     }
 414  
 
 415  
     /**
 416  
      * @see org.kuali.rice.kns.document.DocumentBase#doRouteStatusChange(org.kuali.rice.kew.dto.DocumentRouteStatusChangeDTO)
 417  
      */
 418  
     @Override
 419  
     public void doRouteStatusChange(DocumentRouteStatusChangeDTO statusChangeEvent) {
 420  0
         super.doRouteStatusChange(statusChangeEvent);
 421  
 
 422  0
         KualiWorkflowDocument workflowDocument = getDocumentHeader().getWorkflowDocument();
 423  0
         getNewMaintainableObject().doRouteStatusChange(getDocumentHeader());
 424  
         // commit the changes to the Maintainable BusinessObject when it goes to Processed (ie, fully approved),
 425  
         // and also unlock it
 426  0
         if (workflowDocument.stateIsProcessed()) {
 427  0
             String documentNumber = getDocumentHeader().getDocumentNumber();
 428  0
             newMaintainableObject.setDocumentNumber(documentNumber);
 429  
 
 430  
             //Populate Attachment Property
 431  0
             if (newMaintainableObject.getBusinessObject() instanceof PersistableAttachment) {
 432  0
                 populateAttachmentForBO();
 433  
             }
 434  
 
 435  0
             newMaintainableObject.saveBusinessObject();
 436  
 
 437  0
             if (!getDocumentService().saveDocumentNotes(this)) {
 438  0
                 throw new IllegalStateException("Failed to save document notes, this means that the note target was not ready for notes to be attached when it should have been.");
 439  
             }
 440  
 
 441  
             //Attachment should be deleted from Maintenance Document attachment table
 442  0
             deleteDocumentAttachment();
 443  
 
 444  0
             getMaintenanceDocumentService().deleteLocks(documentNumber);
 445  
 
 446  
             //for issue 3070, check if delete record
 447  0
             if (this.checkAllowsRecordDeletion() && this.checkMaintenanceAction() &&
 448  
                     this.checkDeletePermission(newMaintainableObject.getBusinessObject()))
 449  0
                 newMaintainableObject.deleteBusinessObject();
 450  
         }
 451  
 
 452  
         // unlock the document when its canceled or disapproved
 453  0
         if (workflowDocument.stateIsCanceled() || workflowDocument.stateIsDisapproved()) {
 454  
             //Attachment should be deleted from Maintenance Document attachment table
 455  0
             deleteDocumentAttachment();
 456  
 
 457  0
             String documentNumber = getDocumentHeader().getDocumentNumber();
 458  0
             getMaintenanceDocumentService().deleteLocks(documentNumber);
 459  
         }
 460  0
     }
 461  
 
 462  
     @Override
 463  
     /**
 464  
      * @see org.kuali.rice.kns.document.DocumentBase#getWorkflowEngineDocumentIdsToLock()
 465  
      */
 466  
     public List<Long> getWorkflowEngineDocumentIdsToLock() {
 467  0
         if (newMaintainableObject != null) {
 468  0
             return newMaintainableObject.getWorkflowEngineDocumentIdsToLock();
 469  
         }
 470  0
         return Collections.emptyList();
 471  
     }
 472  
 
 473  
     /**
 474  
      * Pre-Save hook.
 475  
      *
 476  
      * @see org.kuali.rice.kns.document.Document#prepareForSave()
 477  
      */
 478  
     @Override
 479  
     public void prepareForSave() {
 480  0
         if (newMaintainableObject != null) {
 481  0
             newMaintainableObject.prepareForSave();
 482  
         }
 483  0
     }
 484  
 
 485  
     /**
 486  
      * @see org.kuali.rice.kns.document.DocumentBase#processAfterRetrieve()
 487  
      */
 488  
     @Override
 489  
     public void processAfterRetrieve() {
 490  
 
 491  0
         super.processAfterRetrieve();
 492  
             
 493  0
         populateMaintainablesFromXmlDocumentContents();
 494  0
         if (oldMaintainableObject != null) {
 495  0
             oldMaintainableObject.setDocumentNumber(documentNumber);
 496  
         }
 497  0
         if (newMaintainableObject != null) {
 498  0
             newMaintainableObject.setDocumentNumber(documentNumber);
 499  0
             newMaintainableObject.processAfterRetrieve();
 500  
             // If a maintenance lock exists, warn the user.
 501  0
             checkForLockingDocument(false);
 502  
         }
 503  
 
 504  
        
 505  
 
 506  0
     }
 507  
 
 508  
     /**
 509  
      * @return Returns the newMaintainableObject.
 510  
      */
 511  
     public Maintainable getNewMaintainableObject() {
 512  0
         return newMaintainableObject;
 513  
     }
 514  
 
 515  
     /**
 516  
      * @param newMaintainableObject The newMaintainableObject to set.
 517  
      */
 518  
     public void setNewMaintainableObject(Maintainable newMaintainableObject) {
 519  0
         this.newMaintainableObject = newMaintainableObject;
 520  0
     }
 521  
 
 522  
     /**
 523  
      * @return Returns the oldMaintainableObject.
 524  
      */
 525  
     public Maintainable getOldMaintainableObject() {
 526  0
         return oldMaintainableObject;
 527  
     }
 528  
 
 529  
     /**
 530  
      * @param oldMaintainableObject The oldMaintainableObject to set.
 531  
      */
 532  
     public void setOldMaintainableObject(Maintainable oldMaintainableObject) {
 533  0
         this.oldMaintainableObject = oldMaintainableObject;
 534  0
     }
 535  
 
 536  
     @Override
 537  
     public void setDocumentNumber(String documentNumber) {
 538  0
         super.setDocumentNumber(documentNumber);
 539  
 
 540  
         // set the finDocNumber on the Maintainable
 541  0
         oldMaintainableObject.setDocumentNumber(documentNumber);
 542  0
         newMaintainableObject.setDocumentNumber(documentNumber);
 543  
 
 544  0
     }
 545  
 
 546  
     /**
 547  
      * Gets the fieldsClearedOnCopy attribute.
 548  
      *
 549  
      * @return Returns the fieldsClearedOnCopy.
 550  
      */
 551  
     public final boolean isFieldsClearedOnCopy() {
 552  0
         return fieldsClearedOnCopy;
 553  
     }
 554  
 
 555  
     /**
 556  
      * Sets the fieldsClearedOnCopy attribute value.
 557  
      *
 558  
      * @param fieldsClearedOnCopy The fieldsClearedOnCopy to set.
 559  
      */
 560  
     public final void setFieldsClearedOnCopy(boolean fieldsClearedOnCopy) {
 561  0
         this.fieldsClearedOnCopy = fieldsClearedOnCopy;
 562  0
     }
 563  
 
 564  
     /**
 565  
      * Gets the xmlDocumentContents attribute.
 566  
      *
 567  
      * @return Returns the xmlDocumentContents.
 568  
      */
 569  
     public String getXmlDocumentContents() {
 570  0
         return xmlDocumentContents;
 571  
     }
 572  
 
 573  
     /**
 574  
      * Sets the xmlDocumentContents attribute value.
 575  
      *
 576  
      * @param xmlDocumentContents The xmlDocumentContents to set.
 577  
      */
 578  
     public void setXmlDocumentContents(String xmlDocumentContents) {
 579  0
         this.xmlDocumentContents = xmlDocumentContents;
 580  0
     }
 581  
 
 582  
     /**
 583  
      * @see org.kuali.rice.kns.document.Document#getAllowsCopy()
 584  
      */
 585  
     public boolean getAllowsCopy() {
 586  0
         return getMaintenanceDocumentDictionaryService().getAllowsCopy(this);
 587  
     }
 588  
 
 589  
     /**
 590  
      * @see org.kuali.rice.kns.document.MaintenanceDocument#getDisplayTopicFieldInNotes()
 591  
      */
 592  
     public boolean getDisplayTopicFieldInNotes() {
 593  0
         return displayTopicFieldInNotes;
 594  
     }
 595  
 
 596  
     /**
 597  
      * @see org.kuali.rice.kns.document.MaintenanceDocument#setDisplayTopicFieldInNotes(boolean)
 598  
      */
 599  
     public void setDisplayTopicFieldInNotes(boolean displayTopicFieldInNotes) {
 600  0
         this.displayTopicFieldInNotes = displayTopicFieldInNotes;
 601  0
     }
 602  
 
 603  
     @Override
 604  
     /**
 605  
      * Overridden to avoid serializing the xml twice, because of the xmlDocumentContents property of this object
 606  
      */
 607  
     public String serializeDocumentToXml() {
 608  0
         String tempXmlDocumentContents = xmlDocumentContents;
 609  0
         xmlDocumentContents = null;
 610  0
         String xmlForWorkflow = super.serializeDocumentToXml();
 611  0
         xmlDocumentContents = tempXmlDocumentContents;
 612  0
         return xmlForWorkflow;
 613  
     }
 614  
 
 615  
     @Override
 616  
     public void prepareForSave(KualiDocumentEvent event) {
 617  0
         super.prepareForSave(event);
 618  
 
 619  0
         populateDocumentAttachment();
 620  0
         populateXmlDocumentContentsFromMaintainables();
 621  0
     }
 622  
 
 623  
     /**
 624  
      * The attachment BO is proxied in OJB.  For some reason when an attachment does not yet
 625  
      * exist, refreshReferenceObject is not returning null and the proxy cannot be materialized.
 626  
      * So, this method exists to properly handle the proxied attachment BO.  This is a hack
 627  
      * and should be removed post JPA migration.
 628  
      */
 629  
     protected void refreshAttachment() {
 630  0
         if (ObjectUtils.isNull(attachment)) {
 631  0
             this.refreshReferenceObject("attachment");
 632  0
             final boolean isProxy = attachment != null && ProxyHelper.isProxy(attachment);
 633  0
             if (isProxy && ProxyHelper.getRealObject(attachment) == null) {
 634  0
                 attachment = null;
 635  
             }
 636  
         }
 637  0
     }
 638  
 
 639  
     protected void populateAttachmentForBO() {
 640  0
         refreshAttachment();
 641  
 
 642  0
         PersistableAttachment boAttachment = (PersistableAttachment) newMaintainableObject.getBusinessObject();
 643  
 
 644  0
         if (attachment != null) {
 645  
             byte[] fileContents;
 646  0
             fileContents = attachment.getAttachmentContent();
 647  0
             if (fileContents.length > 0) {
 648  0
                 boAttachment.setAttachmentContent(fileContents);
 649  0
                 boAttachment.setFileName(attachment.getFileName());
 650  0
                 boAttachment.setContentType(attachment.getContentType());
 651  
             }
 652  
         }
 653  
 
 654  0
     }
 655  
 
 656  
     public void populateDocumentAttachment() {
 657  0
         refreshAttachment();
 658  
 
 659  0
         if (fileAttachment != null && StringUtils.isNotEmpty(fileAttachment.getFileName())) {
 660  
             //Populate DocumentAttachment BO
 661  0
             if (attachment == null) {
 662  0
                 attachment = new DocumentAttachment();
 663  
             }
 664  
 
 665  
             byte[] fileContents;
 666  
             try {
 667  0
                 fileContents = fileAttachment.getFileData();
 668  0
                 if (fileContents.length > 0) {
 669  0
                     attachment.setFileName(fileAttachment.getFileName());
 670  0
                     attachment.setContentType(fileAttachment.getContentType());
 671  0
                     attachment.setAttachmentContent(fileAttachment.getFileData());
 672  0
                     attachment.setDocumentNumber(getDocumentNumber());
 673  
                 }
 674  0
             } catch (FileNotFoundException e) {
 675  0
                 LOG.error("Error while populating the Document Attachment", e);
 676  0
                 throw new RuntimeException("Could not populate DocumentAttachment object", e);
 677  0
             } catch (IOException e) {
 678  0
                 LOG.error("Error while populating the Document Attachment", e);
 679  0
                 throw new RuntimeException("Could not populate DocumentAttachment object", e);
 680  0
             }
 681  
 
 682  
         }
 683  
 //        else if(attachment != null) {
 684  
 //            //Attachment has been deleted - Need to delete the Attachment Reference Object
 685  
 //            deleteAttachment();
 686  
 //        }
 687  0
     }
 688  
 
 689  
     public void deleteDocumentAttachment() {
 690  0
         KNSServiceLocator.getBusinessObjectService().delete(attachment);
 691  0
         attachment = null;
 692  0
     }
 693  
 
 694  
     /**
 695  
      * Explicitly NOT calling super here.  This is a complete override of the validation
 696  
      * rules behavior.
 697  
      *
 698  
      * @see org.kuali.rice.kns.document.DocumentBase#validateBusinessRules(org.kuali.rice.kns.rule.event.KualiDocumentEvent)
 699  
      */
 700  
     public void validateBusinessRules(KualiDocumentEvent event) {
 701  0
         if (GlobalVariables.getMessageMap().hasErrors()) {
 702  0
             logErrors();
 703  0
             throw new ValidationException("errors occured before business rule");
 704  
         }
 705  
 
 706  
         // check for locking documents for MaintenanceDocuments
 707  0
         if (this instanceof MaintenanceDocument) {
 708  0
             checkForLockingDocument(true);
 709  
         }
 710  
 
 711  
         // Make sure the business object's version number matches that of the database's copy.
 712  0
         if (newMaintainableObject != null) {
 713  0
             if (KNSServiceLocator.getPersistenceStructureService().isPersistable(newMaintainableObject.getBusinessObject().getClass())) {
 714  0
                 PersistableBusinessObject pbObject = KNSServiceLocator.getBusinessObjectService().retrieve(newMaintainableObject.getBusinessObject());
 715  0
                 Long pbObjectVerNbr = ObjectUtils.isNull(pbObject) ? null : pbObject.getVersionNumber();
 716  0
                 Long newObjectVerNbr = newMaintainableObject.getBusinessObject().getVersionNumber();
 717  0
                 if (pbObjectVerNbr != null && !(pbObjectVerNbr.equals(newObjectVerNbr))) {
 718  0
                     GlobalVariables.getMessageMap().putError(KNSConstants.GLOBAL_ERRORS, RiceKeyConstants.ERROR_VERSION_MISMATCH);
 719  0
                     throw new ValidationException("Version mismatch between the local business object and the database business object");
 720  
                 }
 721  
             }
 722  
         }
 723  
 
 724  
         // perform validation against rules engine
 725  0
         if (LOG.isInfoEnabled()) {
 726  0
             LOG.info("invoking rules engine on document " + getDocumentNumber());
 727  
         }
 728  0
         boolean isValid = true;
 729  0
         isValid = KNSServiceLocatorWeb.getKualiRuleService().applyRules(event);
 730  
 
 731  
         // check to see if the br eval passed or failed
 732  0
         if (!isValid) {
 733  0
             logErrors();
 734  
             // TODO: better error handling at the lower level and a better error message are
 735  
             // needed here
 736  0
             throw new ValidationException("business rule evaluation failed");
 737  0
         } else if (GlobalVariables.getMessageMap().hasErrors()) {
 738  0
             logErrors();
 739  0
             if (event instanceof SaveDocumentEvent) {
 740  
                 // for maintenance documents, we want to always actually do a save if the
 741  
                 // user requests a save, even if there are validation or business rules
 742  
                 // failures. this empty if does this, and allows the document to be saved,
 743  
                 // even if there are failures.
 744  
                 // BR or validation failures on a ROUTE even should always stop the route,
 745  
                 // that has not changed
 746  
             } else {
 747  0
                 throw new ValidationException("Unreported errors occured during business rule evaluation (rule developer needs to put meaningful error messages into global ErrorMap)");
 748  
             }
 749  
         }
 750  0
         LOG.debug("validation completed");
 751  
 
 752  0
     }
 753  
 
 754  
 
 755  
     protected void checkForLockingDocument(boolean throwExceptionIfLocked) {
 756  0
         MaintenanceUtils.checkForLockingDocument(this, throwExceptionIfLocked);
 757  0
     }
 758  
 
 759  
     /**
 760  
      * this needs to happen after the document itself is saved, to preserve consistency of the ver_nbr and in the case of initial
 761  
      * save, because this can't be saved until the document is saved initially
 762  
      *
 763  
      * @see org.kuali.rice.kns.document.DocumentBase#postProcessSave(org.kuali.rice.kns.rule.event.KualiDocumentEvent)
 764  
      */
 765  
     @Override
 766  
     public void postProcessSave(KualiDocumentEvent event) {
 767  0
         PersistableBusinessObject bo = getNewMaintainableObject().getBusinessObject();
 768  0
         if (bo instanceof GlobalBusinessObject) {
 769  0
             KNSServiceLocator.getBusinessObjectService().save(bo);
 770  
         }
 771  
         //currently only global documents could change the list of what they're affecting during routing,
 772  
         //so could restrict this to only happening with them, but who knows if that will change, so safest
 773  
         //to always do the delete and re-add...seems a bit inefficient though if nothing has changed, which is
 774  
         //most of the time...could also try to only add/update/delete what's changed, but this is easier
 775  0
         if (!(event instanceof SaveDocumentEvent)) { //don't lock until they route
 776  0
             getMaintenanceDocumentService().deleteLocks(this.getDocumentNumber());
 777  0
             getMaintenanceDocumentService().storeLocks(this.getNewMaintainableObject().generateMaintenanceLocks());
 778  
         }
 779  0
     }
 780  
 
 781  
     /**
 782  
      * @see org.kuali.rice.kns.document.DocumentBase#getDocumentBusinessObject()
 783  
      */
 784  
     @Override
 785  
     public PersistableBusinessObject getDocumentBusinessObject() {
 786  0
         return getNewMaintainableObject().getBusinessObject();
 787  
     }
 788  
 
 789  
     /**
 790  
      * <p>The Note target for maintenance documents is determined by whether or not the underlying
 791  
      * {@link Maintainable} supports business object notes or not.  This is determined via a call to
 792  
      * {@link Maintainable#isBoNotesEnabled()}.  The note target is then derived as follows:
 793  
      * <p/>
 794  
      * <ul>
 795  
      * <li>If the {@link Maintainable} supports business object notes, delegate to {@link #getDocumentBusinessObject()}.
 796  
      * <li>Otherwise, delegate to the default implementation of getNoteTarget on the superclass which will
 797  
      * effectively return a reference to the {@link DocumentHeader}.
 798  
      * </ul>
 799  
      *
 800  
      * @see org.kuali.rice.kns.document.Document#getNoteTarget()
 801  
      */
 802  
     @Override
 803  
     public PersistableBusinessObject getNoteTarget() {
 804  0
         if (getNewMaintainableObject() == null) {
 805  0
             throw new IllegalStateException("Failed to acquire the note target.  The new maintainable object on this document is null.");
 806  
         }
 807  0
         if (getNewMaintainableObject().isBoNotesEnabled()) {
 808  0
             return getDocumentBusinessObject();
 809  
         }
 810  0
         return super.getNoteTarget();
 811  
     }
 812  
 
 813  
     /**
 814  
      * The {@link NoteType} for maintenance documents is determined by whether or not the underlying
 815  
      * {@link Maintainable} supports business object notes or not.  This is determined via a call to
 816  
      * {@link Maintainable#isBoNotesEnabled()}.  The {@link NoteType} is then derived as follows:
 817  
      * <p/>
 818  
      * <ul>
 819  
      * <li>If the {@link Maintainable} supports business object notes, return {@link NoteType#BUSINESS_OBJECT}.
 820  
      * <li>Otherwise, delegate to {@link DocumentBase#getNoteType()}
 821  
      * </ul>
 822  
      *
 823  
      * @see org.kuali.rice.kns.document.Document#getNoteType()
 824  
      * @see org.kuali.rice.kns.document.Document#getNoteTarget()
 825  
      */
 826  
     @Override
 827  
     public NoteType getNoteType() {
 828  0
         if (getNewMaintainableObject().isBoNotesEnabled()) {
 829  0
             return NoteType.BUSINESS_OBJECT;
 830  
         }
 831  0
         return super.getNoteType();
 832  
     }
 833  
 
 834  
     @Override
 835  
     public PropertySerializabilityEvaluator getDocumentPropertySerizabilityEvaluator() {
 836  0
         String docTypeName = "";
 837  0
         if (newMaintainableObject != null) {
 838  0
             docTypeName = getMaintenanceDocumentDictionaryService().getDocumentTypeName(this.newMaintainableObject.getBoClass());
 839  
         } else { // I don't know why we aren't just using the header in the first place
 840  
             // but, in the case where we can't get it in the way above, attempt to get
 841  
             // it off the workflow document header
 842  0
             if (getDocumentHeader() != null && getDocumentHeader().getWorkflowDocument() != null) {
 843  0
                 docTypeName = getDocumentHeader().getWorkflowDocument().getDocumentType();
 844  
             }
 845  
         }
 846  0
         if (!StringUtils.isBlank(docTypeName)) {
 847  0
             DocumentEntry documentEntry = getMaintenanceDocumentDictionaryService().getMaintenanceDocumentEntry(docTypeName);
 848  0
             if (documentEntry != null) {
 849  0
                 WorkflowProperties workflowProperties = documentEntry.getWorkflowProperties();
 850  0
                 WorkflowAttributes workflowAttributes = documentEntry.getWorkflowAttributes();
 851  0
                 return createPropertySerializabilityEvaluator(workflowProperties, workflowAttributes);
 852  
             } else {
 853  0
                 LOG.error("Unable to obtain DD DocumentEntry for document type: '" + docTypeName + "'");
 854  
             }
 855  0
         } else {
 856  0
             LOG.error("Unable to obtain document type name for this document: " + this);
 857  
         }
 858  0
         LOG.error("Returning null for the PropertySerializabilityEvaluator");
 859  0
         return null;
 860  
     }
 861  
 
 862  
     public DocumentAttachment getAttachment() {
 863  0
         return this.attachment;
 864  
     }
 865  
 
 866  
     public void setAttachment(DocumentAttachment attachment) {
 867  0
         this.attachment = attachment;
 868  0
     }
 869  
 
 870  
 
 871  
     /**
 872  
      * This overridden method is used to delete the {@link DocumentHeader} object due to the system not being able to manage the {@link DocumentHeader} object via mapping files
 873  
      *
 874  
      * @see org.kuali.rice.kns.bo.PersistableBusinessObjectBase#postRemove()
 875  
      */
 876  
     @Override
 877  
     protected void postRemove() {
 878  0
         super.postRemove();
 879  0
         getDocumentHeaderService().deleteDocumentHeader(getDocumentHeader());
 880  0
     }
 881  
 
 882  
     /**
 883  
      * This overridden method is used to retrieve the {@link DocumentHeader} object due to the system not being able to manage the {@link DocumentHeader} object via mapping files
 884  
      *
 885  
      * @see org.kuali.rice.kns.bo.PersistableBusinessObjectBase#postLoad()
 886  
      */
 887  
     @Override
 888  
     protected void postLoad() {
 889  0
         super.postLoad();
 890  0
         setDocumentHeader(getDocumentHeaderService().getDocumentHeaderById(getDocumentNumber()));
 891  0
     }
 892  
 
 893  
     /**
 894  
      * This overridden method is used to insert the {@link DocumentHeader} object due to the system not being able to manage the {@link DocumentHeader} object via mapping files
 895  
      *
 896  
      * @see org.kuali.rice.kns.bo.PersistableBusinessObjectBase#prePersist()
 897  
      */
 898  
     @Override
 899  
     protected void prePersist() {
 900  0
         super.prePersist();
 901  0
         getDocumentHeaderService().saveDocumentHeader(getDocumentHeader());
 902  0
     }
 903  
 
 904  
     /**
 905  
      * This overridden method is used to save the {@link DocumentHeader} object due to the system not being able to manage the {@link DocumentHeader} object via mapping files
 906  
      *
 907  
      * @see org.kuali.rice.kns.bo.PersistableBusinessObjectBase#preUpdate()
 908  
      */
 909  
     @Override
 910  
     protected void preUpdate() {
 911  0
         super.preUpdate();
 912  0
         getDocumentHeaderService().saveDocumentHeader(getDocumentHeader());
 913  0
     }
 914  
 
 915  
     /**
 916  
      * This method to check whether the document class implements SessionDocument
 917  
      *
 918  
      * @return
 919  
      */
 920  
     public boolean isSessionDocument() {
 921  0
         return SessionDocument.class.isAssignableFrom(this.getClass());
 922  
     }
 923  
 
 924  
     /**
 925  
      * Returns whether or not the new maintainable object supports custom lock descriptors. Will always return false if the new maintainable is null.
 926  
      *
 927  
      * @see org.kuali.rice.kns.document.Document#useCustomLockDescriptors()
 928  
      * @see org.kuali.rice.kns.maintenance.Maintainable#useCustomLockDescriptors()
 929  
      */
 930  
     @Override
 931  
     public boolean useCustomLockDescriptors() {
 932  0
         return (newMaintainableObject != null && newMaintainableObject.useCustomLockDescriptors());
 933  
     }
 934  
 
 935  
     /**
 936  
      * Returns the custom lock descriptor generated by the new maintainable object, if defined. Will throw a PessimisticLockingException if
 937  
      * the new maintainable is null.
 938  
      *
 939  
      * @see org.kuali.rice.kns.document.Document#getCustomLockDescriptor(org.kuali.rice.kim.bo.Person)
 940  
      * @see org.kuali.rice.kns.maintenance.Maintainable#getCustomLockDescriptor(org.kuali.rice.kim.bo.Person)
 941  
      */
 942  
     @Override
 943  
     public String getCustomLockDescriptor(Person user) {
 944  0
         if (newMaintainableObject == null) {
 945  0
             throw new PessimisticLockingException("Maintenance Document " + getDocumentNumber() +
 946  
                     " is using pessimistic locking with custom lock descriptors, but no new maintainable object has been defined");
 947  
         }
 948  0
         return newMaintainableObject.getCustomLockDescriptor(user);
 949  
     }
 950  
 
 951  
     protected MaintenanceDocumentDictionaryService getMaintenanceDocumentDictionaryService() {
 952  0
         if (maintenanceDocumentDictionaryService == null) {
 953  0
             maintenanceDocumentDictionaryService = KNSServiceLocatorWeb.getMaintenanceDocumentDictionaryService();
 954  
         }
 955  0
         return maintenanceDocumentDictionaryService;
 956  
     }
 957  
 
 958  
     protected MaintenanceDocumentService getMaintenanceDocumentService() {
 959  0
         if (maintenanceDocumentService == null) {
 960  0
             maintenanceDocumentService = KNSServiceLocatorWeb.getMaintenanceDocumentService();
 961  
         }
 962  0
         return maintenanceDocumentService;
 963  
     }
 964  
 
 965  
     protected DocumentHeaderService getDocumentHeaderService() {
 966  0
         if (documentHeaderService == null) {
 967  0
             documentHeaderService = KNSServiceLocatorWeb.getDocumentHeaderService();
 968  
         }
 969  0
         return documentHeaderService;
 970  
     }
 971  
 
 972  
     protected DocumentService getDocumentService() {
 973  0
         if (documentService == null) {
 974  0
             documentService = KNSServiceLocatorWeb.getDocumentService();
 975  
         }
 976  0
         return documentService;
 977  
     }
 978  
 
 979  
     //for issue KULRice3070
 980  
     protected boolean checkAllowsRecordDeletion() {
 981  0
         Boolean allowsRecordDeletion = KNSServiceLocatorWeb.getMaintenanceDocumentDictionaryService().getAllowsRecordDeletion(this.getNewMaintainableObject().getBoClass());
 982  0
         if (allowsRecordDeletion != null) {
 983  0
             return allowsRecordDeletion.booleanValue();
 984  
         } else {
 985  0
             return false;
 986  
         }
 987  
     }
 988  
 
 989  
     //for KULRice3070
 990  
     protected boolean checkMaintenanceAction() {
 991  0
         return this.getNewMaintainableObject().getMaintenanceAction().equals(KNSConstants.MAINTENANCE_DELETE_ACTION);
 992  
     }
 993  
 
 994  
     //for KULRice3070
 995  
     protected boolean checkDeletePermission(BusinessObject businessObject) {
 996  
 
 997  0
         boolean allowsMaintain = false;
 998  
 
 999  0
         String maintDocTypeName = KNSServiceLocatorWeb.getMaintenanceDocumentDictionaryService().getDocumentTypeName(businessObject.getClass());
 1000  
 
 1001  0
         if (StringUtils.isNotBlank(maintDocTypeName)) {
 1002  0
             allowsMaintain = KNSServiceLocatorWeb.getBusinessObjectAuthorizationService().canMaintain(businessObject, GlobalVariables.getUserSession().getPerson(), maintDocTypeName);
 1003  
         }
 1004  0
         return allowsMaintain;
 1005  
     }
 1006  
 }