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.lang.StringUtils; 019 import org.apache.ojb.broker.metadata.ClassNotPersistenceCapableException; 020 import org.kuali.rice.core.api.CoreApiServiceLocator; 021 import org.kuali.rice.core.api.encryption.EncryptionService; 022 import org.kuali.rice.kim.api.identity.Person; 023 import org.kuali.rice.krad.bo.AdHocRoutePerson; 024 import org.kuali.rice.krad.bo.AdHocRouteWorkgroup; 025 import org.kuali.rice.krad.bo.BusinessObject; 026 import org.kuali.rice.krad.bo.DocumentHeader; 027 import org.kuali.rice.krad.bo.Note; 028 import org.kuali.rice.krad.bo.PersistableBusinessObject; 029 import org.kuali.rice.krad.exception.PessimisticLockingException; 030 import org.kuali.rice.krad.service.BusinessObjectService; 031 import org.kuali.rice.krad.service.DataObjectAuthorizationService; 032 import org.kuali.rice.krad.service.DataObjectMetaDataService; 033 import org.kuali.rice.krad.service.DocumentDictionaryService; 034 import org.kuali.rice.krad.service.KRADServiceLocator; 035 import org.kuali.rice.krad.service.KRADServiceLocatorWeb; 036 import org.kuali.rice.krad.service.LookupService; 037 import org.kuali.rice.krad.service.MaintenanceDocumentService; 038 import org.kuali.rice.krad.uif.container.CollectionGroup; 039 import org.kuali.rice.krad.uif.service.impl.ViewHelperServiceImpl; 040 import org.kuali.rice.krad.uif.util.ObjectPropertyUtils; 041 import org.kuali.rice.krad.uif.view.View; 042 import org.kuali.rice.krad.util.KRADConstants; 043 import org.kuali.rice.krad.util.ObjectUtils; 044 import org.kuali.rice.krad.web.form.MaintenanceDocumentForm; 045 046 import java.security.GeneralSecurityException; 047 import java.util.ArrayList; 048 import java.util.Collection; 049 import java.util.Iterator; 050 import java.util.List; 051 import java.util.Map; 052 053 /** 054 * Default implementation of the <code>Maintainable</code> interface 055 * 056 * @author Kuali Rice Team (rice.collab@kuali.org) 057 */ 058 public class MaintainableImpl extends ViewHelperServiceImpl implements Maintainable { 059 private static final long serialVersionUID = 9125271369161634992L; 060 061 private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(MaintainableImpl.class); 062 063 private String documentNumber; 064 private Object dataObject; 065 private Class<?> dataObjectClass; 066 private String maintenanceAction; 067 068 private transient LookupService lookupService; 069 private transient DataObjectAuthorizationService dataObjectAuthorizationService; 070 private transient DataObjectMetaDataService dataObjectMetaDataService; 071 private transient DocumentDictionaryService documentDictionaryService; 072 private transient EncryptionService encryptionService; 073 private transient BusinessObjectService businessObjectService; 074 private transient MaintenanceDocumentService maintenanceDocumentService; 075 076 /** 077 * @see org.kuali.rice.krad.maintenance.Maintainable#retrieveObjectForEditOrCopy(MaintenanceDocument, java.util.Map) 078 */ 079 @Override 080 public Object retrieveObjectForEditOrCopy(MaintenanceDocument document, Map<String, String> dataObjectKeys) { 081 Object dataObject = null; 082 083 try { 084 dataObject = getLookupService().findObjectBySearch(getDataObjectClass(), dataObjectKeys); 085 } catch (ClassNotPersistenceCapableException ex) { 086 if (!document.getOldMaintainableObject().isExternalBusinessObject()) { 087 throw new RuntimeException("Data Object Class: " 088 + getDataObjectClass() 089 + " is not persistable and is not externalizable - configuration error"); 090 } 091 // otherwise, let fall through 092 } 093 094 return dataObject; 095 } 096 097 /** 098 * @see org.kuali.rice.krad.maintenance.Maintainable#setDocumentNumber 099 */ 100 @Override 101 public void setDocumentNumber(String documentNumber) { 102 this.documentNumber = documentNumber; 103 } 104 105 /** 106 * @see org.kuali.rice.krad.maintenance.Maintainable#getDocumentTitle 107 */ 108 @Override 109 public String getDocumentTitle(MaintenanceDocument document) { 110 // default implementation is to allow MaintenanceDocumentBase to 111 // generate the doc title 112 return ""; 113 } 114 115 /** 116 * @see org.kuali.rice.krad.maintenance.Maintainable#getDataObject 117 */ 118 @Override 119 public Object getDataObject() { 120 return dataObject; 121 } 122 123 /** 124 * @see org.kuali.rice.krad.maintenance.Maintainable#setDataObject 125 */ 126 @Override 127 public void setDataObject(Object object) { 128 this.dataObject = object; 129 } 130 131 /** 132 * @see org.kuali.rice.krad.maintenance.Maintainable#getDataObjectClass 133 */ 134 @Override 135 public Class getDataObjectClass() { 136 return dataObjectClass; 137 } 138 139 /** 140 * @see org.kuali.rice.krad.maintenance.Maintainable#setDataObjectClass 141 */ 142 @Override 143 public void setDataObjectClass(Class dataObjectClass) { 144 this.dataObjectClass = dataObjectClass; 145 } 146 147 /** 148 * Persistable business objects are lockable 149 * 150 * @see org.kuali.rice.krad.maintenance.Maintainable#isLockable 151 */ 152 @Override 153 public boolean isLockable() { 154 return KRADServiceLocator.getPersistenceStructureService().isPersistable(getDataObject().getClass()); 155 } 156 157 /** 158 * Returns the data object if its persistable, null otherwise 159 * 160 * @see org.kuali.rice.krad.maintenance.Maintainable#getPersistableBusinessObject 161 */ 162 @Override 163 public PersistableBusinessObject getPersistableBusinessObject() { 164 if (KRADServiceLocator.getPersistenceStructureService().isPersistable(getDataObject().getClass())) { 165 return (PersistableBusinessObject) getDataObject(); 166 } else { 167 return null; 168 } 169 170 } 171 172 /** 173 * @see org.kuali.rice.krad.maintenance.Maintainable#getMaintenanceAction 174 */ 175 @Override 176 public String getMaintenanceAction() { 177 return maintenanceAction; 178 } 179 180 /** 181 * @see org.kuali.rice.krad.maintenance.Maintainable#setMaintenanceAction 182 */ 183 @Override 184 public void setMaintenanceAction(String maintenanceAction) { 185 this.maintenanceAction = maintenanceAction; 186 } 187 188 /** 189 * Note: as currently implemented, every key field for a given 190 * data object class must have a visible getter 191 * 192 * @see org.kuali.rice.krad.maintenance.Maintainable#generateMaintenanceLocks 193 */ 194 @Override 195 public List<MaintenanceLock> generateMaintenanceLocks() { 196 List<MaintenanceLock> maintenanceLocks = new ArrayList<MaintenanceLock>(); 197 StringBuffer lockRepresentation = new StringBuffer(dataObjectClass.getName()); 198 lockRepresentation.append(KRADConstants.Maintenance.LOCK_AFTER_CLASS_DELIM); 199 200 Object bo = getDataObject(); 201 List keyFieldNames = getDocumentDictionaryService().getLockingKeys(getDocumentTypeName()); 202 203 for (Iterator i = keyFieldNames.iterator(); i.hasNext(); ) { 204 String fieldName = (String) i.next(); 205 Object fieldValue = ObjectUtils.getPropertyValue(bo, fieldName); 206 if (fieldValue == null) { 207 fieldValue = ""; 208 } 209 210 // check if field is a secure 211 if (getDataObjectAuthorizationService() 212 .attributeValueNeedsToBeEncryptedOnFormsAndLinks(dataObjectClass, fieldName)) { 213 try { 214 if(CoreApiServiceLocator.getEncryptionService().isEnabled()) { 215 fieldValue = getEncryptionService().encrypt(fieldValue); 216 } 217 } catch (GeneralSecurityException e) { 218 LOG.error("Unable to encrypt secure field for locking representation " + e.getMessage()); 219 throw new RuntimeException( 220 "Unable to encrypt secure field for locking representation " + e.getMessage()); 221 } 222 } 223 224 lockRepresentation.append(fieldName); 225 lockRepresentation.append(KRADConstants.Maintenance.LOCK_AFTER_FIELDNAME_DELIM); 226 lockRepresentation.append(String.valueOf(fieldValue)); 227 if (i.hasNext()) { 228 lockRepresentation.append(KRADConstants.Maintenance.LOCK_AFTER_VALUE_DELIM); 229 } 230 } 231 232 MaintenanceLock maintenanceLock = new MaintenanceLock(); 233 maintenanceLock.setDocumentNumber(documentNumber); 234 maintenanceLock.setLockingRepresentation(lockRepresentation.toString()); 235 maintenanceLocks.add(maintenanceLock); 236 237 return maintenanceLocks; 238 } 239 240 /** 241 * Retrieves the document type name from the data dictionary based on 242 * business object class 243 */ 244 protected String getDocumentTypeName() { 245 return getDocumentDictionaryService().getMaintenanceDocumentTypeName(dataObjectClass); 246 } 247 248 /** 249 * @see org.kuali.rice.krad.maintenance.Maintainable#saveDataObject 250 */ 251 @Override 252 public void saveDataObject() { 253 if (dataObject instanceof PersistableBusinessObject) { 254 getBusinessObjectService().linkAndSave((PersistableBusinessObject) dataObject); 255 } else { 256 throw new RuntimeException( 257 "Cannot save object of type: " + dataObjectClass + " with business object service"); 258 } 259 } 260 261 /** 262 * @see org.kuali.rice.krad.maintenance.Maintainable#deleteDataObject 263 */ 264 @Override 265 public void deleteDataObject() { 266 if (dataObject == null) { 267 return; 268 } 269 270 if (dataObject instanceof PersistableBusinessObject) { 271 getBusinessObjectService().delete((PersistableBusinessObject) dataObject); 272 dataObject = null; 273 } else { 274 throw new RuntimeException( 275 "Cannot delete object of type: " + dataObjectClass + " with business object service"); 276 } 277 } 278 279 /** 280 * @see org.kuali.rice.krad.maintenance.Maintainable#doRouteStatusChange 281 */ 282 @Override 283 public void doRouteStatusChange(DocumentHeader documentHeader) { 284 // no default implementation 285 } 286 287 /** 288 * @see org.kuali.rice.krad.maintenance.Maintainable#getLockingDocumentId 289 */ 290 @Override 291 public String getLockingDocumentId() { 292 return getMaintenanceDocumentService().getLockingDocumentId(this, documentNumber); 293 } 294 295 /** 296 * @see org.kuali.rice.krad.maintenance.Maintainable#getWorkflowEngineDocumentIdsToLock 297 */ 298 @Override 299 public List<String> getWorkflowEngineDocumentIdsToLock() { 300 return null; 301 } 302 303 /** 304 * Default implementation simply returns false to indicate that custom 305 * lock descriptors are not supported by MaintainableImpl. If custom 306 * lock descriptors are needed, the appropriate subclasses should override 307 * this method 308 * 309 * @see org.kuali.rice.krad.maintenance.Maintainable#useCustomLockDescriptors 310 */ 311 @Override 312 public boolean useCustomLockDescriptors() { 313 return false; 314 } 315 316 /** 317 * Default implementation just throws a PessimisticLockingException. 318 * Subclasses of MaintainableImpl that need support for custom lock 319 * descriptors should override this method 320 * 321 * @see org.kuali.rice.krad.maintenance.Maintainable#getCustomLockDescriptor 322 */ 323 @Override 324 public String getCustomLockDescriptor(Person user) { 325 throw new PessimisticLockingException("The Maintainable for document " + documentNumber + 326 " is using pessimistic locking with custom lock descriptors, but the Maintainable has not overridden the getCustomLockDescriptor method"); 327 } 328 329 /** 330 * @see org.kuali.rice.krad.maintenance.Maintainable#isNotesEnabled 331 */ 332 @Override 333 public boolean isNotesEnabled() { 334 return getDataObjectMetaDataService().areNotesSupported(dataObjectClass); 335 } 336 337 /** 338 * @see org.kuali.rice.krad.maintenance.MaintainableImpl#isExternalBusinessObject 339 */ 340 @Override 341 public boolean isExternalBusinessObject() { 342 return false; 343 } 344 345 /** 346 * @see org.kuali.rice.krad.maintenance.MaintainableImpl#prepareExternalBusinessObject 347 */ 348 @Override 349 public void prepareExternalBusinessObject(BusinessObject businessObject) { 350 // by default do nothing 351 } 352 353 /** 354 * Checks whether the data object is not null and has its primary key values populated 355 * 356 * @see org.kuali.rice.krad.maintenance.MaintainableImpl#isOldDataObjectInDocument 357 */ 358 @Override 359 public boolean isOldDataObjectInDocument() { 360 boolean isOldDataObjectInExistence = true; 361 362 if (getDataObject() == null) { 363 isOldDataObjectInExistence = false; 364 } else { 365 Map<String, ?> keyFieldValues = getDataObjectMetaDataService().getPrimaryKeyFieldValues(getDataObject()); 366 for (Object keyValue : keyFieldValues.values()) { 367 if (keyValue == null) { 368 isOldDataObjectInExistence = false; 369 } else if ((keyValue instanceof String) && StringUtils.isBlank((String) keyValue)) { 370 isOldDataObjectInExistence = false; 371 } 372 373 if (!isOldDataObjectInExistence) { 374 break; 375 } 376 } 377 } 378 379 return isOldDataObjectInExistence; 380 } 381 382 /** 383 * @see org.kuali.rice.krad.maintenance.Maintainable#prepareForSave 384 */ 385 @Override 386 public void prepareForSave() { 387 // by default do nothing 388 } 389 390 /** 391 * @see org.kuali.rice.krad.maintenance.Maintainable#processAfterRetrieve 392 */ 393 @Override 394 public void processAfterRetrieve() { 395 // by default do nothing 396 } 397 398 /** 399 * @see org.kuali.rice.krad.maintenance.MaintainableImpl#setupNewFromExisting 400 */ 401 @Override 402 public void setupNewFromExisting(MaintenanceDocument document, Map<String, String[]> parameters) { 403 // by default do nothing 404 } 405 406 /** 407 * @see org.kuali.rice.krad.maintenance.Maintainable#processAfterCopy 408 */ 409 @Override 410 public void processAfterCopy(MaintenanceDocument document, Map<String, String[]> requestParameters) { 411 // by default do nothing 412 } 413 414 /** 415 * @see org.kuali.rice.krad.maintenance.Maintainable#processAfterEdit 416 */ 417 @Override 418 public void processAfterEdit(MaintenanceDocument document, Map<String, String[]> requestParameters) { 419 // by default do nothing 420 } 421 422 /** 423 * @see org.kuali.rice.krad.maintenance.Maintainable#processAfterNew 424 */ 425 @Override 426 public void processAfterNew(MaintenanceDocument document, Map<String, String[]> requestParameters) { 427 // by default do nothing 428 } 429 430 /** 431 * @see org.kuali.rice.krad.maintenance.Maintainable#processAfterPost 432 */ 433 @Override 434 public void processAfterPost(MaintenanceDocument document, Map<String, String[]> requestParameters) { 435 // by default do nothing 436 } 437 438 /** 439 * In the case of edit maintenance adds a new blank line to the old side 440 * 441 * TODO: should this write some sort of missing message on the old side 442 * instead? 443 * 444 * @see org.kuali.rice.krad.uif.service.impl.ViewHelperServiceImpl#processAfterAddLine(org.kuali.rice.krad.uif.view.View, 445 * org.kuali.rice.krad.uif.container.CollectionGroup, java.lang.Object, 446 * java.lang.Object) 447 */ 448 @Override 449 protected void processAfterAddLine(View view, CollectionGroup collectionGroup, Object model, Object addLine) { 450 super.processAfterAddLine(view, collectionGroup, model, addLine); 451 452 // Check for maintenance documents in edit but exclude notes and ad hoc recipients 453 if (model instanceof MaintenanceDocumentForm 454 && KRADConstants.MAINTENANCE_EDIT_ACTION.equals(((MaintenanceDocumentForm)model).getMaintenanceAction()) && !(addLine instanceof Note) && !(addLine instanceof AdHocRoutePerson) && !(addLine instanceof AdHocRouteWorkgroup)) { 455 MaintenanceDocumentForm maintenanceForm = (MaintenanceDocumentForm) model; 456 MaintenanceDocument document = maintenanceForm.getDocument(); 457 458 // get the old object's collection 459 //KULRICE-7970 support multiple level objects 460 String bindingPrefix = collectionGroup.getBindingInfo().getBindByNamePrefix(); 461 String propertyPath = collectionGroup.getPropertyName(); 462 if(bindingPrefix!=""&&bindingPrefix!= null) { 463 propertyPath = bindingPrefix + "." + propertyPath; 464 } 465 466 Collection<Object> oldCollection = ObjectPropertyUtils 467 .getPropertyValue(document.getOldMaintainableObject().getDataObject(), 468 propertyPath); 469 470 471 try { 472 Object blankLine = collectionGroup.getCollectionObjectClass().newInstance(); 473 //Add a blank line to the top of the collection 474 if(oldCollection instanceof List){ 475 ((List) oldCollection).add(0,blankLine); 476 } else { 477 oldCollection.add(blankLine); 478 } 479 } catch (Exception e) { 480 throw new RuntimeException("Unable to create new line instance for old maintenance object", e); 481 } 482 } 483 } 484 485 /** 486 * In the case of edit maintenance deleted the item on the old side 487 * 488 * 489 * @see org.kuali.rice.krad.uif.service.impl.ViewHelperServiceImpl#processAfterDeleteLine(View, 490 * org.kuali.rice.krad.uif.container.CollectionGroup, java.lang.Object, int) 491 */ 492 @Override 493 protected void processAfterDeleteLine(View view, CollectionGroup collectionGroup, Object model, int lineIndex) { 494 super.processAfterDeleteLine(view, collectionGroup, model, lineIndex); 495 496 // Check for maintenance documents in edit but exclude notes and ad hoc recipients 497 if (model instanceof MaintenanceDocumentForm 498 && KRADConstants.MAINTENANCE_EDIT_ACTION.equals(((MaintenanceDocumentForm)model).getMaintenanceAction()) 499 && !collectionGroup.getCollectionObjectClass().getName().equals(Note.class.getName()) 500 && !collectionGroup.getCollectionObjectClass().getName().equals(AdHocRoutePerson.class.getName()) 501 && !collectionGroup.getCollectionObjectClass().getName().equals(AdHocRouteWorkgroup.class.getName())) { 502 MaintenanceDocumentForm maintenanceForm = (MaintenanceDocumentForm) model; 503 MaintenanceDocument document = maintenanceForm.getDocument(); 504 505 // get the old object's collection 506 Collection<Object> oldCollection = ObjectPropertyUtils 507 .getPropertyValue(document.getOldMaintainableObject().getDataObject(), 508 collectionGroup.getPropertyName()); 509 try { 510 // Remove the object at lineIndex from the collection 511 oldCollection.remove(oldCollection.toArray()[lineIndex]); 512 } catch (Exception e) { 513 throw new RuntimeException("Unable to delete line instance for old maintenance object", e); 514 } 515 } 516 } 517 518 /** 519 * Retrieves the document number configured on this maintainable 520 * 521 * @return String document number 522 */ 523 protected String getDocumentNumber() { 524 return this.documentNumber; 525 } 526 527 protected LookupService getLookupService() { 528 if (lookupService == null) { 529 lookupService = KRADServiceLocatorWeb.getLookupService(); 530 } 531 return this.lookupService; 532 } 533 534 public void setLookupService(LookupService lookupService) { 535 this.lookupService = lookupService; 536 } 537 538 protected DataObjectAuthorizationService getDataObjectAuthorizationService() { 539 if (dataObjectAuthorizationService == null) { 540 this.dataObjectAuthorizationService = KRADServiceLocatorWeb.getDataObjectAuthorizationService(); 541 } 542 return dataObjectAuthorizationService; 543 } 544 545 public void setDataObjectAuthorizationService(DataObjectAuthorizationService dataObjectAuthorizationService) { 546 this.dataObjectAuthorizationService = dataObjectAuthorizationService; 547 } 548 549 protected DataObjectMetaDataService getDataObjectMetaDataService() { 550 if (dataObjectMetaDataService == null) { 551 this.dataObjectMetaDataService = KRADServiceLocatorWeb.getDataObjectMetaDataService(); 552 } 553 return dataObjectMetaDataService; 554 } 555 556 public void setDataObjectMetaDataService(DataObjectMetaDataService dataObjectMetaDataService) { 557 this.dataObjectMetaDataService = dataObjectMetaDataService; 558 } 559 560 public DocumentDictionaryService getDocumentDictionaryService() { 561 if (documentDictionaryService == null) { 562 this.documentDictionaryService = KRADServiceLocatorWeb.getDocumentDictionaryService(); 563 } 564 return documentDictionaryService; 565 } 566 567 public void setDocumentDictionaryService(DocumentDictionaryService documentDictionaryService) { 568 this.documentDictionaryService = documentDictionaryService; 569 } 570 571 protected EncryptionService getEncryptionService() { 572 if (encryptionService == null) { 573 encryptionService = CoreApiServiceLocator.getEncryptionService(); 574 } 575 return encryptionService; 576 } 577 578 public void setEncryptionService(EncryptionService encryptionService) { 579 this.encryptionService = encryptionService; 580 } 581 582 protected BusinessObjectService getBusinessObjectService() { 583 if (businessObjectService == null) { 584 businessObjectService = KRADServiceLocator.getBusinessObjectService(); 585 } 586 return businessObjectService; 587 } 588 589 public void setBusinessObjectService(BusinessObjectService businessObjectService) { 590 this.businessObjectService = businessObjectService; 591 } 592 593 protected MaintenanceDocumentService getMaintenanceDocumentService() { 594 if (maintenanceDocumentService == null) { 595 maintenanceDocumentService = KRADServiceLocatorWeb.getMaintenanceDocumentService(); 596 } 597 return maintenanceDocumentService; 598 } 599 600 public void setMaintenanceDocumentService(MaintenanceDocumentService maintenanceDocumentService) { 601 this.maintenanceDocumentService = maintenanceDocumentService; 602 } 603 }