1 /** 2 * Copyright 2005-2014 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.krad.bo; 17 18 import org.apache.commons.lang.StringUtils; 19 import org.apache.ojb.broker.PersistenceBroker; 20 import org.apache.ojb.broker.PersistenceBrokerAware; 21 import org.apache.ojb.broker.PersistenceBrokerException; 22 import org.kuali.rice.core.api.mo.common.GloballyUnique; 23 import org.kuali.rice.core.api.mo.common.Versioned; 24 import org.kuali.rice.krad.service.KRADServiceLocator; 25 import org.kuali.rice.krad.service.PersistenceService; 26 import org.kuali.rice.krad.service.PersistenceStructureService; 27 28 import javax.persistence.Column; 29 import javax.persistence.MappedSuperclass; 30 import javax.persistence.PostLoad; 31 import javax.persistence.PostPersist; 32 import javax.persistence.PostRemove; 33 import javax.persistence.PostUpdate; 34 import javax.persistence.PrePersist; 35 import javax.persistence.PreRemove; 36 import javax.persistence.PreUpdate; 37 import javax.persistence.Transient; 38 import javax.persistence.Version; 39 import java.util.ArrayList; 40 import java.util.Collection; 41 import java.util.List; 42 import java.util.UUID; 43 44 /** 45 * @author Kuali Rice Team (rice.collab@kuali.org) 46 */ 47 @MappedSuperclass 48 public abstract class PersistableBusinessObjectBase extends BusinessObjectBase implements PersistableBusinessObject, PersistenceBrokerAware, Versioned, GloballyUnique { 49 private static final long serialVersionUID = 1451642350593233282L; 50 private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(PersistableBusinessObjectBase.class); 51 52 @Version 53 @Column(name="VER_NBR") 54 protected Long versionNumber; 55 @Column(name="OBJ_ID") 56 private String objectId; 57 @Transient 58 private boolean newCollectionRecord; 59 @Transient 60 protected PersistableBusinessObjectExtension extension; 61 62 private static transient PersistenceService persistenceService; 63 private static transient PersistenceStructureService persistenceStructureService; 64 65 /** 66 * @see PersistableBusinessObject#getVersionNumber() 67 */ 68 public Long getVersionNumber() { 69 return versionNumber; 70 } 71 72 /** 73 * @see PersistableBusinessObject#getVersionNumber() 74 */ 75 public void setVersionNumber(Long versionNumber) { 76 this.versionNumber = versionNumber; 77 } 78 79 80 /** 81 * getter for the guid based object id that is assignable to all objects, in order to support custom attributes a mapping must 82 * also be added to the OJB file and a column must be added to the database for each business object that extension attributes 83 * are supposed to work on. 84 * 85 * @return 86 */ 87 public String getObjectId() { 88 return objectId; 89 } 90 91 /** 92 * setter for the guid based object id that is assignable to all objects, in order to support custom attributes a mapping must 93 * also be added to the OJB file and column must be added to the database for each business object that extension attributes are 94 * supposed to work on. 95 * 96 * @param objectId 97 */ 98 public void setObjectId(String objectId) { 99 this.objectId = objectId; 100 } 101 102 103 /** 104 * Gets the newCollectionRecord attribute. 105 * 106 * @return Returns the newCollectionRecord. 107 */ 108 public boolean isNewCollectionRecord() { 109 return newCollectionRecord; 110 } 111 112 /** 113 * Sets the newCollectionRecord attribute value. 114 * 115 * @param isNewCollectionRecord The newCollectionRecord to set. 116 */ 117 public void setNewCollectionRecord(boolean isNewCollectionRecord) { 118 this.newCollectionRecord = isNewCollectionRecord; 119 } 120 121 /** 122 * Implementation of the OJB afterDelete hook which delegates to {@link #postRemove()}. This method is final 123 * because it is recommended that sub-classes override and implement postRemove if they need to take 124 * advantage of this persistence hook. 125 * 126 * @see org.apache.ojb.broker.PersistenceBrokerAware#afterDelete(org.apache.ojb.broker.PersistenceBroker) 127 */ 128 public final void afterDelete(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 129 postRemove(); 130 } 131 132 /** 133 * Default implementation of the JPA {@link PostRemove} hook. This implementation currently does nothing, 134 * however sub-classes can override and implement this method if needed. 135 * 136 * <p>This method is currently invoked by the corresponding OJB {@link #afterDelete(PersistenceBroker)} hook. 137 */ 138 @PostRemove 139 protected void postRemove() { 140 // do nothing 141 } 142 143 /** 144 * Implementation of the OJB afterInsert hook which delegates to {@link #postPersist()}. This method is final 145 * because it is recommended that sub-classes override and implement postPersist if they need to take 146 * advantage of this persistence hook. 147 * 148 * @see org.apache.ojb.broker.PersistenceBrokerAware#afterInsert(org.apache.ojb.broker.PersistenceBroker) 149 */ 150 public final void afterInsert(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 151 postPersist(); 152 } 153 154 /** 155 * Default implementation of the JPA {@link PostPersist} hook. This implementation currently does nothing, 156 * however sub-classes can override and implement this method if needed. 157 * 158 * <p>This method is currently invoked by the corresponding OJB {@link #afterInsert(PersistenceBroker)} hook. 159 */ 160 @PostPersist 161 protected void postPersist() { 162 // do nothing 163 } 164 165 /** 166 * Implementation of the OJB afterLookup hook which delegates to {@link #postLoad()}. This method is final 167 * because it is recommended that sub-classes override and implement postLoad if they need to take 168 * advantage of this persistence hook. 169 * 170 * @see org.apache.ojb.broker.PersistenceBrokerAware#afterLookup(org.apache.ojb.broker.PersistenceBroker) 171 */ 172 public final void afterLookup(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 173 postLoad(); 174 } 175 176 /** 177 * Default implementation of the JPA {@link PostLoad} hook. This implementation currently does nothing, 178 * however sub-classes can override and implement this method if needed. 179 * 180 * <p>This method is currently invoked by the corresponding OJB {@link #afterLookup(PersistenceBroker)} hook. 181 */ 182 @PostLoad 183 protected void postLoad() { 184 // do nothing 185 } 186 187 /** 188 * Implementation of the OJB afterUpdate hook which delegates to {@link #postUpdate()}. This method is final 189 * because it is recommended that sub-classes override and implement postUpdate if they need to take 190 * advantage of this persistence hook. 191 * 192 * @see org.apache.ojb.broker.PersistenceBrokerAware#afterUpdate(org.apache.ojb.broker.PersistenceBroker) 193 */ 194 public final void afterUpdate(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 195 postUpdate(); 196 } 197 198 /** 199 * Default implementation of the JPA {@link PostUpdate} hook. This implementation currently does nothing, 200 * however sub-classes can override and implement this method if needed. 201 * 202 * <p>This method is currently invoked by the corresponding OJB {@link #afterUpdate(PersistenceBroker)} hook. 203 */ 204 @PostUpdate 205 protected void postUpdate() { 206 // do nothing 207 } 208 209 /** 210 * Implementation of the OJB beforeDelete hook which delegates to {@link #preRemove()}. This method is final 211 * because it is recommended that sub-classes override and implement preRemove if they need to take 212 * advantage of this persistence hook. 213 * 214 * @see org.apache.ojb.broker.PersistenceBrokerAware#beforeDelete(org.apache.ojb.broker.PersistenceBroker) 215 */ 216 public final void beforeDelete(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 217 preRemove(); 218 } 219 220 /** 221 * Default implementation of the JPA {@link PreRemove} hook. This implementation currently does nothing, 222 * however sub-classes can implement this method if needed. 223 * 224 * <p>This method is currently invoked by the corresponding OJB {@link #beforeDelete(PersistenceBroker)} hook. 225 */ 226 @PreRemove 227 protected void preRemove() { 228 // do nothing 229 } 230 231 /** 232 * Implementation of the OJB beforeInsert hook which delegates to {@link #prePersist()}. This method is final 233 * because it is recommended that sub-classes override and implement prePersist if they need to take 234 * advantage of this persistence hook. 235 * 236 * @see org.apache.ojb.broker.PersistenceBrokerAware#beforeInsert(org.apache.ojb.broker.PersistenceBroker) 237 */ 238 public final void beforeInsert(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 239 //setObjectId(UUID.randomUUID().toString()); 240 setObjectId(null); 241 prePersist(); 242 } 243 244 /** 245 * Default implementation of the JPA {@link PrePersist} hook which generates the unique objectId for this 246 * persistable business object if it does not already have one. Any sub-class which overrides this method 247 * should take care to invoke super.prePersist to ensure that the objectId for this persistable 248 * business object is generated properly. 249 * 250 * <p>This method is currently invoked by the corresponding OJB {@link #beforeInsert(PersistenceBroker)} hook. 251 */ 252 @PrePersist 253 protected void prePersist() { 254 generateAndSetObjectIdIfNeeded(); 255 } 256 257 /** 258 * Implementation of the OJB beforeUpdate hook which delegates to {@link #preUpdate()}. This method is final 259 * because it is recommended that sub-classes override and implement preUpdate if they need to take 260 * advantage of this persistence hook. 261 * 262 * @see org.apache.ojb.broker.PersistenceBrokerAware#beforeUpdate(org.apache.ojb.broker.PersistenceBroker) 263 */ 264 public final void beforeUpdate(PersistenceBroker persistenceBroker) throws PersistenceBrokerException { 265 preUpdate(); 266 } 267 268 /** 269 * Default implementation of the JPA {@link PreUpdate} hook which generates the unique objectId for this 270 * persistable business object if it does not already have one. Any sub-class which overrides this method 271 * should take care to invoke super.preUpdate to ensure that the objectId for this persistable 272 * business object is generated properly. 273 * 274 * <p>This method is currently invoked by the corresponding OJB {@link #beforeUpdate(PersistenceBroker)} hook. 275 */ 276 @PreUpdate 277 protected void preUpdate() { 278 generateAndSetObjectIdIfNeeded(); 279 } 280 281 /** 282 * If this PersistableBusinessObject does not already have a unique objectId, this method will generate 283 * one and set it's value on this object. 284 */ 285 private void generateAndSetObjectIdIfNeeded() { 286 if (StringUtils.isEmpty(getObjectId())) { 287 setObjectId(UUID.randomUUID().toString()); 288 } 289 } 290 291 /** 292 * getService Refreshes the reference objects from the primitive values. 293 * 294 * @see org.kuali.rice.krad.bo.BusinessObject#refresh() 295 */ 296 public void refresh() { 297 getPersistenceService().retrieveNonKeyFields(this); 298 } 299 300 /** 301 * @see BusinessObject#refresh() 302 */ 303 public void refreshNonUpdateableReferences() { 304 getPersistenceService().refreshAllNonUpdatingReferences(this); 305 } 306 307 public void refreshReferenceObject(String referenceObjectName) { 308 if ( StringUtils.isNotBlank(referenceObjectName) && !StringUtils.equals(referenceObjectName, "extension")) { 309 final PersistenceStructureService pss = getPersistenceStructureService(); 310 if ( pss.hasReference(this.getClass(), referenceObjectName) || pss.hasCollection(this.getClass(), referenceObjectName)) { 311 getPersistenceService().retrieveReferenceObject( this, referenceObjectName); 312 } else { 313 LOG.warn( "refreshReferenceObject() called with non-reference property: " + referenceObjectName ); 314 } 315 } 316 } 317 318 /** 319 * @see PersistableBusinessObject#buildListOfDeletionAwareLists() 320 */ 321 public List<Collection<PersistableBusinessObject>> buildListOfDeletionAwareLists() { 322 return new ArrayList<Collection<PersistableBusinessObject>>(); 323 } 324 325 public void linkEditableUserFields() { 326 // do nothing 327 } 328 329 public PersistableBusinessObjectExtension getExtension() { 330 if ( extension == null 331 && getPersistenceStructureService().isPersistable(this.getClass())) { 332 try { 333 Class<? extends PersistableBusinessObjectExtension> extensionClass = getPersistenceStructureService().getBusinessObjectAttributeClass( getClass(), "extension" ); 334 if ( extensionClass != null ) { 335 extension = extensionClass.newInstance(); 336 } 337 } catch ( Exception ex ) { 338 LOG.error( "unable to create extension object", ex ); 339 } 340 } 341 return extension; 342 } 343 344 public void setExtension(PersistableBusinessObjectExtension extension) { 345 this.extension = extension; 346 } 347 348 /** 349 * @return the persistenceService 350 */ 351 protected static PersistenceService getPersistenceService() { 352 if ( persistenceService == null ) { 353 persistenceService = KRADServiceLocator.getPersistenceService(); 354 } 355 return persistenceService; 356 } 357 358 protected static PersistenceStructureService getPersistenceStructureService() { 359 if ( persistenceStructureService == null ) { 360 persistenceStructureService = KRADServiceLocator.getPersistenceStructureService(); 361 } 362 return persistenceStructureService; 363 } 364 365 }