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.service; 017 018 import org.kuali.rice.krad.bo.BusinessObject; 019 import org.kuali.rice.krad.datadictionary.DataDictionaryEntry; 020 import org.kuali.rice.krad.datadictionary.ReferenceDefinition; 021 import org.kuali.rice.krad.datadictionary.state.StateMapping; 022 import org.kuali.rice.krad.datadictionary.validation.AttributeValueReader; 023 import org.kuali.rice.krad.datadictionary.validation.result.DictionaryValidationResult; 024 import org.kuali.rice.krad.document.Document; 025 import org.kuali.rice.krad.document.TransactionalDocument; 026 027 import java.beans.PropertyDescriptor; 028 029 /** 030 * Defines the API for the validating against the data dictionary. 031 * 032 * @author Kuali Rice Team (rice.collab@kuali.org) 033 */ 034 public interface DictionaryValidationService { 035 036 /** 037 * Validates the contents of a document (i.e. attributes within a document) against the data dictionary. 038 * 039 * @param document - document to validate 040 */ 041 public void validateDocument(Document document); 042 043 /** 044 * Validates the contents of a document and recursively validates any of its updatable references 045 * 046 * @param document the document 047 * @param maxDepth the maximum numbers of levels to recurse 048 * @param validateRequired whether to validate whether a field is required and is currently blank 049 */ 050 public void validateDocumentAndUpdatableReferencesRecursively(Document document, int maxDepth, 051 boolean validateRequired); 052 053 /** 054 * Validates the contents of a document and recursively validates any of its updatable references 055 * 056 * @param document the document 057 * @param maxDepth the maximum numbers of levels to recurse 058 * @param validateRequired whether to validate whether a field is required and is currently blank 059 * @param chompLastLetterSFromCollectionName if true, the error path for any collections encountered will have the 060 * last "s" removed from the collection name if it ends 061 * with the letter "s". If false, this method acts like {@link #validateDocumentAndUpdatableReferencesRecursively(Document, 062 * int, boolean)} 063 */ 064 public void validateDocumentAndUpdatableReferencesRecursively(Document document, int maxDepth, 065 boolean validateRequired, boolean chompLastLetterSFromCollectionName); 066 067 /** 068 * Validates the specified attribute of the given document against the data dictionary. 069 * 070 * @param document 071 * @param attributeName 072 * @param errorPrefix 073 */ 074 public void validateDocumentAttribute(Document document, String attributeName, String errorPrefix); 075 076 /** 077 * Validates an object using its class name as the entry name to look up its metadata in the dictionary. 078 * 079 * @param object - an object to validate 080 * @return the dictionary validation result object associated with this validation 081 */ 082 public DictionaryValidationResult validate(Object object); 083 084 /** 085 * Validate an object with the passed in dictionary entryName and the specific attribute to be evaluated 086 * 087 * @param object - an object to validate 088 * @param entryName - the dictionary entry name to look up the metadata associated with this object 089 * @param attributeName - the name of the attribute (field) on the object that should be validated 090 * @param doOptionalProcessing true if the validation should do optional validation (e.g. to check if empty values 091 * are required or not), false otherwise 092 * @return the dictionary validation result object associated with this validation 093 * @since 1.1 094 */ 095 public DictionaryValidationResult validate(Object object, String entryName, String attributeName, 096 boolean doOptionalProcessing); 097 098 /** 099 * Same as {@link DictionaryValidationService#validate(Object, String, String, boolean) except that it provides an 100 * explicit 101 * data dictionary 102 * entry to use for the purpose of validation. 103 * 104 * @param object - an object to validate 105 * @param entryName - the dictionary entry name to use in association with error look ups 106 * @param entry - the dictionary entry to use for validation 107 * @param doOptionalProcessing true if the validation should do optional validation (e.g. to check if empty values 108 * are required or not), false otherwise 109 * @return the dictionary validation result object associated with this validation 110 * @since 1.1 111 */ 112 public DictionaryValidationResult validate(Object object, String entryName, DataDictionaryEntry entry, 113 boolean doOptionalProcessing); 114 115 /** 116 * Validates the object agains the next state (or current state if there is no next state). 117 * 118 * <p>When no stateMapping exists on the DataDictionaryEntry that applies for this object, validation is considered 119 * stateless and all constraints are processed regardless of their states attribute.</p> 120 * 121 * @param object 122 * @return the dictionary validation result object associated with this validation 123 * @since 2.2 124 */ 125 public DictionaryValidationResult validateAgainstNextState(Object object); 126 127 /** 128 * Validates the object against the state specified. 129 * 130 * <p>Important note: Alternatively the state can be changed on the 131 * object itself and another validation method can be used instead of this one (in practice, you'd revert the 132 * state on the object if validation returns errors).</p> 133 * 134 * @param object 135 * @param validationState 136 * @return the dictionary validation result object associated with this validation 137 * @since 2.2 138 */ 139 public DictionaryValidationResult validateAgainstState(Object object, String validationState); 140 141 /** 142 * Same as other validate methods, except allows you to provide the attributeValueReader directly for evaluation 143 * 144 * @param valueReader - an object to validate 145 * @param doOptionalProcessing true if the validation should do optional validation (e.g. to check if empty values 146 * are required or not), false otherwise 147 * @return the dictionary validation result object associated with this validation 148 * @since 1.1 149 */ 150 public DictionaryValidationResult validate(AttributeValueReader valueReader, boolean doOptionalProcessing, 151 String validationState, StateMapping stateMapping); 152 153 /** 154 * Encapsulates <code>{@link #validateBusinessObject(BusinessObject) and returns boolean so one doesn't need to 155 * check the 156 * ErrorMap.Validates the business object primitive attributes against the data dictionary. Adds errors to the map 157 * as they are 158 * encountered.<br/> 159 * <br/> 160 * Makes no error path adjustments 161 * 162 * @param businessObject - business object to validate 163 * @return boolean validOrNot 164 */ 165 public boolean isBusinessObjectValid(BusinessObject businessObject); 166 167 /** 168 * Encapsulates <code>{@link #validateBusinessObject(BusinessObject) and returns boolean so one doesn't need to 169 * check the 170 * ErrorMap.Validates the business object primitive attributes against the data dictionary. Adds errors to the map 171 * as they are 172 * encountered.<br/> 173 * <br/> 174 * Makes no error path adjustments 175 * 176 * @param businessObject - business object to validate 177 * @param prefix - error prefix 178 * @return boolean valid or not 179 */ 180 public boolean isBusinessObjectValid(BusinessObject businessObject, String prefix); 181 182 /** 183 * Validates the business object primitive attributes against the data dictionary. Adds errors to the map as they 184 * are 185 * encountered. 186 * 187 * @param businessObject - business object to validate 188 * @deprecated since 1.1 - use validate(Object.class) instead 189 */ 190 @Deprecated 191 public void validateBusinessObject(BusinessObject businessObject); 192 193 /** 194 * Validates the business object primitive attributes against the data dictionary. Adds errors to the map as they 195 * are 196 * encountered. 197 * 198 * @param businessObject - business object to validate 199 * @param validateRequired - whether to execute required field checks 200 * @deprecated since 1.1 - use validate(Object.class) instead 201 */ 202 @Deprecated 203 public void validateBusinessObject(BusinessObject businessObject, boolean validateRequired); 204 205 /** 206 * This method examines the populated BusinessObject bo instance passed in for a member named by the referenceName. 207 * If this 208 * member exists, and if this member is a descendent of BusinessObject, then an existence check proceeds. 209 * 210 * First the foreign keys for this reference are gathered, and then examined to see if they have values. If they do 211 * not have 212 * values, the method ends with a true return value. If they all have values, then an object with those primary 213 * keys 214 * is retrieve 215 * from the database. If one is retrieve, then the reference exists, and True is returned. Otherwise, false is 216 * returned. 217 * 218 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 219 * errorMap with no 220 * prefix, other than what has already been pushed onto the errorMap. 221 * 222 * @param bo - The bo whose reference is being tested. 223 * @param reference - The ReferenceDefinition to be existence tested. 224 * @return True if no exceptions occur and the object exists in the db, false otherwise. 225 */ 226 public boolean validateReferenceExists(BusinessObject bo, ReferenceDefinition reference); 227 228 /** 229 * This method examines the populated BusinessObject bo instance passed in for a member named by the referenceName. 230 * If this 231 * member exists, and if this member is a descendent of BusinessObject, then an existence check proceeds. 232 * 233 * First the foreign keys for this reference are gathered, and then examined to see if they have values. If they do 234 * not have 235 * values, the method ends with a true return value. If they all have values, then an object with those primary 236 * keys 237 * is retrieve 238 * from the database. If one is retrieve, then the reference exists, and True is returned. Otherwise, false is 239 * returned. 240 * 241 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 242 * errorMap with no 243 * prefix, other than what has already been pushed onto the errorMap. 244 * 245 * @param bo - The bo whose reference is being tested. 246 * @param referenceName - The name of the member to be existence tested. 247 * @return True if no exceptions occur and the object exists in the db, false otherwise. 248 */ 249 public boolean validateReferenceExists(BusinessObject bo, String referenceName); 250 251 /** 252 * This method retrieves the reference from the DB, and then tests whether the object is active. 253 * 254 * It will return false if there is no activeIndicator field on this object, if the object doesnt exist in the DB, 255 * if the field 256 * doesnt exist or cannot be cast as a boolean, if the field value is null, or if the field value is false. 257 * 258 * It will only return true if the reference bo is present, the field is present, it is a boolean and non-null, and 259 * the value is 260 * true. 261 * 262 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 263 * errorMap with no 264 * prefix, other than what has already been pushed onto the errorMap. 265 * 266 * @param bo 267 * @param reference 268 * @return 269 */ 270 public boolean validateReferenceIsActive(BusinessObject bo, ReferenceDefinition reference); 271 272 /** 273 * This method retrieves the reference from the DB, and then tests whether the object is active. 274 * 275 * It will return false if there is no activeIndicator field on this object, if the object doesnt exist in the DB, 276 * if the field 277 * doesnt exist or cannot be cast as a boolean, if the field value is null, or if the field value is false. 278 * 279 * It will only return true if the reference bo is present, the field is present, it is a boolean and non-null, and 280 * the value is 281 * true. 282 * 283 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 284 * errorMap with no 285 * prefix, other than what has already been pushed onto the errorMap. 286 * 287 * @param bo 288 * @param referenceName 289 * @return 290 */ 291 public boolean validateReferenceIsActive(BusinessObject bo, String referenceName); 292 293 /** 294 * validateReferenceExistsAndIsActive intelligently tests the designated reference on the bo for both existence and 295 * active status, where 296 * appropriate 297 * 298 * <p>It will not test anything if the foreign-key fields for the given reference aren't filled out with values, 299 * and 300 * it 301 * will not test active status if the reference doesn't exist.</p> 302 * 303 * <p>Further, it will only test active status where the correct flag is set.</p> 304 * 305 * <p>On failures of either sort, it will put the relevant errors into the GlobalVariables errorMap, and return a 306 * false. If there 307 * are no failures, or nothing can be tested because the foreign-key fields arent fully filled out, it will return 308 * true and add 309 * no errors.</p> 310 * 311 * <p>This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 312 * errorMap with no 313 * prefix, other than what has already been pushed onto the errorMap.</p> 314 * 315 * @param bo - the BusinessObject instance to be tested. 316 * @param reference - the ReferenceDefinition to control the nature of the testing. 317 * @return true or false as per the criteria above 318 */ 319 public boolean validateReferenceExistsAndIsActive(BusinessObject bo, ReferenceDefinition reference); 320 321 /** 322 * This method intelligently tests the designated reference on the bo for both existence and active status, where 323 * appropriate. 324 * 325 * It will not test anything if the foreign-key fields for the given reference arent filled out with values, and it 326 * will not 327 * test active status if the reference doesnt exist. 328 * 329 * Note that it will not fail or raise any error if all of the foreign-keys are filled with a value. If this needs 330 * to be tested 331 * (ie, the 'if any field is filled, then all must be filled' rule), you'll have to do that separately. 332 * 333 * Further, it will only test active status where the correct flag is set. 334 * 335 * On failures of either sort, it will put the relevant errors into the GlobalVariables errorMap, and return a 336 * false. If there 337 * are no failures, or nothing can be tested because the foreign-key fields arent fully filled out, it will return 338 * true and add 339 * no errors. 340 * 341 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 342 * errorMap with no 343 * prefix, other than what has already been pushed onto the errorMap. 344 * 345 * @param bo - the BusinessObject instance to be tested. 346 * @param referenceName - the member name on the bo to be tested for existence and active-state 347 * @param attributeToHighlightOnFail - the fieldName to highlight with the error message on a failure 348 * @param displayFieldName - the human-readable display name of the failed field, to go in the error message 349 * @return true or false as per the criteria above 350 */ 351 public boolean validateReferenceExistsAndIsActive(BusinessObject bo, String referenceName, 352 String attributeToHighlightOnFail, String displayFieldName); 353 354 /** 355 * This method does an existence check against all references of a BusinessObject as defined in the 356 * MaintenanceDocument.xml file 357 * for that business object. 358 * 359 * Appropriate errors will also be placed in the GlobalVariables.ErrorMap. 360 * 361 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 362 * errorMap with no 363 * prefix, other than what has already been pushed onto the errorMap. 364 * 365 * @param bo - BusinessObject instance that should be tested 366 * @return true if all passed existence tests, false if any failed 367 */ 368 public boolean validateDefaultExistenceChecks(BusinessObject bo); 369 370 /** 371 * Does an existence check against all references configured as a default existence check in the maintenance 372 * document data dictionary file for the given business object 373 * 374 * Appropriate errors will also be placed in the GlobalVariables.ErrorMap. 375 * 376 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 377 * errorMap 378 * with no 379 * prefix, other than what has already been pushed onto the errorMap. 380 * 381 * @param bo parent business object instance to retrieve default checks for 382 * @param newCollectionItem new collection line to validate 383 * @param collectionName name of the collection in the parent 384 * @return true if all passed existence tests, false if any failed 385 */ 386 public boolean validateDefaultExistenceChecksForNewCollectionItem(BusinessObject bo, 387 BusinessObject newCollectionItem, String collectionName); 388 389 /** 390 * This method does an existence check against all references of a transactionalDocument 391 * 392 * Appropriate errors will also be placed in the GlobalVariables.ErrorMap. 393 * 394 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 395 * errorMap 396 * with no 397 * prefix, other than what has already been pushed onto the errorMap. 398 * 399 * @param document document instance that should be tested 400 * @return true if all passed existence tests, false if any failed 401 */ 402 public boolean validateDefaultExistenceChecksForTransDoc(TransactionalDocument document); 403 404 /** 405 * This method does an existence check against all references of a transactionalDocument 406 * 407 * Appropriate errors will also be placed in the GlobalVariables.ErrorMap. 408 * 409 * This method assumes that you already have the errorPath set exactly as desired, and adds new errors to the 410 * errorMap 411 * with no 412 * prefix, other than what has already been pushed onto the errorMap. 413 * 414 * @param document document instance that should be tested 415 * @param accountingLine that should be tested 416 * @param collectionName that should be tested 417 * @return true if all passed existence tests, false if any failed 418 */ 419 public boolean validateDefaultExistenceChecksForNewCollectionItem(TransactionalDocument document, 420 BusinessObject accountingLine, String collectionName); 421 422 /** 423 * @deprecated since 1.1 424 */ 425 @Deprecated 426 public void validatePrimitiveFromDescriptor(String entryName, Object object, PropertyDescriptor propertyDescriptor, 427 String errorPrefix, boolean validateRequired); 428 }