001    /**
002     * Copyright 2005-2013 The Kuali Foundation
003     *
004     * Licensed under the Educational Community License, Version 2.0 (the "License");
005     * you may not use this file except in compliance with the License.
006     * You may obtain a copy of the License at
007     *
008     * http://www.opensource.org/licenses/ecl2.php
009     *
010     * Unless required by applicable law or agreed to in writing, software
011     * distributed under the License is distributed on an "AS IS" BASIS,
012     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013     * See the License for the specific language governing permissions and
014     * limitations under the License.
015     */
016    package org.kuali.rice.krad.maintenance;
017    
018    import org.apache.commons.collections.CollectionUtils;
019    import org.apache.commons.lang.StringUtils;
020    import org.apache.ojb.broker.core.proxy.ProxyHelper;
021    import org.kuali.rice.core.api.config.property.ConfigContext;
022    import org.kuali.rice.core.api.util.RiceKeyConstants;
023    import org.kuali.rice.kew.api.KewApiServiceLocator;
024    import org.kuali.rice.kew.api.WorkflowDocument;
025    import org.kuali.rice.kew.api.doctype.DocumentType;
026    import org.kuali.rice.kew.framework.postprocessor.DocumentRouteStatusChange;
027    import org.kuali.rice.kim.api.identity.Person;
028    import org.kuali.rice.krad.bo.DocumentAttachment;
029    import org.kuali.rice.krad.bo.DocumentHeader;
030    import org.kuali.rice.krad.bo.GlobalBusinessObject;
031    import org.kuali.rice.krad.bo.MultiDocumentAttachment;
032    import org.kuali.rice.krad.bo.Note;
033    import org.kuali.rice.krad.bo.PersistableAttachment;
034    import org.kuali.rice.krad.bo.PersistableAttachmentList;
035    import org.kuali.rice.krad.bo.PersistableBusinessObject;
036    import org.kuali.rice.krad.datadictionary.DocumentEntry;
037    import org.kuali.rice.krad.datadictionary.WorkflowAttributes;
038    import org.kuali.rice.krad.datadictionary.WorkflowProperties;
039    import org.kuali.rice.krad.document.DocumentBase;
040    import org.kuali.rice.krad.document.SessionDocument;
041    import org.kuali.rice.krad.exception.PessimisticLockingException;
042    import org.kuali.rice.krad.exception.ValidationException;
043    import org.kuali.rice.krad.rules.rule.event.KualiDocumentEvent;
044    import org.kuali.rice.krad.rules.rule.event.SaveDocumentEvent;
045    import org.kuali.rice.krad.service.DocumentDictionaryService;
046    import org.kuali.rice.krad.service.DocumentHeaderService;
047    import org.kuali.rice.krad.service.DocumentService;
048    import org.kuali.rice.krad.service.KRADServiceLocator;
049    import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
050    import org.kuali.rice.krad.service.MaintenanceDocumentService;
051    import org.kuali.rice.krad.util.GlobalVariables;
052    import org.kuali.rice.krad.util.KRADConstants;
053    import org.kuali.rice.krad.util.NoteType;
054    import org.kuali.rice.krad.util.ObjectUtils;
055    import org.kuali.rice.krad.util.documentserializer.PropertySerializabilityEvaluator;
056    import org.w3c.dom.Document;
057    import org.w3c.dom.Node;
058    import org.w3c.dom.NodeList;
059    import org.xml.sax.InputSource;
060    import org.xml.sax.SAXException;
061    
062    import javax.persistence.CascadeType;
063    import javax.persistence.Column;
064    import javax.persistence.Entity;
065    import javax.persistence.FetchType;
066    import javax.persistence.JoinColumn;
067    import javax.persistence.ManyToMany;
068    import javax.persistence.ManyToOne;
069    import javax.persistence.Table;
070    import javax.persistence.Transient;
071    import javax.xml.parsers.DocumentBuilder;
072    import javax.xml.parsers.DocumentBuilderFactory;
073    import javax.xml.parsers.ParserConfigurationException;
074    import java.io.IOException;
075    import java.io.StringReader;
076    import java.util.ArrayList;
077    import java.util.Arrays;
078    import java.util.Collections;
079    import java.util.List;
080    
081    /**
082     * Document class for all maintenance documents which wraps the maintenance object in
083     * a <code>Maintainable</code> that is also used for various callbacks
084     *
085     * <p>
086     * The maintenance xml structure will be: <maintainableDocumentContents maintainableImplClass="className">
087     * <oldMaintainableObject>... </oldMaintainableObject> <newMaintainableObject>... </newMaintainableObject>
088     * </maintainableDocumentContents> Maintenance Document
089     * </p>
090     *
091     * @author Kuali Rice Team (rice.collab@kuali.org)
092     */
093    @Entity
094    @Table(name = "KRNS_MAINT_DOC_T")
095    public class MaintenanceDocumentBase extends DocumentBase implements MaintenanceDocument, SessionDocument {
096        private static final long serialVersionUID = -505085142412593305L;
097        private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(MaintenanceDocumentBase.class);
098    
099        public static final String MAINTAINABLE_IMPL_CLASS = "maintainableImplClass";
100        public static final String OLD_MAINTAINABLE_TAG_NAME = "oldMaintainableObject";
101        public static final String NEW_MAINTAINABLE_TAG_NAME = "newMaintainableObject";
102        public static final String MAINTENANCE_ACTION_TAG_NAME = "maintenanceAction";
103        public static final String NOTES_TAG_NAME = "notes";
104    
105        @Transient
106        private static transient DocumentDictionaryService documentDictionaryService;
107        @Transient
108        private static transient MaintenanceDocumentService maintenanceDocumentService;
109        @Transient
110        private static transient DocumentHeaderService documentHeaderService;
111        @Transient
112        private static transient DocumentService documentService;
113    
114        @Transient
115        protected Maintainable oldMaintainableObject;
116        @Transient
117        protected Maintainable newMaintainableObject;
118    
119        @Column(name = "DOC_CNTNT", length = 4096)
120        protected String xmlDocumentContents;
121        @Transient
122        protected boolean fieldsClearedOnCopy;
123        @Transient
124        protected boolean displayTopicFieldInNotes = false;
125        @Transient
126        protected String attachmentPropertyName;
127        @Transient
128        protected String attachmentListPropertyName;
129        @Transient
130        protected String attachmentCollectionName;
131    
132        @ManyToOne(fetch = FetchType.LAZY,
133                cascade = {CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE}) @JoinColumn(name = "DOC_HDR_ID",
134                insertable = false, updatable = false)
135        protected DocumentAttachment attachment;
136    
137        @ManyToMany(fetch = FetchType.LAZY,
138                cascade = {CascadeType.PERSIST, CascadeType.MERGE, CascadeType.REMOVE}) @JoinColumn(name = "DOC_HDR_ID",
139                insertable = false, updatable = false)
140        protected List<MultiDocumentAttachment> attachments;
141    
142        public String getAttachmentPropertyName() {
143            return this.attachmentPropertyName;
144        }
145    
146        public void setAttachmentPropertyName(String attachmentPropertyName) {
147            this.attachmentPropertyName = attachmentPropertyName;
148        }
149    
150        public String getAttachmentListPropertyName() {
151            return this.attachmentListPropertyName;
152        }
153    
154        public void setAttachmentListPropertyName(String attachmentListPropertyName) {
155            this.attachmentListPropertyName = attachmentListPropertyName;
156        }
157    
158        public String getAttachmentCollectionName() {
159            return this.attachmentCollectionName;
160        }
161    
162        public void setAttachmentCollectionName(String attachmentCollectionName) {
163            this.attachmentCollectionName = attachmentCollectionName;
164        }
165    
166        public MaintenanceDocumentBase() {
167            super();
168            fieldsClearedOnCopy = false;
169        }
170    
171        /**
172         * Initializies the maintainables.
173         */
174        public MaintenanceDocumentBase(String documentTypeName) {
175            this();
176            Class clazz = getDocumentDictionaryService().getMaintainableClass(documentTypeName);
177            try {
178                oldMaintainableObject = (Maintainable) clazz.newInstance();
179                newMaintainableObject = (Maintainable) clazz.newInstance();
180    
181                // initialize maintainable with a data object
182                Class<?> dataObjectClazz = getDocumentDictionaryService().getMaintenanceDataObjectClass(documentTypeName);
183                oldMaintainableObject.setDataObject(dataObjectClazz.newInstance());
184                oldMaintainableObject.setDataObjectClass(dataObjectClazz);
185                newMaintainableObject.setDataObject(dataObjectClazz.newInstance());
186                newMaintainableObject.setDataObjectClass(dataObjectClazz);
187            } catch (InstantiationException e) {
188                LOG.error("Unable to initialize maintainables of type " + clazz.getName());
189                throw new RuntimeException("Unable to initialize maintainables of type " + clazz.getName());
190            } catch (IllegalAccessException e) {
191                LOG.error("Unable to initialize maintainables of type " + clazz.getName());
192                throw new RuntimeException("Unable to initialize maintainables of type " + clazz.getName());
193            }
194        }
195    
196        /**
197         * Builds out the document title for maintenance documents
198         *
199         * <p>This will get loaded into the flex doc and passed into
200         * workflow. It will be searchable.
201         * </p>
202         *
203         * @return document title
204         */
205        @Override
206        public String getDocumentTitle() {
207            String documentTitle = "";
208    
209            documentTitle = newMaintainableObject.getDocumentTitle(this);
210            if (StringUtils.isNotBlank(documentTitle)) {
211                // if doc title has been overridden by maintainable, use it
212                return documentTitle;
213            }
214    
215            // TODO - build out with bo label once we get the data dictionary stuff in place
216            // build out the right classname
217            String className = newMaintainableObject.getDataObject().getClass().getName();
218            String truncatedClassName = className.substring(className.lastIndexOf('.') + 1);
219            if (isOldDataObjectInDocument()) {
220                documentTitle = "Edit ";
221            } else {
222                documentTitle = "New ";
223            }
224            documentTitle += truncatedClassName + " - ";
225            documentTitle += this.getDocumentHeader().getDocumentDescription() + " ";
226            return documentTitle;
227        }
228    
229        /**
230         * Check if oldMaintainable is specified in the XML of the maintenance document
231         *
232         * @param xmlDocument Maintenance document in XML form
233         * @return true if an oldMainainable exists in the xmlDocument, false otherwise
234         */
235        protected boolean isOldMaintainableInDocument(Document xmlDocument) {
236            boolean isOldMaintainableInExistence = false;
237            if (xmlDocument.getElementsByTagName(OLD_MAINTAINABLE_TAG_NAME).getLength() > 0) {
238                isOldMaintainableInExistence = true;
239            }
240            return isOldMaintainableInExistence;
241        }
242    
243        /**
244         * @see org.kuali.rice.krad.maintenance.Maintainable#isOldDataObjectInDocument()
245         */
246        @Override
247        public boolean isOldDataObjectInDocument() {
248            boolean isOldBusinessObjectInExistence = false;
249            if (oldMaintainableObject == null || oldMaintainableObject.getDataObject() == null) {
250                isOldBusinessObjectInExistence = false;
251            } else {
252                isOldBusinessObjectInExistence = oldMaintainableObject.isOldDataObjectInDocument();
253            }
254            return isOldBusinessObjectInExistence;
255        }
256    
257        /**
258         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#isNew()
259         */
260        @Override
261        public boolean isNew() {
262            return MaintenanceUtils.isMaintenanceDocumentCreatingNewRecord(newMaintainableObject.getMaintenanceAction());
263        }
264    
265        /**
266         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#isEdit()
267         */
268        @Override
269        public boolean isEdit() {
270            if (KRADConstants.MAINTENANCE_EDIT_ACTION.equalsIgnoreCase(newMaintainableObject.getMaintenanceAction())) {
271                return true;
272            } else {
273                return false;
274            }
275        }
276    
277        /**
278         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#isNewWithExisting()
279         */
280        @Override
281        public boolean isNewWithExisting() {
282            if (KRADConstants.MAINTENANCE_NEWWITHEXISTING_ACTION.equalsIgnoreCase(
283                    newMaintainableObject.getMaintenanceAction())) {
284                return true;
285            } else {
286                return false;
287            }
288        }
289    
290        /**
291         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#populateMaintainablesFromXmlDocumentContents()
292         */
293        @Override
294        public void populateMaintainablesFromXmlDocumentContents() {
295            // get a hold of the parsed xml document, then read the classname,
296            // then instantiate one to two instances depending on content
297            // then populate those instances
298            if (!StringUtils.isEmpty(xmlDocumentContents)) {
299                DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
300                try {
301                    DocumentBuilder builder = factory.newDocumentBuilder();
302                    Document xmlDocument = builder.parse(new InputSource(new StringReader(xmlDocumentContents)));
303                    String clazz = xmlDocument.getDocumentElement().getAttribute(MAINTAINABLE_IMPL_CLASS);
304                    if (isOldMaintainableInDocument(xmlDocument)) {
305                        oldMaintainableObject = (Maintainable) Class.forName(clazz).newInstance();
306                        Object dataObject = getDataObjectFromXML(OLD_MAINTAINABLE_TAG_NAME);
307    
308                        String oldMaintenanceAction = getMaintenanceAction(xmlDocument, OLD_MAINTAINABLE_TAG_NAME);
309                        oldMaintainableObject.setMaintenanceAction(oldMaintenanceAction);
310    
311                        oldMaintainableObject.setDataObject(dataObject);
312                        oldMaintainableObject.setDataObjectClass(dataObject.getClass());
313                    }
314                    newMaintainableObject = (Maintainable) Class.forName(clazz).newInstance();
315                    Object bo = getDataObjectFromXML(NEW_MAINTAINABLE_TAG_NAME);
316                    newMaintainableObject.setDataObject(bo);
317                    newMaintainableObject.setDataObjectClass(bo.getClass());
318    
319                    String newMaintenanceAction = getMaintenanceAction(xmlDocument, NEW_MAINTAINABLE_TAG_NAME);
320                    newMaintainableObject.setMaintenanceAction(newMaintenanceAction);
321    
322                    if (newMaintainableObject.isNotesEnabled()) {
323                        List<Note> notes = getNotesFromXml(NOTES_TAG_NAME);
324                        setNotes(notes);
325                    }
326                } catch (ParserConfigurationException e) {
327                    LOG.error("Error while parsing document contents", e);
328                    throw new RuntimeException("Could not load document contents from xml", e);
329                } catch (SAXException e) {
330                    LOG.error("Error while parsing document contents", e);
331                    throw new RuntimeException("Could not load document contents from xml", e);
332                } catch (IOException e) {
333                    LOG.error("Error while parsing document contents", e);
334                    throw new RuntimeException("Could not load document contents from xml", e);
335                } catch (InstantiationException e) {
336                    LOG.error("Error while parsing document contents", e);
337                    throw new RuntimeException("Could not load document contents from xml", e);
338                } catch (IllegalAccessException e) {
339                    LOG.error("Error while parsing document contents", e);
340                    throw new RuntimeException("Could not load document contents from xml", e);
341                } catch (ClassNotFoundException e) {
342                    LOG.error("Error while parsing document contents", e);
343                    throw new RuntimeException("Could not load document contents from xml", e);
344                }
345            }
346        }
347    
348        /**
349         * This method is a lame containment of ugly DOM walking code. This is ONLY necessary because of the version
350         * conflicts between Xalan.jar in 2.6.x and 2.7. As soon as we can upgrade to 2.7, this will be switched to using
351         * XPath, which is faster and much easier on the eyes.
352         *
353         * @param xmlDocument
354         * @param oldOrNewElementName - String oldMaintainableObject or newMaintainableObject
355         * @return the value of the element, or null if none was there
356         */
357        protected String getMaintenanceAction(Document xmlDocument, String oldOrNewElementName) {
358            if (StringUtils.isBlank(oldOrNewElementName)) {
359                throw new IllegalArgumentException("oldOrNewElementName may not be blank, null, or empty-string.");
360            }
361    
362            String maintenanceAction = null;
363            NodeList rootChildren = xmlDocument.getDocumentElement().getChildNodes();
364            for (int i = 0; i < rootChildren.getLength(); i++) {
365                Node rootChild = rootChildren.item(i);
366                if (oldOrNewElementName.equalsIgnoreCase(rootChild.getNodeName())) {
367                    NodeList maintChildren = rootChild.getChildNodes();
368                    for (int j = 0; j < maintChildren.getLength(); j++) {
369                        Node maintChild = maintChildren.item(j);
370                        if (MAINTENANCE_ACTION_TAG_NAME.equalsIgnoreCase(maintChild.getNodeName())) {
371                            maintenanceAction = maintChild.getChildNodes().item(0).getNodeValue();
372                        }
373                    }
374                }
375            }
376            return maintenanceAction;
377        }
378    
379        /**
380         * Get notes from XML
381         *
382         * @param notesTagName the xml tag name of the notes
383         * @return list of <code>Note</code>s
384         */
385        private List<Note> getNotesFromXml(String notesTagName) {
386            String notesXml = StringUtils.substringBetween(xmlDocumentContents, "<" + notesTagName + ">",
387                    "</" + notesTagName + ">");
388            if (StringUtils.isBlank(notesXml)) {
389                return Collections.emptyList();
390            }
391            List<Note> notes = (List<Note>) KRADServiceLocator.getXmlObjectSerializerService().fromXml(notesXml);
392            if (notes == null) {
393                return Collections.emptyList();
394            }
395            return notes;
396        }
397    
398        /**
399         * Get data object from XML
400         *
401         * <p>
402         * Retrieves substring of document contents from maintainable tag name. Then use xml service to translate xml into
403         * a business object.
404         * </p>
405         *
406         * @param maintainableTagName the xml tag name of the maintainable
407         * @return data object
408         */
409        protected Object getDataObjectFromXML(String maintainableTagName) {
410            String maintXml = StringUtils.substringBetween(xmlDocumentContents, "<" + maintainableTagName + ">",
411                    "</" + maintainableTagName + ">");
412    
413            boolean ignoreMissingFields = false;
414            String classAndDocTypeNames = ConfigContext.getCurrentContextConfig().getProperty(KRADConstants.Config.IGNORE_MISSIONG_FIELDS_ON_DESERIALIZE);
415            if (!StringUtils.isEmpty(classAndDocTypeNames)) {
416                String classNameOnXML = StringUtils.substringBetween(xmlDocumentContents, "<" + maintainableTagName + "><", ">");
417                String classNamesNoSpaces = removeSpacesAround(classAndDocTypeNames);
418                List<String> classAndDocTypeNamesList = Arrays.asList(org.apache.commons.lang.StringUtils.split(classNamesNoSpaces, ","));
419                String originalDocTypeId = getDocumentHeader().getWorkflowDocument().getDocumentTypeId();
420                DocumentType docType = KewApiServiceLocator.getDocumentTypeService().getDocumentTypeById(originalDocTypeId);
421    
422                while (docType != null && !ignoreMissingFields) {
423                    for(String classNameOrDocTypeName : classAndDocTypeNamesList){
424                        if (docType.getName().equalsIgnoreCase(classNameOrDocTypeName) ||
425                            classNameOnXML.equalsIgnoreCase(classNameOrDocTypeName)) {
426                                ignoreMissingFields = true;
427                                break;
428                        }
429                    }
430                    if (!StringUtils.isEmpty(docType.getParentId())) {
431                        docType = KewApiServiceLocator.getDocumentTypeService().getDocumentTypeById(docType.getParentId());
432                    } else {
433                        docType = null;
434                    }
435                }
436            }
437            if (!ignoreMissingFields) {
438                return KRADServiceLocator.getXmlObjectSerializerService().fromXml(maintXml);
439            } else {
440                return KRADServiceLocator.getXmlObjectSerializerIgnoreMissingFieldsService().fromXml(maintXml);
441            }
442        }
443    
444        /**
445         * Removes the spaces around the elements on a csv list of elements.
446         * <p>
447         * A null input will return a null output.
448         * </p>
449         *
450         * @param csv a list of elements in csv format e.g. foo, bar, baz
451         * @return a list of elements in csv format without spaces e.g. foo,bar,baz
452         */
453        private String removeSpacesAround(String csv) {
454            if (csv == null) {
455                return null;
456            }
457    
458            final StringBuilder result = new StringBuilder();
459            for (final String value : csv.split(",")) {
460                if (!"".equals(value.trim())) {
461                    result.append(value.trim());
462                    result.append(",");
463                }
464            }
465    
466            //remove trailing comma
467            int i = result.lastIndexOf(",");
468            if (i != -1) {
469                result.deleteCharAt(i);
470            }
471    
472            return result.toString();
473        }
474    
475        /**
476         * Populates the xml document contents from the maintainables.
477         *
478         * @see MaintenanceDocument#populateXmlDocumentContentsFromMaintainables()
479         */
480        @Override
481        public void populateXmlDocumentContentsFromMaintainables() {
482            StringBuilder docContentBuffer = new StringBuilder();
483            docContentBuffer.append("<maintainableDocumentContents maintainableImplClass=\"").append(
484                    newMaintainableObject.getClass().getName()).append("\">");
485    
486            // if business objects notes are enabled then we need to persist notes to the XML
487            if (getNewMaintainableObject().isNotesEnabled()) {
488                docContentBuffer.append("<" + NOTES_TAG_NAME + ">");
489                // copy notes to a non-ojb Proxied ArrayList to get rid of the usage of those proxies
490                // note: XmlObjectSerializerServiceImpl should be doing this for us but it does not
491                // appear to be working (at least in this case) and the xml comes through
492                // with the fully qualified ListProxyDefault class name from OJB embedded inside it.
493                List<Note> noteList = new ArrayList<Note>();
494                for (Note note : getNotes()) {
495                    noteList.add(note);
496                }
497                docContentBuffer.append(KRADServiceLocator.getXmlObjectSerializerService().toXml(noteList));
498                docContentBuffer.append("</" + NOTES_TAG_NAME + ">");
499            }
500            if (oldMaintainableObject != null && oldMaintainableObject.getDataObject() != null) {
501                // TODO: refactor this out into a method
502                docContentBuffer.append("<" + OLD_MAINTAINABLE_TAG_NAME + ">");
503    
504                Object oldBo = oldMaintainableObject.getDataObject();
505    
506                // hack to resolve XStream not dealing well with Proxies
507                if (oldBo instanceof PersistableBusinessObject) {
508                    ObjectUtils.materializeAllSubObjects((PersistableBusinessObject) oldBo);
509                }
510    
511                docContentBuffer.append(
512                        KRADServiceLocator.getBusinessObjectSerializerService().serializeBusinessObjectToXml(oldBo));
513    
514                // add the maintainable's maintenanceAction
515                docContentBuffer.append("<" + MAINTENANCE_ACTION_TAG_NAME + ">");
516                docContentBuffer.append(oldMaintainableObject.getMaintenanceAction());
517                docContentBuffer.append("</" + MAINTENANCE_ACTION_TAG_NAME + ">\n");
518    
519                docContentBuffer.append("</" + OLD_MAINTAINABLE_TAG_NAME + ">");
520            }
521            docContentBuffer.append("<" + NEW_MAINTAINABLE_TAG_NAME + ">");
522    
523            Object newBo = newMaintainableObject.getDataObject();
524    
525            if (newBo instanceof PersistableBusinessObject) {
526                // hack to resolve XStream not dealing well with Proxies
527                ObjectUtils.materializeAllSubObjects((PersistableBusinessObject) newBo);
528            }
529    
530            docContentBuffer.append(KRADServiceLocator.getBusinessObjectSerializerService().serializeBusinessObjectToXml(
531                    newBo));
532    
533            // add the maintainable's maintenanceAction
534            docContentBuffer.append("<" + MAINTENANCE_ACTION_TAG_NAME + ">");
535            docContentBuffer.append(newMaintainableObject.getMaintenanceAction());
536            docContentBuffer.append("</" + MAINTENANCE_ACTION_TAG_NAME + ">\n");
537    
538            docContentBuffer.append("</" + NEW_MAINTAINABLE_TAG_NAME + ">");
539            docContentBuffer.append("</maintainableDocumentContents>");
540            xmlDocumentContents = docContentBuffer.toString();
541        }
542    
543        /**
544         * @see org.kuali.rice.krad.document.DocumentBase#doRouteStatusChange(org.kuali.rice.kew.framework.postprocessor.DocumentRouteStatusChange)
545         */
546        @Override
547        public void doRouteStatusChange(DocumentRouteStatusChange statusChangeEvent) {
548            super.doRouteStatusChange(statusChangeEvent);
549    
550            WorkflowDocument workflowDocument = getDocumentHeader().getWorkflowDocument();
551            getNewMaintainableObject().doRouteStatusChange(getDocumentHeader());
552            // commit the changes to the Maintainable BusinessObject when it goes to Processed (ie, fully approved),
553            // and also unlock it
554            if (workflowDocument.isProcessed()) {
555                String documentNumber = getDocumentHeader().getDocumentNumber();
556                newMaintainableObject.setDocumentNumber(documentNumber);
557    
558                //Populate Attachment Property
559                if (newMaintainableObject.getDataObject() instanceof PersistableAttachment) {
560                    populateAttachmentBeforeSave();
561                }
562    
563                //Populate Attachment Property
564                if (newMaintainableObject.getDataObject() instanceof PersistableAttachmentList) {
565                    populateBoAttachmentListBeforeSave();
566                }
567    
568                newMaintainableObject.saveDataObject();
569    
570                if (!getDocumentService().saveDocumentNotes(this)) {
571                    throw new IllegalStateException(
572                            "Failed to save document notes, this means that the note target was not ready for notes to be attached when it should have been.");
573                }
574    
575                //Attachment should be deleted from Maintenance Document attachment table
576                deleteDocumentAttachment();
577                deleteDocumentAttachmentList();
578    
579                getMaintenanceDocumentService().deleteLocks(documentNumber);
580    
581                //for issue 3070, check if delete record
582                if (this.checkAllowsRecordDeletion() && this.checkMaintenanceAction() &&
583                        this.checkDeletePermission(newMaintainableObject.getDataObject())) {
584                    newMaintainableObject.deleteDataObject();
585                }
586            }
587    
588            // unlock the document when its canceled or disapproved
589            if (workflowDocument.isCanceled() || workflowDocument.isDisapproved() || workflowDocument.isRecalled()) {
590                //Attachment should be deleted from Maintenance Document attachment table
591                deleteDocumentAttachment();
592                deleteDocumentAttachmentList();
593    
594                String documentNumber = getDocumentHeader().getDocumentNumber();
595                getMaintenanceDocumentService().deleteLocks(documentNumber);
596            }
597        }
598    
599        /**
600         * @see org.kuali.rice.krad.document.DocumentBase#getWorkflowEngineDocumentIdsToLock()
601         */
602        @Override
603        public List<String> getWorkflowEngineDocumentIdsToLock() {
604            if (newMaintainableObject != null) {
605                return newMaintainableObject.getWorkflowEngineDocumentIdsToLock();
606            }
607            return Collections.emptyList();
608        }
609    
610        /**
611         * @see org.kuali.rice.krad.document.Document#prepareForSave()
612         */
613        @Override
614        public void prepareForSave() {
615            if (newMaintainableObject != null) {
616                newMaintainableObject.prepareForSave();
617            }
618        }
619    
620        /**
621         * @see org.kuali.rice.krad.document.DocumentBase#processAfterRetrieve()
622         */
623        @Override
624        public void processAfterRetrieve() {
625    
626            super.processAfterRetrieve();
627    
628            populateMaintainablesFromXmlDocumentContents();
629            if (oldMaintainableObject != null) {
630                oldMaintainableObject.setDocumentNumber(documentNumber);
631            }
632            if (newMaintainableObject != null) {
633                newMaintainableObject.setDocumentNumber(documentNumber);
634                newMaintainableObject.processAfterRetrieve();
635                if (newMaintainableObject.getDataObject() instanceof PersistableAttachment) {
636                    populateAttachmentForBO();
637                }
638                if (newMaintainableObject.getDataObject() instanceof PersistableAttachmentList) {
639                    populateAttachmentListForBO();
640                }
641                // If a maintenance lock exists, warn the user.
642                checkForLockingDocument(false);
643            }
644        }
645    
646        /**
647         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#getNewMaintainableObject()
648         */
649        @Override
650        public Maintainable getNewMaintainableObject() {
651            return newMaintainableObject;
652        }
653    
654        /**
655         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#setNewMaintainableObject(Maintainable)
656         */
657        @Override
658        public void setNewMaintainableObject(Maintainable newMaintainableObject) {
659            this.newMaintainableObject = newMaintainableObject;
660        }
661    
662        /**
663         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#getOldMaintainableObject()
664         */
665        @Override
666        public Maintainable getOldMaintainableObject() {
667            return oldMaintainableObject;
668        }
669    
670        /**
671         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#setOldMaintainableObject(Maintainable)
672         */
673        @Override
674        public void setOldMaintainableObject(Maintainable oldMaintainableObject) {
675            this.oldMaintainableObject = oldMaintainableObject;
676        }
677    
678        /**
679         * @see org.kuali.rice.krad.document.DocumentBase#setDocumentNumber(java.lang.String)
680         */
681        @Override
682        public void setDocumentNumber(String documentNumber) {
683            super.setDocumentNumber(documentNumber);
684    
685            // set the finDocNumber on the Maintainable
686            oldMaintainableObject.setDocumentNumber(documentNumber);
687            newMaintainableObject.setDocumentNumber(documentNumber);
688        }
689    
690        /**
691         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#isFieldsClearedOnCopy()
692         */
693        @Override
694        public final boolean isFieldsClearedOnCopy() {
695            return fieldsClearedOnCopy;
696        }
697    
698        /**
699         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#setFieldsClearedOnCopy(boolean)
700         */
701        @Override
702        public final void setFieldsClearedOnCopy(boolean fieldsClearedOnCopy) {
703            this.fieldsClearedOnCopy = fieldsClearedOnCopy;
704        }
705    
706        /**
707         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#getXmlDocumentContents()
708         */
709        @Override
710        public String getXmlDocumentContents() {
711            return xmlDocumentContents;
712        }
713    
714        /**
715         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#setXmlDocumentContents(String)
716         */
717        @Override
718        public void setXmlDocumentContents(String xmlDocumentContents) {
719            this.xmlDocumentContents = xmlDocumentContents;
720        }
721    
722        /**
723         * @see org.kuali.rice.krad.document.Document#getAllowsCopy()
724         */
725        @Override
726        public boolean getAllowsCopy() {
727            return getDocumentDictionaryService().getAllowsCopy(this);
728        }
729    
730        /**
731         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#isDisplayTopicFieldInNotes()
732         */
733        @Override
734        public boolean isDisplayTopicFieldInNotes() {
735            return displayTopicFieldInNotes;
736        }
737    
738        /**
739         * @see MaintenanceDocument#setDisplayTopicFieldInNotes(boolean)
740         */
741        @Override
742        public void setDisplayTopicFieldInNotes(boolean displayTopicFieldInNotes) {
743            this.displayTopicFieldInNotes = displayTopicFieldInNotes;
744        }
745    
746        /**
747         * Overridden to avoid serializing the xml twice, because of the xmlDocumentContents property of this object
748         */
749        @Override
750        public String serializeDocumentToXml() {
751            String tempXmlDocumentContents = xmlDocumentContents;
752            xmlDocumentContents = null;
753            String xmlForWorkflow = super.serializeDocumentToXml();
754            xmlDocumentContents = tempXmlDocumentContents;
755            return xmlForWorkflow;
756        }
757    
758        /**
759         * @see DocumentBase#prepareForSave(org.kuali.rice.krad.rules.rule.event.KualiDocumentEvent)
760         */
761        @Override
762        public void prepareForSave(KualiDocumentEvent event) {
763            super.prepareForSave(event);
764            if (newMaintainableObject.getDataObject() instanceof PersistableAttachment) {
765                populateDocumentAttachment();
766                populateAttachmentForBO();
767                //clear out attachment file for old data object so it isn't serialized in doc content
768                if (oldMaintainableObject.getDataObject() instanceof PersistableAttachment) {
769                    ((PersistableAttachment) oldMaintainableObject.getDataObject()).setAttachmentContent(null);
770                }
771            }
772            if (newMaintainableObject.getDataObject() instanceof PersistableAttachmentList) {
773                populateDocumentAttachmentList();
774                populateAttachmentListForBO();
775                if (oldMaintainableObject.getDataObject() instanceof PersistableAttachmentList) {
776                    for (PersistableAttachment pa : ((PersistableAttachmentList<PersistableAttachment>) oldMaintainableObject
777                            .getDataObject()).getAttachments()) {
778                        pa.setAttachmentContent(null);
779                    }
780                }
781            }
782            populateXmlDocumentContentsFromMaintainables();
783        }
784    
785        /**
786         * The attachment BO is proxied in OJB.  For some reason when an attachment does not yet exist,
787         * refreshReferenceObject is not returning null and the proxy cannot be materialized. So, this method exists to
788         * properly handle the proxied attachment BO.  This is a hack and should be removed post JPA migration.
789         */
790        protected void refreshAttachment() {
791            if (ObjectUtils.isNull(attachment)) {
792                this.refreshReferenceObject("attachment");
793                final boolean isProxy = attachment != null && ProxyHelper.isProxy(attachment);
794                if (isProxy && ProxyHelper.getRealObject(attachment) == null) {
795                    attachment = null;
796                }
797            }
798        }
799    
800        protected void refreshAttachmentList() {
801            if (ObjectUtils.isNull(attachments)) {
802                this.refreshReferenceObject("attachments");
803                final boolean isProxy = attachments != null && ProxyHelper.isProxy(attachments);
804                if (isProxy && ProxyHelper.getRealObject(attachments) == null) {
805                    attachments = null;
806                }
807            }
808        }
809    
810        public void populateAttachmentForBO() {
811            // TODO: need to convert this from using struts form file
812    
813        }
814    
815        public void populateDocumentAttachment() {
816            // TODO: need to convert this from using struts form file
817            //        refreshAttachment();
818            //
819            //        if (fileAttachment != null && StringUtils.isNotEmpty(fileAttachment.getFileName())) {
820            //            //Populate DocumentAttachment BO
821            //            if (attachment == null) {
822            //                attachment = new DocumentAttachment();
823            //            }
824            //
825            //            byte[] fileContents;
826            //            try {
827            //                fileContents = fileAttachment.getFileData();
828            //                if (fileContents.length > 0) {
829            //                    attachment.setFileName(fileAttachment.getFileName());
830            //                    attachment.setContentType(fileAttachment.getContentType());
831            //                    attachment.setAttachmentContent(fileAttachment.getFileData());
832            //                    attachment.setDocumentNumber(getDocumentNumber());
833            //                }
834            //            } catch (FileNotFoundException e) {
835            //                LOG.error("Error while populating the Document Attachment", e);
836            //                throw new RuntimeException("Could not populate DocumentAttachment object", e);
837            //            } catch (IOException e) {
838            //                LOG.error("Error while populating the Document Attachment", e);
839            //                throw new RuntimeException("Could not populate DocumentAttachment object", e);
840            //            }
841            //        }
842            ////        else if(attachment != null) {
843            ////            //Attachment has been deleted - Need to delete the Attachment Reference Object
844            ////            deleteAttachment();
845            ////        }
846        }
847    
848        public void populateAttachmentListForBO() { }
849    
850        public void populateAttachmentBeforeSave() { }
851    
852        public void populateDocumentAttachmentList() { }
853    
854        public void populateBoAttachmentListBeforeSave() { }
855    
856        public void deleteDocumentAttachment() {
857            KRADServiceLocator.getBusinessObjectService().delete(attachment);
858            attachment = null;
859        }
860    
861        public void deleteDocumentAttachmentList() {
862            if (CollectionUtils.isNotEmpty(attachments)) {
863                KRADServiceLocator.getBusinessObjectService().delete(attachments);
864                attachments = null;
865            }
866        }
867    
868        /**
869         * Explicitly NOT calling super here.  This is a complete override of the validation rules behavior.
870         *
871         * @see org.kuali.rice.krad.document.DocumentBase#validateBusinessRules(org.kuali.rice.krad.rules.rule.event.KualiDocumentEvent)
872         */
873        @Override
874        public void validateBusinessRules(KualiDocumentEvent event) {
875            if (GlobalVariables.getMessageMap().hasErrors()) {
876                logErrors();
877                throw new ValidationException("errors occured before business rule");
878            }
879    
880            // check for locking documents for MaintenanceDocuments
881            checkForLockingDocument(true);
882    
883            // Make sure the business object's version number matches that of the databases copy.
884            if (newMaintainableObject != null) {
885                if (KRADServiceLocator.getPersistenceStructureService().isPersistable(
886                        newMaintainableObject.getDataObject().getClass())) {
887                    PersistableBusinessObject pbObject = KRADServiceLocator.getBusinessObjectService().retrieve(
888                            (PersistableBusinessObject) newMaintainableObject.getDataObject());
889                    Long pbObjectVerNbr = ObjectUtils.isNull(pbObject) ? null : pbObject.getVersionNumber();
890                    Long newObjectVerNbr =
891                            ((PersistableBusinessObject) newMaintainableObject.getDataObject()).getVersionNumber();
892    
893                    if (pbObjectVerNbr != null && !(pbObjectVerNbr.equals(newObjectVerNbr))) {
894                        GlobalVariables.getMessageMap().putError(KRADConstants.GLOBAL_ERRORS,
895                                RiceKeyConstants.ERROR_VERSION_MISMATCH);
896                        throw new ValidationException(
897                                "Version mismatch between the local business object and the database business object");
898                    }
899                }
900            }
901    
902            // perform validation against rules engine
903            if (LOG.isInfoEnabled()) {
904                LOG.info("invoking rules engine on document " + getDocumentNumber());
905            }
906    
907            boolean isValid = true;
908            isValid = KRADServiceLocatorWeb.getKualiRuleService().applyRules(event);
909    
910            // check to see if the br eval passed or failed
911            if (!isValid) {
912                logErrors();
913                // TODO: better error handling at the lower level and a better error message are
914                // needed here
915                throw new ValidationException("business rule evaluation failed");
916            } else if (GlobalVariables.getMessageMap().hasErrors()) {
917                logErrors();
918                if (event instanceof SaveDocumentEvent) {
919                    // for maintenance documents, we want to always actually do a save if the
920                    // user requests a save, even if there are validation or business rules
921                    // failures. this empty if does this, and allows the document to be saved,
922                    // even if there are failures.
923                    // BR or validation failures on a ROUTE even should always stop the route,
924                    // that has not changed
925                } else {
926                    throw new ValidationException(
927                            "Unreported errors occured during business rule evaluation (rule developer needs to put meaningful error messages into global ErrorMap)");
928                }
929            }
930    
931            LOG.debug("validation completed");
932        }
933    
934        protected void checkForLockingDocument(boolean throwExceptionIfLocked) {
935            MaintenanceUtils.checkForLockingDocument(this, throwExceptionIfLocked);
936        }
937    
938        /**
939         * this needs to happen after the document itself is saved, to preserve consistency of the ver_nbr and in the case
940         * of initial save, because this can't be saved until the document is saved initially
941         *
942         * @see org.kuali.rice.krad.document.DocumentBase#postProcessSave(org.kuali.rice.krad.rules.rule.event.KualiDocumentEvent)
943         */
944        @Override
945        public void postProcessSave(KualiDocumentEvent event) {
946            if (getNewMaintainableObject().getDataObject() instanceof PersistableBusinessObject) {
947                PersistableBusinessObject bo = (PersistableBusinessObject) getNewMaintainableObject().getDataObject();
948                if (bo instanceof GlobalBusinessObject) {
949                    KRADServiceLocator.getBusinessObjectService().save(bo);
950                }
951            }
952    
953            //currently only global documents could change the list of what they're affecting during routing,
954            //so could restrict this to only happening with them, but who knows if that will change, so safest
955            //to always do the delete and re-add...seems a bit inefficient though if nothing has changed, which is
956            //most of the time...could also try to only add/update/delete what's changed, but this is easier
957            if (!(event instanceof SaveDocumentEvent)) { //don't lock until they route
958                getMaintenanceDocumentService().deleteLocks(this.getDocumentNumber());
959                getMaintenanceDocumentService().storeLocks(this.getNewMaintainableObject().generateMaintenanceLocks());
960            }
961        }
962    
963        /**
964         * @see org.kuali.rice.krad.maintenance.MaintenanceDocument#getDocumentDataObject()
965         */
966        @Override
967        public Object getDocumentDataObject() {
968            return getNewMaintainableObject().getDataObject();
969        }
970    
971        /**
972         * <p>The Note target for maintenance documents is determined by whether or not the underlying {@link Maintainable}
973         * supports business object notes or not.  This is determined via a call to {@link
974         * Maintainable#isBoNotesEnabled()}.
975         * The note target is then derived as follows: <p/> <ul> <li>If the {@link Maintainable} supports business object
976         * notes, delegate to {@link #getDocumentDataObject()}. <li>Otherwise, delegate to the default implementation of
977         * getNoteTarget on the superclass which will effectively return a reference to the {@link DocumentHeader}. </ul>
978         *
979         * @see org.kuali.rice.krad.document.Document#getNoteTarget()
980         */
981        @Override
982        public PersistableBusinessObject getNoteTarget() {
983            if (getNewMaintainableObject() == null) {
984                throw new IllegalStateException(
985                        "Failed to acquire the note target.  The new maintainable object on this document is null.");
986            }
987            if (getNewMaintainableObject().isNotesEnabled()) {
988                return (PersistableBusinessObject) getDocumentDataObject();
989            }
990            return super.getNoteTarget();
991        }
992    
993        /**
994         * The {@link NoteType} for maintenance documents is determined by whether or not the underlying {@link
995         * Maintainable} supports business object notes or not.  This is determined via a call to {@link
996         * Maintainable#   isBoNotesEnabled()}.  The {@link NoteType} is then derived as follows: <p/> <ul> <li>If the
997         * {@link
998         * Maintainable} supports business object notes, return {@link NoteType#BUSINESS_OBJECT}. <li>Otherwise, delegate
999         * to
1000         * {@link DocumentBase#getNoteType()} </ul>
1001         *
1002         * @see org.kuali.rice.krad.document.Document#getNoteType()
1003         * @see org.kuali.rice.krad.document.Document#getNoteTarget()
1004         */
1005        @Override
1006        public NoteType getNoteType() {
1007            if (getNewMaintainableObject().isNotesEnabled()) {
1008                return NoteType.BUSINESS_OBJECT;
1009            }
1010            return super.getNoteType();
1011        }
1012    
1013        @Override
1014        public PropertySerializabilityEvaluator getDocumentPropertySerizabilityEvaluator() {
1015            String docTypeName = "";
1016            if (newMaintainableObject != null) {
1017                docTypeName = getDocumentDictionaryService().getMaintenanceDocumentTypeName(
1018                        this.newMaintainableObject.getDataObjectClass());
1019            } else { // I don't know why we aren't just using the header in the first place
1020                // but, in the case where we can't get it in the way above, attempt to get
1021                // it off the workflow document header
1022                if (getDocumentHeader() != null && getDocumentHeader().getWorkflowDocument() != null) {
1023                    docTypeName = getDocumentHeader().getWorkflowDocument().getDocumentTypeName();
1024                }
1025            }
1026            if (!StringUtils.isBlank(docTypeName)) {
1027                DocumentEntry documentEntry = getDocumentDictionaryService().getMaintenanceDocumentEntry(docTypeName);
1028                if (documentEntry != null) {
1029                    WorkflowProperties workflowProperties = documentEntry.getWorkflowProperties();
1030                    WorkflowAttributes workflowAttributes = documentEntry.getWorkflowAttributes();
1031                    return createPropertySerializabilityEvaluator(workflowProperties, workflowAttributes);
1032                } else {
1033                    LOG.error("Unable to obtain DD DocumentEntry for document type: '" + docTypeName + "'");
1034                }
1035            } else {
1036                LOG.error("Unable to obtain document type name for this document: " + this);
1037            }
1038            LOG.error("Returning null for the PropertySerializabilityEvaluator");
1039            return null;
1040        }
1041    
1042        public DocumentAttachment getAttachment() {
1043            return this.attachment;
1044        }
1045    
1046        public void setAttachment(DocumentAttachment attachment) {
1047            this.attachment = attachment;
1048        }
1049    
1050        public List<MultiDocumentAttachment> getAttachments() {
1051            return this.attachments;
1052        }
1053    
1054        public void setAttachments(List<MultiDocumentAttachment> attachments) {
1055            this.attachments = attachments;
1056        }
1057    
1058        /**
1059         * This overridden method is used to delete the {@link DocumentHeader} object due to the system not being able to
1060         * manage the {@link DocumentHeader} object via mapping files
1061         *
1062         * @see org.kuali.rice.krad.bo.PersistableBusinessObjectBase#postRemove()
1063         */
1064        @Override
1065        protected void postRemove() {
1066            super.postRemove();
1067            getDocumentHeaderService().deleteDocumentHeader(getDocumentHeader());
1068        }
1069    
1070        /**
1071         * This overridden method is used to retrieve the {@link DocumentHeader} object due to the system not being able to
1072         * manage the {@link DocumentHeader} object via mapping files
1073         *
1074         * @see org.kuali.rice.krad.bo.PersistableBusinessObjectBase#postLoad()
1075         */
1076        @Override
1077        protected void postLoad() {
1078            super.postLoad();
1079            setDocumentHeader(getDocumentHeaderService().getDocumentHeaderById(getDocumentNumber()));
1080        }
1081    
1082        /**
1083         * This overridden method is used to insert the {@link DocumentHeader} object due to the system not being able to
1084         * manage the {@link DocumentHeader} object via mapping files
1085         *
1086         * @see org.kuali.rice.krad.bo.PersistableBusinessObjectBase#prePersist()
1087         */
1088        @Override
1089        protected void prePersist() {
1090            super.prePersist();
1091            getDocumentHeaderService().saveDocumentHeader(getDocumentHeader());
1092        }
1093    
1094        /**
1095         * This overridden method is used to save the {@link DocumentHeader} object due to the system not being able to
1096         * manage the {@link DocumentHeader} object via mapping files
1097         *
1098         * @see org.kuali.rice.krad.bo.PersistableBusinessObjectBase#preUpdate()
1099         */
1100        @Override
1101        protected void preUpdate() {
1102            super.preUpdate();
1103            getDocumentHeaderService().saveDocumentHeader(getDocumentHeader());
1104        }
1105    
1106        /**
1107         * This method to check whether the document class implements SessionDocument
1108         *
1109         * TODO: move to KNS maintenance document base
1110         *
1111         * @return
1112         */
1113        public boolean isSessionDocument() {
1114            return SessionDocument.class.isAssignableFrom(this.getClass());
1115        }
1116    
1117        /**
1118         * Returns whether or not the new maintainable object supports custom lock descriptors. Will always return false if
1119         * the new maintainable is null.
1120         *
1121         * @see org.kuali.rice.krad.document.Document#useCustomLockDescriptors()
1122         * @see org.kuali.rice.krad.maintenance.Maintainable#useCustomLockDescriptors()
1123         */
1124        @Override
1125        public boolean useCustomLockDescriptors() {
1126            return (newMaintainableObject != null && newMaintainableObject.useCustomLockDescriptors());
1127        }
1128    
1129        /**
1130         * Returns the custom lock descriptor generated by the new maintainable object, if defined. Will throw a
1131         * PessimisticLockingException if the new maintainable is null.
1132         *
1133         * @see org.kuali.rice.krad.document.Document#getCustomLockDescriptor(org.kuali.rice.kim.api.identity.Person)
1134         * @see org.kuali.rice.krad.maintenance.Maintainable#getCustomLockDescriptor(org.kuali.rice.kim.api.identity.Person)
1135         */
1136        @Override
1137        public String getCustomLockDescriptor(Person user) {
1138            if (newMaintainableObject == null) {
1139                throw new PessimisticLockingException("Maintenance Document " + getDocumentNumber() +
1140                        " is using pessimistic locking with custom lock descriptors, but no new maintainable object has been defined");
1141            }
1142            return newMaintainableObject.getCustomLockDescriptor(user);
1143        }
1144    
1145        protected DocumentDictionaryService getDocumentDictionaryService() {
1146            if (documentDictionaryService == null) {
1147                documentDictionaryService = KRADServiceLocatorWeb.getDocumentDictionaryService();
1148            }
1149            return documentDictionaryService;
1150        }
1151    
1152        protected MaintenanceDocumentService getMaintenanceDocumentService() {
1153            if (maintenanceDocumentService == null) {
1154                maintenanceDocumentService = KRADServiceLocatorWeb.getMaintenanceDocumentService();
1155            }
1156            return maintenanceDocumentService;
1157        }
1158    
1159        protected DocumentHeaderService getDocumentHeaderService() {
1160            if (documentHeaderService == null) {
1161                documentHeaderService = KRADServiceLocatorWeb.getDocumentHeaderService();
1162            }
1163            return documentHeaderService;
1164        }
1165    
1166        protected DocumentService getDocumentService() {
1167            if (documentService == null) {
1168                documentService = KRADServiceLocatorWeb.getDocumentService();
1169            }
1170            return documentService;
1171        }
1172    
1173        //for issue KULRice3070
1174        protected boolean checkAllowsRecordDeletion() {
1175            Boolean allowsRecordDeletion = KRADServiceLocatorWeb.getDocumentDictionaryService().getAllowsRecordDeletion(
1176                    this.getNewMaintainableObject().getDataObjectClass());
1177            if (allowsRecordDeletion != null) {
1178                return allowsRecordDeletion.booleanValue();
1179            } else {
1180                return false;
1181            }
1182        }
1183    
1184        //for KULRice3070
1185        protected boolean checkMaintenanceAction() {
1186            return this.getNewMaintainableObject().getMaintenanceAction().equals(KRADConstants.MAINTENANCE_DELETE_ACTION);
1187        }
1188    
1189        //for KULRice3070
1190        protected boolean checkDeletePermission(Object dataObject) {
1191            boolean allowsMaintain = false;
1192    
1193            String maintDocTypeName = KRADServiceLocatorWeb.getDocumentDictionaryService().getMaintenanceDocumentTypeName(
1194                    dataObject.getClass());
1195    
1196            if (StringUtils.isNotBlank(maintDocTypeName)) {
1197                allowsMaintain = KRADServiceLocatorWeb.getDataObjectAuthorizationService().canMaintain(dataObject,
1198                        GlobalVariables.getUserSession().getPerson(), maintDocTypeName);
1199            }
1200            return allowsMaintain;
1201        }
1202    }