1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.kpme.tklm.time.detail.validation;
17
18 import org.apache.commons.lang.StringUtils;
19 import org.joda.time.*;
20 import org.kuali.kpme.core.api.assignment.Assignment;
21 import org.kuali.kpme.core.api.assignment.AssignmentDescriptionKey;
22 import org.kuali.kpme.core.api.calendar.entry.CalendarEntry;
23 import org.kuali.kpme.core.api.earncode.EarnCode;
24 import org.kuali.kpme.core.api.job.JobContract;
25 import org.kuali.kpme.core.calendar.entry.CalendarEntryBo;
26 import org.kuali.kpme.core.service.HrServiceLocator;
27 import org.kuali.kpme.core.util.HrConstants;
28 import org.kuali.kpme.core.util.HrContext;
29 import org.kuali.kpme.core.util.TKUtils;
30 import org.kuali.kpme.core.util.ValidationUtils;
31 import org.kuali.kpme.tklm.api.common.TkConstants;
32 import org.kuali.kpme.tklm.api.leave.block.LeaveBlock;
33 import org.kuali.kpme.tklm.api.leave.summary.LeaveSummaryContract;
34 import org.kuali.kpme.tklm.api.time.clocklog.ClockLog;
35 import org.kuali.kpme.tklm.api.time.timeblock.TimeBlock;
36 import org.kuali.kpme.tklm.common.CalendarValidationUtil;
37 import org.kuali.kpme.tklm.leave.calendar.validation.LeaveCalendarValidationUtil;
38 import org.kuali.kpme.tklm.leave.service.LmServiceLocator;
39 import org.kuali.kpme.tklm.time.clocklog.ClockLogBo;
40 import org.kuali.kpme.tklm.time.detail.web.TimeDetailActionFormBase;
41 import org.kuali.kpme.tklm.time.service.TkServiceLocator;
42 import org.kuali.kpme.tklm.time.timesheet.TimesheetDocument;
43 import org.kuali.rice.krad.util.ObjectUtils;
44
45 import java.math.BigDecimal;
46 import java.util.ArrayList;
47 import java.util.List;
48
49 public class TimeDetailValidationUtil extends CalendarValidationUtil {
50
51 public static List<String> validateLeaveEntry(TimeDetailActionFormBase tdaf) throws Exception {
52
53
54 List<String> errorMsgList = new ArrayList<String>();
55 CalendarEntry payCalendarEntry = tdaf.getCalendarEntry();
56 if(ObjectUtils.isNotNull(payCalendarEntry)) {
57 LeaveBlock lb = null;
58 if(StringUtils.isNotEmpty(tdaf.getLmLeaveBlockId())) {
59 lb = LmServiceLocator.getLeaveBlockService().getLeaveBlock(tdaf.getLmLeaveBlockId());
60 }
61 errorMsgList.addAll(CalendarValidationUtil.validateEarnCode(tdaf.getSelectedEarnCode(),tdaf.getStartDate(),tdaf.getEndDate()));
62 if(errorMsgList.isEmpty()) {
63 LeaveSummaryContract ls = LmServiceLocator.getLeaveSummaryService().getLeaveSummaryAsOfDate(HrContext.getTargetPrincipalId(), TKUtils.formatDateString(tdaf.getEndDate()));
64
65 BigDecimal leaveAmount = tdaf.getLeaveAmount();
66 if(leaveAmount == null) {
67 Long startTime = TKUtils.convertDateStringToDateTimeWithoutZone(tdaf.getStartDate(), tdaf.getStartTime()).getMillis();
68 Long endTime = TKUtils.convertDateStringToDateTimeWithoutZone(tdaf.getEndDate(), tdaf.getEndTime()).getMillis();
69 leaveAmount = TKUtils.getHoursBetween(startTime, endTime);
70 }
71
72
73
74
75
76
77
78
79 errorMsgList.addAll(TimeDetailValidationUtil.validateLeaveParametersByEarnCodeRecordMethod(tdaf));
80 errorMsgList.addAll(LeaveCalendarValidationUtil.validateAvailableLeaveBalanceForUsage(tdaf.getSelectedEarnCode(),
81 tdaf.getStartDate(), tdaf.getEndDate(), leaveAmount, lb));
82
83 errorMsgList.addAll(LeaveCalendarValidationUtil.validateLeaveAccrualRuleMaxUsage(ls, tdaf.getSelectedEarnCode(),
84 tdaf.getStartDate(), tdaf.getEndDate(), leaveAmount, lb));
85 errorMsgList.addAll(LeaveCalendarValidationUtil.validateHoursUnderTwentyFour(tdaf.getSelectedEarnCode(),
86 tdaf.getStartDate(), tdaf.getEndDate(),leaveAmount));
87 }
88 }
89 return errorMsgList;
90 }
91
92 public static List<String> validateLeaveParametersByEarnCodeRecordMethod(TimeDetailActionFormBase lcf) {
93 List<String> errors = new ArrayList<String>();
94 if (StringUtils.isNotBlank(lcf.getSelectedEarnCode()) && lcf.getCalendarEntry() != null) {
95
96
97 CalendarEntry calendarEntry = lcf.getCalendarEntry();
98 EarnCode earnCode = HrServiceLocator.getEarnCodeService().getEarnCode(lcf.getSelectedEarnCode(), calendarEntry.getEndPeriodFullDateTime().toLocalDate());
99 if(earnCode != null) {
100 if(earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_TIME)) {
101 return LeaveCalendarValidationUtil.validateTimeParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getStartTime(), lcf.getEndTime(), lcf.getSelectedAssignment(), lcf.getLmLeaveBlockId(), null);
102 }
103
104
105
106
107 else if (earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_HOUR)) {
108 return validateHourParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getLeaveAmount());
109 }
110 else if (earnCode.getRecordMethod().equalsIgnoreCase(HrConstants.EARN_CODE_DAY)) {
111 return validateDayParametersForLeaveEntry(earnCode, lcf.getCalendarEntry(), lcf.getStartDate(), lcf.getEndDate(), lcf.getLeaveAmount());
112 }
113 }
114 }
115 return errors;
116 }
117
118
119
120
121
122
123
124
125 @Deprecated
126 public static List<String> validateEarnCode(String earnCode, String startDateString, String endDateString) {
127 List<String> errors = new ArrayList<String>();
128
129 LocalDate tempDate = TKUtils.formatDateTimeStringNoTimezone(startDateString).toLocalDate();
130 LocalDate localEnd = TKUtils.formatDateTimeStringNoTimezone(endDateString).toLocalDate();
131
132 while(!localEnd.isBefore(tempDate)) {
133 if(!ValidationUtils.validateEarnCode(earnCode, tempDate)) {
134 errors.add("Earn Code " + earnCode + " is not available for " + tempDate);
135 break;
136 }
137 tempDate = tempDate.plusDays(1);
138 }
139
140 return errors;
141 }
142
143
144
145
146
147
148
149 public static List<String> validateTimeEntryDetails(TimeDetailActionFormBase tdaf) {
150 boolean spanningWeeks = false;
151 boolean acrossDays = false;
152
153 if(tdaf.getAcrossDays() != null) {
154 acrossDays = tdaf.getAcrossDays().equalsIgnoreCase("y");
155 }
156
157 return validateTimeEntryDetails(
158 tdaf.getHours(), tdaf.getAmount(), tdaf.getStartTime(), tdaf.getEndTime(),
159 tdaf.getStartDate(), tdaf.getEndDate(), tdaf.getTimesheetDocument(),
160 tdaf.getSelectedEarnCode(), tdaf.getSelectedAssignment(),
161 acrossDays, tdaf.getTkTimeBlockId(), tdaf.getOvertimePref()
162 );
163 }
164
165 public static List<String> validateTimeEntryDetails(BigDecimal hours, BigDecimal amount, String startTimeS, String endTimeS, String startDateS, String endDateS, TimesheetDocument timesheetDocument, String selectedEarnCode, String selectedAssignment, boolean acrossDays, String timeblockId, String overtimePref) {
166 List<String> errors = new ArrayList<String>();
167 LocalDate savedStartDate = TKUtils.formatDateString(startDateS);
168 LocalDate savedEndDate = TKUtils.formatDateString(endDateS);
169
170 if (timesheetDocument == null) {
171 errors.add("No timesheet document found.");
172 }
173 if (errors.size() > 0) return errors;
174
175 CalendarEntry payCalEntry = timesheetDocument.getCalendarEntry();
176 EarnCode earnCode = null;
177 if (StringUtils.isNotBlank(selectedEarnCode)) {
178 earnCode = HrServiceLocator.getEarnCodeService().getEarnCode(selectedEarnCode, TKUtils.formatDateTimeStringNoTimezone(endDateS).toLocalDate());
179 }
180 boolean isTimeRecordMethod = earnCode != null && StringUtils.equalsIgnoreCase(earnCode.getRecordMethod(), HrConstants.EARN_CODE_TIME);
181
182 errors.addAll(CalendarValidationUtil.validateDates(startDateS, endDateS));
183
184 if (isTimeRecordMethod) {
185 errors.addAll(CalendarValidationUtil.validateTimes(startTimeS, endTimeS));
186 }
187 if (errors.size() > 0) return errors;
188
189 Long startTime;
190 Long endTime;
191
192 if (!isTimeRecordMethod) {
193 startTimeS = "0:0";
194 endTimeS = "0:0";
195 }
196
197 if (acrossDays && !endTimeS.equals("0:00")) { endDateS = startDateS;}
198
199 startTime = TKUtils.convertDateStringToDateTimeWithoutZone(startDateS, startTimeS).getMillis();
200 endTime = TKUtils.convertDateStringToDateTimeWithoutZone(endDateS, endTimeS).getMillis();
201
202 errors.addAll(CalendarValidationUtil.validateInterval(payCalEntry, startTime, endTime));
203 if (errors.size() > 0) return errors;
204
205 if (isTimeRecordMethod) {
206 if (startTimeS == null) errors.add("The start time is blank.");
207 if (endTimeS == null) errors.add("The end time is blank.");
208 if (startTime - endTime == 0) errors.add("Start time and end time cannot be equivalent");
209 }
210 if (errors.size() > 0) return errors;
211
212 DateTime startTemp = new DateTime(startTime);
213 DateTime endTemp = new DateTime(endTime);
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229 AssignmentDescriptionKey assignKey = HrServiceLocator.getAssignmentService().getAssignmentDescriptionKey(selectedAssignment);
230 Assignment assign = HrServiceLocator.getAssignmentService().getAssignmentForTargetPrincipal(assignKey, startTemp.toLocalDate());
231 if (assign == null) errors.add("Assignment is not valid for start date " + TKUtils.formatDate(new LocalDate(startTime)));
232 assign = HrServiceLocator.getAssignmentService().getAssignmentForTargetPrincipal(assignKey, endTemp.toLocalDate());
233 if (assign == null) errors.add("Assignment is not valid for end date " + TKUtils.formatDate(new LocalDate(endTime)));
234 if (errors.size() > 0) return errors;
235
236
237
238
239
240
241
242
243
244
245
246 if (acrossDays && hours == null && amount == null) {
247 if (savedEndDate.isAfter(savedStartDate)
248 && startTemp.getHourOfDay() > endTemp.getHourOfDay()
249 && !(endTemp.getDayOfYear() - startTemp.getDayOfYear() <= 1
250 && endTemp.getHourOfDay() == 0)) {
251 errors.add("The \"apply to each day\" box should not be checked.");
252 }
253 }
254 if (errors.size() > 0) return errors;
255
256
257
258
259 if ((startTime.compareTo(endTime) > 0 || endTime.compareTo(startTime) < 0)) {
260 errors.add("The time or date is not valid.");
261 }
262 if (errors.size() > 0) return errors;
263
264
265
266
267
268
269
270
271
272 if (amount != null && earnCode != null && StringUtils.equals(earnCode.getEarnCodeType(), HrConstants.EARN_CODE_AMOUNT)) {
273 if (amount.equals(BigDecimal.ZERO)) {
274 errors.add("Amount cannot be zero.");
275 }
276 if (amount.scale() > 2) {
277 errors.add("Amount cannot have more than two digits after decimal point.");
278 }
279 }
280 if (errors.size() > 0) return errors;
281
282
283
284
285
286 if (hours != null && earnCode != null && StringUtils.equals(earnCode.getEarnCodeType(), HrConstants.EARN_CODE_HOUR)) {
287 if (hours.equals(BigDecimal.ZERO)) {
288 errors.add("Hours cannot be zero.");
289 }
290 if (hours.scale() > 2) {
291 errors.add("Hours cannot have more than two digits after decimal point.");
292 }
293
294
295
296
297
298
299
300
301
302
303
304 }
305 if (errors.size() > 0) return errors;
306
307
308
309
310
311
312
313 if(hours != null && hours.compareTo(new BigDecimal(24.0)) > 0) {
314 errors.add("Hours cannot exceed 24.");
315 }
316
317
318
319
320
321 boolean isRegularEarnCode = StringUtils.equals(assign.getJob().getPayTypeObj().getRegEarnCode(),selectedEarnCode);
322 startTime = TKUtils.convertDateStringToDateTime(startDateS, startTimeS).getMillis();
323 endTime = TKUtils.convertDateStringToDateTime(endDateS, endTimeS).getMillis();
324
325 errors.addAll(validateOverlap(startTime, endTime, acrossDays, startDateS, endTimeS,startTemp, endTemp, timesheetDocument, timeblockId, isRegularEarnCode, selectedEarnCode));
326 if (errors.size() > 0) return errors;
327
328
329
330
331 return errors;
332 }
333
334 public static List<String> validateOverlap(Long startTime, Long endTime, boolean acrossDays, String startDateS, String endTimeS, DateTime startTemp, DateTime endTemp, TimesheetDocument timesheetDocument, String timeblockId, boolean isRegularEarnCode, String selectedEarnCode) {
335 List<String> errors = new ArrayList<String>();
336 Interval addedTimeblockInterval = new Interval(startTime, endTime);
337 List<Interval> dayInt = new ArrayList<Interval>();
338
339
340 ClockLog lastClockLog = TkServiceLocator.getClockLogService().getLastClockLog(HrContext.getTargetPrincipalId());
341 if(lastClockLog != null &&
342 (lastClockLog.getClockAction().equals(TkConstants.CLOCK_IN)
343 || lastClockLog.getClockAction().equals(TkConstants.LUNCH_IN))) {
344 DateTime lastClockDateTime = lastClockLog.getClockDateTime();
345
346
347
348
349
350
351
352 DateTime currentTime = DateTime.now();
353 if (lastClockDateTime.getMillis() > currentTime.getMillis()) {
354 currentTime = new DateTime(lastClockDateTime.getMillis());
355 }
356
357 Interval currentClockInInterval = new Interval(lastClockDateTime, currentTime);
358 if (isRegularEarnCode && addedTimeblockInterval.overlaps(currentClockInInterval)) {
359 errors.add("The time block you are trying to add overlaps with the current clock action.");
360 return errors;
361 }
362 }
363
364 if (acrossDays) {
365 DateTime start = new DateTime(startTime);
366
367
368
369
370
371
372
373
374
375
376
377
378 DateTime end = TKUtils.convertDateStringToDateTime(startDateS, endTimeS);
379
380 if (endTemp.getDayOfYear() - startTemp.getDayOfYear() < 1) {
381 end = new DateTime(endTime);
382 }
383 DateTime groupEnd = new DateTime(endTime);
384 Long startLong = start.getMillis();
385 Long endLong = end.getMillis();
386
387
388 DateMidnight midNight = new DateMidnight(endLong);
389 while (start.isBefore(groupEnd.getMillis()) && ((endLong >= startLong) || end.isEqual(midNight))) {
390 Interval tempInt = null;
391 if (end.isEqual(midNight)) {
392 tempInt = addedTimeblockInterval;
393 } else {
394 tempInt = new Interval(startLong, endLong);
395 }
396 dayInt.add(tempInt);
397 start = start.plusDays(1);
398 end = end.plusDays(1);
399 startLong = start.getMillis();
400 endLong = end.getMillis();
401 }
402 } else {
403 dayInt.add(addedTimeblockInterval);
404 }
405
406 for (TimeBlock timeBlock : timesheetDocument.getTimeBlocks()) {
407 if (errors.size() == 0 && StringUtils.equals(timeBlock.getEarnCodeType(), HrConstants.EARN_CODE_TIME)) {
408
409 JobContract aJob = HrServiceLocator.getJobService().getJob(timeBlock.getPrincipalId(), timeBlock.getJobNumber(), timeBlock.getBeginDateTime().toLocalDate());
410 if(aJob != null && aJob.getPayTypeObj() != null && isRegularEarnCode && !StringUtils.equals(aJob.getPayTypeObj().getRegEarnCode(),timeBlock.getEarnCode())) {
411 continue;
412 }
413
414 Interval timeBlockInterval = new Interval(timeBlock.getBeginDateTime(), timeBlock.getEndDateTime());
415 for (Interval intv : dayInt) {
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430 DateTime start_dt_timezone = new DateTime(startTime);
431 DateTime end_dt_timezone = new DateTime(endTime);
432 Interval converted_intv = new Interval(start_dt_timezone.getMillis(), end_dt_timezone.getMillis());
433 List<Interval> intervals = new ArrayList<Interval>();
434 if (acrossDays) {
435 List<LocalDate> localDates = new ArrayList<LocalDate>();
436 LocalDate startDay = new LocalDate(start_dt_timezone);
437 DateTimeZone userTimeZone = DateTimeZone.forID(HrServiceLocator.getTimezoneService().getUserTimezone(timesheetDocument.getPrincipalId()));
438 if (userTimeZone==null) {
439 userTimeZone = HrServiceLocator.getTimezoneService().getTargetUserTimezoneWithFallback();
440 }
441
442
443 int days = end_dt_timezone.withZone(userTimeZone).toLocalTime().equals(new LocalTime(0,0,0)) ? Days.daysBetween(startDay, new LocalDate(end_dt_timezone)).getDays() : Days.daysBetween(startDay, new LocalDate(end_dt_timezone)).getDays()+1;
444 for (int i=0; i < days; i++) {
445 LocalDate d = startDay.withFieldAdded(DurationFieldType.days(), i);
446 localDates.add(d);
447 }
448 for (LocalDate localDate : localDates) {
449 DateTime startDateTime = localDate.toDateTime(start_dt_timezone.toLocalTime());
450 DateTime endDateTime = localDate.toDateTime(end_dt_timezone.toLocalTime());
451 endDateTime = endDateTime.isBefore(startDateTime) ? endDateTime.plusDays(1) : endDateTime;
452
453 intervals.add(new Interval(startDateTime,endDateTime));
454 }
455
456 } else {
457 intervals.add(converted_intv);
458 }
459
460 for (Interval interval : intervals) {
461 if (isRegularEarnCode && timeBlockInterval.overlaps(interval) && (timeblockId == null || timeblockId.compareTo(timeBlock.getTkTimeBlockId()) != 0)) {
462 errors.add("The time block you are trying to add overlaps with an existing time block.");
463 return errors;
464 }else if(timeBlockInterval.overlaps(interval) && (timeblockId == null || timeblockId.compareTo(timeBlock.getTkTimeBlockId()) != 0)){
465 errors.add("The time block you are trying to add overlaps with an existing time block.");
466 return errors;
467 }
468 }
469 }
470 }
471 }
472
473 return errors;
474 }
475
476
477
478
479
480
481
482 @Deprecated
483 public static List<String> validateDates(String startDateS, String endDateS) {
484 List<String> errors = new ArrayList<String>();
485 if (errors.size() == 0 && StringUtils.isEmpty(startDateS)) errors.add("The start date is blank.");
486 if (errors.size() == 0 && StringUtils.isEmpty(endDateS)) errors.add("The end date is blank.");
487 return errors;
488 }
489
490
491
492
493
494
495
496 @Deprecated
497 public static List<String> validateTimes(String startTimeS, String endTimeS) {
498 List<String> errors = new ArrayList<String>();
499 if (errors.size() == 0 && startTimeS == null) errors.add("The start time is blank.");
500 if (errors.size() == 0 && endTimeS == null) errors.add("The end time is blank.");
501 return errors;
502 }
503
504
505
506
507
508
509
510
511 @Deprecated
512 public static List<String> validateInterval(CalendarEntryBo payCalEntry, Long startTime, Long endTime) {
513 List<String> errors = new ArrayList<String>();
514 LocalDateTime pcb_ldt = payCalEntry.getBeginPeriodLocalDateTime();
515 LocalDateTime pce_ldt = payCalEntry.getEndPeriodLocalDateTime();
516
517 DateTime p_cal_b_dt = pcb_ldt.toDateTime();
518 DateTime p_cal_e_dt = pce_ldt.toDateTime();
519
520 Interval payInterval = new Interval(p_cal_b_dt, p_cal_e_dt);
521 if (errors.size() == 0 && !payInterval.contains(startTime)) {
522 errors.add("The start date/time is outside the pay period");
523 }
524 if (errors.size() == 0 && !payInterval.contains(endTime) && p_cal_e_dt.getMillis() != endTime) {
525 errors.add("The end date/time is outside the pay period");
526 }
527 return errors;
528 }
529
530 }