1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package org.kuali.kpme.core.util;
17
18 import java.math.BigDecimal;
19 import java.net.UnknownHostException;
20 import java.sql.Timestamp;
21 import java.text.ParseException;
22 import java.text.SimpleDateFormat;
23 import java.util.ArrayList;
24 import java.util.Date;
25 import java.util.LinkedHashMap;
26 import java.util.List;
27 import java.util.Map;
28 import java.util.Set;
29 import java.util.TimeZone;
30
31 import org.apache.commons.lang.ObjectUtils;
32 import org.apache.commons.lang.StringUtils;
33 import org.apache.log4j.Logger;
34 import org.joda.time.DateTime;
35 import org.joda.time.DateTimeConstants;
36 import org.joda.time.DateTimeFieldType;
37 import org.joda.time.DateTimeZone;
38 import org.joda.time.Interval;
39 import org.joda.time.LocalDate;
40 import org.joda.time.Period;
41 import org.kuali.kpme.core.KPMEConstants;
42 import org.kuali.kpme.core.assignment.Assignment;
43 import org.kuali.kpme.core.calendar.entry.CalendarEntry;
44 import org.kuali.kpme.core.job.Job;
45 import org.kuali.kpme.core.service.HrServiceLocator;
46 import org.kuali.kpme.core.task.Task;
47 import org.kuali.kpme.core.workarea.WorkArea;
48 import org.kuali.rice.core.api.config.property.ConfigContext;
49 import org.kuali.rice.core.api.util.RiceKeyConstants;
50 import org.kuali.rice.core.api.util.type.KualiDecimal;
51 import org.kuali.rice.kim.api.identity.principal.EntityNamePrincipalName;
52 import org.kuali.rice.kim.api.services.KimApiServiceLocator;
53 import org.kuali.rice.krad.util.GlobalVariables;
54 import org.kuali.rice.krad.util.KRADConstants;
55
56 import javax.servlet.http.HttpServletRequest;
57
58 public class TKUtils {
59
60 private static final Logger LOG = Logger.getLogger(TKUtils.class);
61
62
63
64
65
66 public static String getDayOfMonthFromDateString(String dateString) {
67 String[] date = dateString.split("/");
68 return date[1];
69 }
70
71 public static String getSystemTimeZone() {
72 String configTimezone = TimeZone.getDefault().getID();
73 if (ConfigContext.getCurrentContextConfig() != null
74 && StringUtils.isNotBlank(ConfigContext.getCurrentContextConfig().getProperty(KPMEConstants.ConfigSettings.KPME_SYSTEM_TIMEZONE).trim())) {
75 String tempTimeZoneId = ConfigContext.getCurrentContextConfig().getProperty(KPMEConstants.ConfigSettings.KPME_SYSTEM_TIMEZONE);
76
77 if (TimeZone.getTimeZone(tempTimeZoneId) != null) {
78 configTimezone = ConfigContext.getCurrentContextConfig().getProperty(KPMEConstants.ConfigSettings.KPME_SYSTEM_TIMEZONE);
79 } else {
80 LOG.error("Timezone set by configuration parameter " + KPMEConstants.ConfigSettings.KPME_SYSTEM_TIMEZONE + " is not a valid time zone id. Using the systems default time zone instead.");
81 }
82 }
83
84
85 return configTimezone;
86 }
87
88 public static DateTimeZone getSystemDateTimeZone() {
89 return DateTimeZone.forID(TKUtils.getSystemTimeZone());
90 }
91
92 public static final LocalDate END_OF_TIME = new LocalDate(9999, 12, 31);
93
94 public static long getDaysBetween(LocalDate startDate, LocalDate endDate) {
95 long daysBetween = 0;
96
97 LocalDate currentDate = startDate;
98 while (currentDate.isBefore(endDate)) {
99 daysBetween++;
100 currentDate = currentDate.plusDays(1);
101 }
102
103 return daysBetween;
104 }
105
106 public static BigDecimal getHoursBetween(long start, long end) {
107 long diff = end - start;
108 BigDecimal hrsReminder = new BigDecimal((diff / 3600000.0) % 24);
109
110 if (hrsReminder.compareTo(BigDecimal.ZERO) == 0 && diff > 0) {
111 return new BigDecimal(diff / 3600000.0).setScale(HrConstants.BIG_DECIMAL_SCALE, HrConstants.BIG_DECIMAL_SCALE_ROUNDING).abs();
112 }
113 return hrsReminder.setScale(HrConstants.BIG_DECIMAL_SCALE, HrConstants.BIG_DECIMAL_SCALE_ROUNDING).abs();
114 }
115
116 public static String formatAssignmentKey(Long jobNumber, Long workArea, Long task) {
117 String assignmentKey = StringUtils.EMPTY;
118
119 String jobNumberString = ObjectUtils.toString(jobNumber, "0");
120 String workAreaString = ObjectUtils.toString(workArea, "0");
121 String taskString = ObjectUtils.toString(task, "0");
122
123 if (!jobNumberString.equals("0") || !workAreaString.equals("0") || !taskString.equals("0")) {
124 assignmentKey = StringUtils.join(new String[] {jobNumberString, workAreaString, taskString}, HrConstants.ASSIGNMENT_KEY_DELIMITER);
125 }
126
127 return assignmentKey;
128 }
129
130 public static Map<String, String> formatAssignmentDescription(Assignment assignment) {
131 Map<String, String> assignmentDescriptions = new LinkedHashMap<String, String>();
132 String assignmentDescKey = formatAssignmentKey(assignment.getJobNumber(), assignment.getWorkArea(), assignment.getTask());
133 String assignmentDescValue = HrServiceLocator.getAssignmentService().getAssignmentDescription(assignment.getPrincipalId(), assignment.getJobNumber(), assignment.getWorkArea(), assignment.getTask(), assignment.getEffectiveLocalDate());
134 assignmentDescriptions.put(assignmentDescKey, assignmentDescValue);
135
136 return assignmentDescriptions;
137 }
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169 public static List<Interval> getDaySpanForCalendarEntry(CalendarEntry calendarEntry, DateTimeZone timeZone) {
170 DateTime beginDateTime = calendarEntry.getBeginPeriodLocalDateTime().toDateTime(timeZone);
171 DateTime endDateTime = calendarEntry.getEndPeriodLocalDateTime().toDateTime(timeZone);
172
173 List<Interval> dayIntervals = new ArrayList<Interval>();
174
175 DateTime currDateTime = beginDateTime;
176 while (currDateTime.isBefore(endDateTime)) {
177 DateTime prevDateTime = currDateTime;
178 currDateTime = currDateTime.plusDays(1);
179 Interval daySpan = new Interval(prevDateTime, currDateTime);
180 dayIntervals.add(daySpan);
181 }
182
183 return dayIntervals;
184 }
185
186 public static long convertHoursToMillis(BigDecimal hours) {
187 return hours.multiply(HrConstants.BIG_DECIMAL_MS_IN_H, HrConstants.MATH_CONTEXT).longValue();
188 }
189
190 public static BigDecimal convertMillisToHours(long millis) {
191 return (new BigDecimal(millis)).divide(HrConstants.BIG_DECIMAL_MS_IN_H, HrConstants.MATH_CONTEXT);
192 }
193
194 public static BigDecimal convertMillisToMinutes(long millis) {
195 return (new BigDecimal(millis)).divide(HrConstants.BIG_DECIMAL_MS_IN_M, HrConstants.MATH_CONTEXT);
196 }
197
198 public static BigDecimal convertMillisToDays(long millis) {
199 BigDecimal hrs = convertMillisToHours(millis);
200 return hrs.divide(HrConstants.BIG_DECIMAL_HRS_IN_DAY, HrConstants.MATH_CONTEXT);
201 }
202
203 public static BigDecimal convertMinutesToHours(BigDecimal minutes) {
204 return minutes.divide(HrConstants.BIG_DECIMAL_60, HrConstants.MATH_CONTEXT);
205 }
206
207 public static int convertMillisToWholeDays(long millis) {
208 BigDecimal days = convertMillisToDays(millis);
209 return Integer.parseInt(days.setScale(0, BigDecimal.ROUND_UP).toString());
210 }
211
212
213
214
215
216 public static boolean isVirtualWorkDay(DateTime beginPeriodDateTime) {
217 return (beginPeriodDateTime.getHourOfDay() != 0 || beginPeriodDateTime.getMinuteOfHour() != 0
218 && beginPeriodDateTime.get(DateTimeFieldType.halfdayOfDay()) != DateTimeConstants.AM);
219 }
220
221
222
223
224
225
226
227
228
229
230 public static DateTime convertDateStringToDateTime(String dateStr, String timeStr) {
231
232 String[] date = dateStr.split("/");
233 String[] time = timeStr.split(":");
234
235 DateTimeZone dtz = DateTimeZone.forID(HrServiceLocator.getTimezoneService().getTargetUserTimezone());
236
237
238
239
240
241 DateTime dateTime = new DateTime(
242 Integer.parseInt(date[2]),
243 Integer.parseInt(date[0]),
244 Integer.parseInt(date[1]),
245 Integer.parseInt(time[0]),
246 Integer.parseInt(time[1]),
247 0, 0, dtz);
248
249 return dateTime;
250 }
251
252 public static DateTime convertDateStringToDateTimeWithTimeZone(String dateStr, String timeStr, DateTimeZone dtz) {
253
254 String[] date = dateStr.split("/");
255 String[] time = timeStr.split(":");
256
257
258
259
260
261 DateTime dateTime = new DateTime(
262 Integer.parseInt(date[2]),
263 Integer.parseInt(date[0]),
264 Integer.parseInt(date[1]),
265 Integer.parseInt(time[0]),
266 Integer.parseInt(time[1]),
267 0, 0, dtz);
268
269 return dateTime;
270 }
271
272 public static DateTime convertDateStringToDateTimeWithoutZone(String dateStr, String timeStr) {
273
274 String[] date = dateStr.split("/");
275 String[] time = timeStr.split(":");
276
277
278
279
280
281 DateTime dateTime = new DateTime(
282 Integer.parseInt(date[2]),
283 Integer.parseInt(date[0]),
284 Integer.parseInt(date[1]),
285 Integer.parseInt(time[0]),
286 Integer.parseInt(time[1]),
287 0, 0);
288
289 return dateTime;
290 }
291
292 public static String getIPAddressFromRequest(String remoteAddress) {
293
294
295 if (remoteAddress.indexOf(':') > -1) {
296 LOG.warn("ignoring IPv6 address for clock-in: " + remoteAddress);
297 remoteAddress = "";
298 }
299
300 return remoteAddress;
301 }
302
303 public static String getIPNumber() {
304 try {
305 return java.net.InetAddress.getLocalHost().getHostAddress();
306 } catch (UnknownHostException e) {
307 return "unknown";
308 }
309 }
310
311 public static String getIPAddressFromRequest(HttpServletRequest request) {
312
313
314 String fwdIp = request.getHeader("X-Forwarded-For");
315 if (fwdIp != null) {
316 LOG.info("Forwarded IP: " + fwdIp);
317 return fwdIp;
318 }
319
320 String ip = request.getRemoteAddr();
321 if (ip.indexOf(':') > -1) {
322 LOG.warn("ignoring IPv6 address for clock-in: " + ip);
323 ip = "";
324 }
325
326 return ip;
327 }
328
329
330 public static Timestamp subtractOneSecondFromTimestamp(Timestamp originalTimestamp) {
331 DateTime dt = new DateTime(originalTimestamp);
332 dt = dt.minusSeconds(1);
333 return new Timestamp(dt.getMillis());
334 }
335
336 public static String formatDate(LocalDate localDate) {
337 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
338 return sdf.format(localDate.toDate());
339 }
340
341 public static String formatDateTimeShort(DateTime dateTime) {
342 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
343 return sdf.format(dateTime.toDate());
344 }
345
346 public static String formatDateTimeLong(DateTime dateTime){
347 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy HH:mm:ss");
348 return sdf.format(dateTime.toDate());
349 }
350
351 public static LocalDate formatDateString(String date){
352 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
353 try {
354 return LocalDate.fromDateFields(sdf.parse(date));
355 } catch (ParseException e) {
356 return null;
357 }
358 }
359
360 public static String formatTimeShort(String dateString) {
361 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy HH:mm:ss");
362 Date tempDate = null;
363 try {
364 tempDate = sdf.parse(dateString);
365 } catch (ParseException e) {
366 e.printStackTrace();
367 }
368 return new SimpleDateFormat("HH:mm").format(tempDate);
369 }
370
371 public static DateTime formatDateTimeString(String dateTime) {
372 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
373 DateTimeZone dtz = DateTimeZone.forID(HrServiceLocator.getTimezoneService().getUserTimezone());
374 try {
375 return new DateTime(sdf.parse(dateTime)).withZone(dtz);
376 } catch (ParseException e) {
377 return null;
378 }
379 }
380
381 public static DateTime formatDateTimeStringNoTimezone(String dateTime) {
382 SimpleDateFormat sdf = new SimpleDateFormat("MM/dd/yyyy");
383 try {
384 return new DateTime(sdf.parse(dateTime));
385 } catch (ParseException e) {
386 return null;
387 }
388 }
389
390
391
392
393
394
395
396
397
398
399
400
401 public static String getTimezoneOffset(DateTime date) {
402 StringBuilder o = new StringBuilder();
403
404 int offsetms = date.getZone().getOffset(date);
405 boolean negative = offsetms < 0;
406 if (negative) offsetms = offsetms * -1;
407
408 Period period = new Period(offsetms);
409 if (negative) o.append('-');
410 o.append(StringUtils.leftPad(period.getHours() + "", 2, '0'));
411 o.append(StringUtils.leftPad(period.getMinutes() + "", 2, '0'));
412
413 return o.toString();
414 }
415
416 public static String arrayToString(String[] stringArray) {
417 StringBuilder str = new StringBuilder();
418 for (String string : stringArray) {
419 str.append(string);
420 }
421 return str.toString();
422 }
423
424
425
426
427 public static int getSessionTimeoutTime() {
428 return StringUtils.isBlank(ConfigContext.getCurrentContextConfig().getProperty(KPMEConstants.ConfigSettings.SESSION_TIMEOUT))
429 ? 2700 :
430 Integer.parseInt(ConfigContext.getCurrentContextConfig().getProperty(KPMEConstants.ConfigSettings.SESSION_TIMEOUT));
431 }
432
433 public static Timestamp getCurrentTimestamp() {
434 return new Timestamp(System.currentTimeMillis());
435 }
436
437 public static List<Interval> createDaySpan(DateTime beginDateTime, DateTime endDateTime, DateTimeZone zone) {
438 beginDateTime = beginDateTime.toDateTime(zone);
439 endDateTime = endDateTime.toDateTime(zone);
440 List<Interval> dayIntervals = new ArrayList<Interval>();
441
442 DateTime currDateTime = beginDateTime;
443 while (currDateTime.isBefore(endDateTime)) {
444 DateTime prevDateTime = currDateTime;
445 currDateTime = currDateTime.plusDays(1);
446 Interval daySpan = new Interval(prevDateTime, currDateTime);
447 dayIntervals.add(daySpan);
448 }
449
450 return dayIntervals;
451 }
452
453 public static List<Interval> getDaySpanForCalendarEntry(CalendarEntry calendarEntry) {
454 return getDaySpanForCalendarEntry(calendarEntry, HrServiceLocator.getTimezoneService().getTargetUserTimezoneWithFallback());
455 }
456
457 public static List<Interval> getFullWeekDaySpanForCalendarEntry(CalendarEntry calendarEntry) {
458 return getFullWeekDaySpanForCalendarEntry(calendarEntry, HrServiceLocator.getTimezoneService().getTargetUserTimezoneWithFallback());
459 }
460
461 public static List<Interval> getFullWeekDaySpanForCalendarEntry(CalendarEntry calendarEntry, DateTimeZone timeZone) {
462 DateTime beginDateTime = calendarEntry.getBeginPeriodLocalDateTime().toDateTime(timeZone);
463 DateTime endDateTime = calendarEntry.getEndPeriodLocalDateTime().toDateTime(timeZone);
464
465 List<Interval> dayIntervals = new ArrayList<Interval>();
466
467 DateTime currDateTime = beginDateTime;
468 if (beginDateTime.getDayOfWeek() != 7) {
469 currDateTime = beginDateTime.plusDays(0 - beginDateTime.getDayOfWeek());
470 }
471
472 int afterEndDate = 6 - endDateTime.getDayOfWeek();
473 if (endDateTime.getDayOfWeek() == 7 && endDateTime.getHourOfDay() != 0) {
474 afterEndDate = 6;
475 }
476 if (endDateTime.getHourOfDay() == 0) {
477 afterEndDate += 1;
478 }
479 DateTime aDate = endDateTime.plusDays(afterEndDate);
480 while (currDateTime.isBefore(aDate)) {
481 DateTime prevDateTime = currDateTime;
482 currDateTime = currDateTime.plusDays(1);
483 Interval daySpan = new Interval(prevDateTime, currDateTime);
484 dayIntervals.add(daySpan);
485 }
486
487 return dayIntervals;
488 }
489
490 public static int getWorkDays(DateTime startDate, DateTime endDate) {
491 int workDays = 0;
492
493 DateTime currentDate = startDate;
494 while (!currentDate.isAfter(endDate)) {
495 if (!isWeekend(currentDate)) {
496 workDays++;
497 }
498 currentDate = currentDate.plusDays(1);
499 }
500
501 return workDays;
502 }
503
504 public static boolean isWeekend(DateTime date) {
505 return date.getDayOfWeek() == DateTimeConstants.SATURDAY || date.getDayOfWeek() == DateTimeConstants.SUNDAY;
506 }
507
508
509
510
511
512
513
514
515
516 public static String getFromDateString(String dateString) {
517 String fromDateString = StringUtils.EMPTY;
518
519 if (StringUtils.startsWith(dateString, ">=")) {
520 fromDateString = StringUtils.substringAfter(dateString, ">=");
521 } else if (StringUtils.contains(dateString, "..")) {
522 fromDateString = StringUtils.substringBefore(dateString, "..");
523 }
524
525 return fromDateString;
526 }
527
528
529
530
531
532
533
534
535
536 public static String getToDateString(String dateString) {
537 String toDateString = StringUtils.EMPTY;
538
539 if (StringUtils.startsWith(dateString, "<=")) {
540 toDateString = StringUtils.substringAfter(dateString, "<=");
541 } else if (StringUtils.contains(dateString, "..")) {
542 toDateString = StringUtils.substringAfter(dateString, "..");
543 }
544
545 return toDateString;
546 }
547
548
549 public static boolean isDateEqualOrBetween(DateTime date, String searchDateString) {
550 boolean valid = false;
551
552 String fromDateString = TKUtils.getFromDateString(searchDateString);
553 DateTime fromDate = TKUtils.formatDateTimeString(fromDateString);
554 String toDateString = TKUtils.getToDateString(searchDateString);
555 DateTime toDate = TKUtils.formatDateTimeString(toDateString);
556
557 if (date != null) {
558 if (fromDate != null ? (date.equals(fromDate) || date.isAfter(fromDate)) : true
559 && toDate != null ? (date.isBefore(toDate) || date.equals(toDate)) : true) {
560 valid = true;
561 }
562 }
563
564 return valid;
565 }
566
567 public static String getRandomColor(Set<String> randomColors) {
568 String[] letters = "0,1,2,3,4,5,6,7,8,9,A,B,C,D,E,F".split(",");
569 String color = "#";
570 for (int i = 0; i < 6; i++) {
571 int index = (int) Math.round(Math.random() * 15);
572 color += letters[index];
573 }
574 if (randomColors.contains(color)) {
575 color = getRandomColor(randomColors);
576 }
577 return color;
578 }
579
580
581
582
583 public static BigDecimal cleanNumeric( String value ) {
584 String cleanedValue = value.replaceAll( "[^-0-9.]", "" );
585
586 if ( cleanedValue.lastIndexOf( '-' ) > 0 ) {
587 if ( cleanedValue.charAt( 0 ) == '-' ) {
588 cleanedValue = "-" + cleanedValue.replaceAll( "-", "" );
589 } else {
590 cleanedValue = cleanedValue.replaceAll( "-", "" );
591 }
592 }
593
594 int decimalLoc = cleanedValue.lastIndexOf( '.' );
595 if ( cleanedValue.indexOf( '.' ) != decimalLoc ) {
596 cleanedValue = cleanedValue.substring( 0, decimalLoc ).replaceAll( "\\.", "" ) + cleanedValue.substring( decimalLoc );
597 }
598 try {
599 return new BigDecimal( cleanedValue );
600 } catch ( NumberFormatException ex ) {
601 GlobalVariables.getMessageMap().putError(KRADConstants.DOCUMENT_ERRORS, RiceKeyConstants.ERROR_CUSTOM, new String[] { "Invalid Numeric Input: " + value });
602 return null;
603 }
604 }
605
606 public static String getDocumentDescription(String principalId, LocalDate effectiveDate) {
607 StringBuffer docDescription = new StringBuffer();
608 EntityNamePrincipalName principalName = null;
609 if (principalId != null) {
610 principalName = KimApiServiceLocator.getIdentityService().getDefaultNamesForPrincipalId(principalId);
611 }
612 String personName = (principalName != null && principalName.getDefaultName() != null) ? principalName.getDefaultName().getCompositeName() : StringUtils.EMPTY;
613 String date = TKUtils.formatDate(effectiveDate);
614 docDescription.append(personName + " (" + principalId + ") - " + date);
615 return docDescription.toString();
616 }
617
618
619
620
621
622 public static DateTime convertTimeForDifferentTimeZone(DateTime aDateTime, DateTimeZone fromTimeZone, DateTimeZone toTimeZone) {
623 if(fromTimeZone == null || toTimeZone == null || fromTimeZone.equals(toTimeZone))
624 return aDateTime;
625
626 Long millisOfSysTime = fromTimeZone.getMillisKeepLocal(toTimeZone, aDateTime.getMillis());
627 DateTime toTime = new DateTime(millisOfSysTime);
628
629 return toTime;
630 }
631
632
633 }