001 /** 002 * Copyright 2004-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.hr.time.util; 017 018 import java.math.BigDecimal; 019 import java.sql.Date; 020 import java.util.HashMap; 021 import java.util.List; 022 import java.util.Map; 023 024 import org.apache.commons.lang.StringUtils; 025 import org.kuali.hr.earncodesec.EarnCodeSecurity; 026 import org.kuali.hr.location.Location; 027 import org.kuali.hr.paygrade.PayGrade; 028 import org.kuali.hr.time.accrual.AccrualCategory; 029 import org.kuali.hr.time.accrual.TimeOffAccrual; 030 import org.kuali.hr.time.authorization.DepartmentalRule; 031 import org.kuali.hr.time.calendar.Calendar; 032 import org.kuali.hr.time.department.Department; 033 import org.kuali.hr.time.earncode.EarnCode; 034 import org.kuali.hr.time.earngroup.EarnGroup; 035 import org.kuali.hr.time.earngroup.EarnGroupDefinition; 036 import org.kuali.hr.time.paytype.PayType; 037 import org.kuali.hr.time.salgroup.SalGroup; 038 import org.kuali.hr.time.service.base.TkServiceLocator; 039 import org.kuali.hr.time.task.Task; 040 import org.kuali.hr.time.workarea.WorkArea; 041 import org.kuali.kfs.coa.businessobject.Chart; 042 import org.kuali.rice.kim.api.identity.Person; 043 import org.kuali.rice.kim.api.services.KimApiServiceLocator; 044 import org.kuali.rice.krad.service.KRADServiceLocator; 045 046 /** 047 * A few methods to assist with various validation tasks. 048 */ 049 public class ValidationUtils { 050 051 /** 052 * For DepartmentalRule objects, if a work area is defined, you can not 053 * leave the department field with a wildcard. Permission for wildcarding 054 * will be checked with other methods. 055 * 056 * @param dr The DepartmentalRule to examine. 057 * @return true if valid, false otherwise. 058 */ 059 public static boolean validateWorkAreaDeptWildcarding(DepartmentalRule dr) { 060 boolean ret = true; 061 062 if (StringUtils.equals(dr.getDept(), TkConstants.WILDCARD_CHARACTER)) { 063 ret = dr.getWorkArea().equals(TkConstants.WILDCARD_LONG); 064 } 065 066 return ret; 067 } 068 069 /** 070 * Most basic validation: Only checks for presence in the database. 071 */ 072 public static boolean validateWorkArea(Long workArea) { 073 return validateWorkArea(workArea, null); 074 } 075 076 /** 077 * Most basic validation: Only checks for presence in the database. 078 */ 079 public static boolean validateDepartment(String department) { 080 return validateDepartment(department, null); 081 } 082 083 /** 084 * Most basic validation: Only checks for presence in the database. 085 */ 086 public static boolean validateAccrualCategory(String accrualCategory) { 087 return validateAccrualCategory(accrualCategory, null); 088 } 089 090 091 public static boolean validateSalGroup(String salGroup, Date asOfDate) { 092 boolean valid = false; 093 094 if (StringUtils.equals(salGroup, TkConstants.WILDCARD_CHARACTER)) { 095 valid = true; 096 } else if (asOfDate != null) { 097 SalGroup sg = TkServiceLocator.getSalGroupService().getSalGroup(salGroup, asOfDate); 098 valid = (sg != null); 099 } else { 100 int count = TkServiceLocator.getSalGroupService().getSalGroupCount(salGroup); 101 valid = (count > 0); 102 } 103 104 return valid; 105 } 106 107 public static boolean validateEarnCode(String earnCode, Date asOfDate) { 108 boolean valid = false; 109 110 if (asOfDate != null) { 111 EarnCode ec = TkServiceLocator.getEarnCodeService().getEarnCode(earnCode, asOfDate); 112 valid = (ec != null); 113 } else { 114 int count = TkServiceLocator.getEarnCodeService().getEarnCodeCount(earnCode); 115 valid = (count > 0); 116 } 117 118 return valid; 119 } 120 121 public static boolean validateEarnCodeOfAccrualCategory(String earnCode, String accrualCategory, String principalId, Date asOfDate) { 122 boolean valid = false; 123 124 return valid; 125 } 126 127 public static boolean validateAccCategory(String accrualCategory, Date asOfDate) { 128 boolean valid = false; 129 130 if (asOfDate != null) { 131 AccrualCategory ac = TkServiceLocator.getAccrualCategoryService().getAccrualCategory(accrualCategory, asOfDate); 132 valid = (ac != null); 133 } else { 134 Map<String, String> fieldValues = new HashMap<String, String>(); 135 fieldValues.put("accrualCategory", accrualCategory); 136 int matches = KRADServiceLocator.getBusinessObjectService().countMatching(AccrualCategory.class, fieldValues); 137 138 valid = matches > 0; 139 } 140 141 return valid; 142 } 143 144 public static boolean validateLocation(String location, Date asOfDate) { 145 boolean valid = false; 146 147 if (asOfDate != null) { 148 Location l = TkServiceLocator.getLocationService().getLocation(location, asOfDate); 149 valid = (l != null); 150 } else { 151 int count = TkServiceLocator.getLocationService().getLocationCount(location); 152 valid = (count > 0); 153 } 154 155 return valid; 156 } 157 158 public static boolean validatePayType(String payType, Date asOfDate) { 159 boolean valid = false; 160 161 if (asOfDate != null) { 162 PayType pt = TkServiceLocator.getPayTypeService().getPayType(payType, asOfDate); 163 valid = (pt != null); 164 } else { 165 int count = TkServiceLocator.getPayTypeService().getPayTypeCount(payType); 166 valid = (count > 0); 167 } 168 169 return valid; 170 } 171 172 173 public static boolean validatePayGrade(String payGrade, Date asOfDate) { 174 boolean valid = false; 175 176 if (asOfDate != null) { 177 PayGrade pg = TkServiceLocator.getPayGradeService().getPayGrade(payGrade, asOfDate); 178 valid = (pg != null); 179 } else { 180 int count = TkServiceLocator.getPayGradeService().getPayGradeCount(payGrade); 181 valid = (count > 0); 182 } 183 184 return valid; 185 } 186 187 /** 188 * 189 * @param earnCode 190 * @param otEarnCode If true, earn code is valid ONLY if it is an overtime earn code. 191 * @param asOfDate 192 * @return 193 */ 194 public static boolean validateEarnCode(String earnCode, boolean otEarnCode, Date asOfDate) { 195 boolean valid = false; 196 197 if (asOfDate != null) { 198 EarnCode ec = TkServiceLocator.getEarnCodeService().getEarnCode(earnCode, asOfDate); 199 valid = (ec != null) && (otEarnCode ? ec.getOvtEarnCode().booleanValue() : true); 200 } 201 202 return valid; 203 } 204 205 /** 206 * Checks for row presence of a department, and optionally whether or not 207 * it is active as of the specified date. 208 */ 209 public static boolean validateDepartment(String department, Date asOfDate) { 210 boolean valid = false; 211 212 if (StringUtils.isEmpty(department)) { 213 // do nothing, let false be returned. 214 } else if (asOfDate != null) { 215 Department d = TkServiceLocator.getDepartmentService().getDepartment(department, asOfDate); 216 valid = (d != null); 217 } else { 218 int count = TkServiceLocator.getDepartmentService().getDepartmentCount(department); 219 valid = (count > 0); 220 } 221 222 return valid; 223 } 224 225 public static boolean validateChart(String chart) { 226 boolean valid = false; 227 228 if (!StringUtils.isEmpty(chart)) { 229 Object o = KRADServiceLocator.getBusinessObjectService().findBySinglePrimaryKey(Chart.class, chart); 230 valid = (o instanceof Chart); 231 } 232 233 return valid; 234 } 235 236 /** 237 * Checks for row presence of a work area, and optionally whether or not 238 * it is active as of the specified date. 239 */ 240 public static boolean validateWorkArea(Long workArea, Date asOfDate) { 241 return ValidationUtils.validateWorkArea(workArea, null, asOfDate); 242 } 243 244 public static boolean validateWorkArea(Long workArea, String dept, Date asOfDate) { 245 boolean valid = false; 246 247 if (workArea == null) { 248 valid = false; 249 } else if (workArea.equals(TkConstants.WILDCARD_LONG)) { 250 valid = true; 251 } else if (asOfDate != null) { 252 WorkArea wa = TkServiceLocator.getWorkAreaService().getWorkArea(workArea, asOfDate); 253 if (wa != null && dept != null) { 254 valid = StringUtils.equalsIgnoreCase(dept, wa.getDept()); 255 } else { 256 valid = (wa != null); 257 } 258 } else { 259 // Not valid if no date is passed. 260 } 261 262 return valid; 263 } 264 /** 265 * Checks for row presence of a Accrual Category, and optionally whether or not 266 * it is active as of the specified date. 267 */ 268 public static boolean validateAccrualCategory(String accrualCategory, Date asOfDate) { 269 boolean valid = false; 270 271 if (StringUtils.equals(accrualCategory, TkConstants.WILDCARD_CHARACTER)) { 272 valid = true; 273 } else if (asOfDate != null) { 274 AccrualCategory ac = TkServiceLocator.getAccrualCategoryService().getAccrualCategory(accrualCategory, asOfDate); 275 valid = (ac != null); 276 } 277 278 return valid; 279 } 280 281 /** 282 * Checks for row presence of a principal Id, and optionally whether or not 283 * it is active as of the specified date. 284 */ 285 public static boolean validatePrincipalId(String principalId) { 286 boolean valid = false; 287 if (principalId != null) { 288 Person p = KimApiServiceLocator.getPersonService().getPerson(principalId); 289 valid = (p != null); 290 } 291 return valid; 292 } 293 294 /** 295 * No wildcarding is accounted for in this method. 296 * @param task Task "Long Name" 297 * @param asOfDate Can be null, if we just want to look for the general case. 298 * @return True if the task is present / valid. 299 */ 300 public static boolean validateTask(Long task, Date asOfDate) { 301 boolean valid = false; 302 303 if (task != null && asOfDate != null) { 304 Task t = TkServiceLocator.getTaskService().getTask(task, asOfDate); 305 valid = (t != null); 306 } else if (task != null) { 307 int count = TkServiceLocator.getTaskService().getTaskCount(task); 308 valid = (count > 0); 309 } 310 311 return valid; 312 } 313 314 /** 315 * No wildcarding is accounted for in this method. 316 * @param earnGroup EarnCodeGroup 317 * @param asOfDate Can be null, if we just want to look for the general case. 318 * @return True if the EarnCodeGroup is present / valid. 319 */ 320 public static boolean validateEarnGroup(String earnGroup, Date asOfDate) { 321 boolean valid = false; 322 323 if (earnGroup != null && asOfDate != null) { 324 EarnGroup eg = TkServiceLocator.getEarnGroupService().getEarnGroup(earnGroup, asOfDate); 325 valid = (eg != null); 326 } else if (earnGroup != null) { 327 int count = TkServiceLocator.getEarnGroupService().getEarnGroupCount(earnGroup); 328 valid = (count > 0); 329 } 330 331 return valid; 332 } 333 334 /** 335 * @param earnGroup EarnCodeGroup 336 * @param asOfDate 337 * @return True if the EarnCodeGroup has overtime earn codes 338 */ 339 public static boolean earnGroupHasOvertimeEarnCodes(String earnGroup, Date asOfDate) { 340 if (earnGroup != null && asOfDate != null) { 341 EarnGroup eg = TkServiceLocator.getEarnGroupService().getEarnGroup(earnGroup, asOfDate); 342 if(eg != null) { 343 for(EarnGroupDefinition egd : eg.getEarnGroups()) { 344 if(egd.getEarnCode() != null) { 345 EarnCode ec = TkServiceLocator.getEarnCodeService().getEarnCode(egd.getEarnCode(), asOfDate); 346 if(ec != null && ec.getOvtEarnCode()) { 347 return true; 348 } 349 } 350 } 351 } 352 } 353 354 return false; 355 } 356 357 358 /** 359 * Checks for row presence of a pay calendar 360 */ 361 public static boolean validateCalendar(String calendarName) { 362 Map<String, String> fieldValues = new HashMap<String, String>(); 363 fieldValues.put("calendarName", calendarName); 364 int matches = KRADServiceLocator.getBusinessObjectService().countMatching(Calendar.class, fieldValues); 365 366 return matches > 0; 367 } 368 369 public static boolean duplicateDeptEarnCodeExists(EarnCodeSecurity deptEarnCode) { 370 boolean valid = false; 371 int count = TkServiceLocator.getEarnCodeSecurityService().getEarnCodeSecurityCount 372 (deptEarnCode.getDept(), deptEarnCode.getHrSalGroup(), deptEarnCode.getEarnCode(), deptEarnCode.isEmployee() ? "1" : "0", 373 deptEarnCode.isApprover() ? "1" : "0", deptEarnCode.getLocation(), deptEarnCode.getActive() ? "Y" : "N", deptEarnCode.getEffectiveDate(), null); 374 if(count == 1) { 375 valid = true; 376 count = TkServiceLocator.getEarnCodeSecurityService().getEarnCodeSecurityCount 377 (deptEarnCode.getDept(), deptEarnCode.getHrSalGroup(), deptEarnCode.getEarnCode(), deptEarnCode.isEmployee() ? "1" : "0", 378 deptEarnCode.isApprover() ? "1" : "0", deptEarnCode.getLocation(), deptEarnCode.getActive() ? "Y" : "N", deptEarnCode.getEffectiveDate(), deptEarnCode.getHrEarnCodeSecurityId()); 379 if(count == 1) { 380 valid = false; 381 } 382 } else if(count > 1) { 383 valid = true; 384 } 385 386 return valid; 387 } 388 389 public static boolean duplicateTimeOffAccrual (TimeOffAccrual timeOffAccrual) { 390 boolean valid = false; 391 int count = TkServiceLocator.getTimeOffAccrualService().getTimeOffAccrualCount 392 (timeOffAccrual.getAccrualCategory(), timeOffAccrual.getEffectiveDate(), timeOffAccrual.getPrincipalId(), null); 393 if(count == 1) { 394 valid = true; 395 count = TkServiceLocator.getTimeOffAccrualService().getTimeOffAccrualCount 396 (timeOffAccrual.getAccrualCategory(), timeOffAccrual.getEffectiveDate(), timeOffAccrual.getPrincipalId(), timeOffAccrual.getLmAccrualId()); 397 if(count == 1) { 398 valid = false; 399 } 400 } else if(count > 1) { 401 valid = true; 402 } 403 return valid; 404 } 405 406 /** 407 * Checks for date not more than one year in the future or current date 408 * 409 */ 410 411 public static boolean validateOneYearFutureDate(Date date){ 412 java.util.Calendar startDate = java.util.Calendar.getInstance(); 413 startDate.add(java.util.Calendar.DATE, -1); 414 startDate.set(java.util.Calendar.SECOND, 0); 415 startDate.set(java.util.Calendar.MINUTE, 0); 416 startDate.set(java.util.Calendar.HOUR_OF_DAY, 0); 417 java.util.Calendar endDate = java.util.Calendar.getInstance(); 418 endDate.add(java.util.Calendar.YEAR, 1); // One year after the current date 419 return date.compareTo(startDate.getTime()) * date.compareTo(endDate.getTime()) <= 0; 420 } 421 422 /** 423 * Checks for date not more than one year in the future and does not consider past date 424 * 425 */ 426 427 public static boolean validateOneYearFutureEffectiveDate(Date date){ 428 java.util.Calendar startDate = java.util.Calendar.getInstance(); 429 startDate.set(java.util.Calendar.MILLISECOND, 0); 430 startDate.set(java.util.Calendar.SECOND, 0); 431 startDate.set(java.util.Calendar.MINUTE, 0); 432 startDate.set(java.util.Calendar.HOUR_OF_DAY, 0); 433 startDate.add(java.util.Calendar.YEAR, 1); // One year after the current date 434 return date.compareTo(startDate.getTime()) <= 0; 435 } 436 437 /** 438 * Checks for date in the future 439 * 440 */ 441 442 public static boolean validateFutureDate(Date date){ 443 java.util.Calendar startDate = java.util.Calendar.getInstance(); 444 startDate.add(java.util.Calendar.DATE, 0); 445 startDate.set(java.util.Calendar.SECOND, 0); 446 startDate.set(java.util.Calendar.MINUTE, 0); 447 startDate.set(java.util.Calendar.HOUR_OF_DAY, 0); 448 return date.compareTo(startDate.getTime()) > 0; 449 } 450 451 /** 452 * Checks for row presence of a pay calendar by calendar type 453 */ 454 public static boolean validateCalendarByType(String calendarName, String calendarType) { 455 Map<String, String> fieldValues = new HashMap<String, String>(); 456 fieldValues.put("calendarName", calendarName); 457 fieldValues.put("calendarTypes", calendarType); 458 int matches = KRADServiceLocator.getBusinessObjectService().countMatching(Calendar.class, fieldValues); 459 460 return matches > 0; 461 } 462 463 public static boolean validateRecordMethod(String recordMethod, String accrualCategory, Date asOfDate) { 464 boolean valid = false; 465 if (asOfDate != null) { 466 AccrualCategory ac = TkServiceLocator.getAccrualCategoryService().getAccrualCategory(accrualCategory, asOfDate); 467 if (ac != null 468 && ac.getUnitOfTime() != null) { 469 if (TkConstants.RECORD_METHOD.HOUR.equals(ac.getUnitOfTime()) 470 && (TkConstants.RECORD_METHOD.HOUR.equals(recordMethod)) 471 || TkConstants.RECORD_METHOD.TIME.equals(recordMethod)) { 472 valid = true; 473 } else { 474 valid = StringUtils.equalsIgnoreCase(ac.getUnitOfTime(), recordMethod); 475 } 476 477 } 478 } 479 return valid; 480 } 481 482 }