View Javadoc
1   /**
2    * Copyright 2004-2015 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.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       * @param dateString the format has to be mm/dd/yyyy
64       * @return dayOfMonth
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         // if the hours is exact duplicate of 24 hours
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 /*    public static String getAssignmentString(String principalId, Long jobNumber, Long workArea, Long task, LocalDate asOfDate) {
140     	StringBuilder builder = new StringBuilder();
141     	
142     	if (jobNumber != null && workArea != null && task != null) {
143         	Job jobObj = HrServiceLocator.getJobService().getJob(principalId, jobNumber, asOfDate);
144         	WorkArea workAreaObj = HrServiceLocator.getWorkAreaService().getWorkAreaWithoutRoles(workArea, asOfDate);
145         	Task taskObj = HrServiceLocator.getTaskService().getTask(task, asOfDate);
146         	
147         	String workAreaDescription = workAreaObj != null ? workAreaObj.getDescription() : StringUtils.EMPTY;
148         	KualiDecimal compensationRate = jobObj != null ? jobObj.getCompRate() : KualiDecimal.ZERO;
149         	String department = jobObj != null ? jobObj.getDept() : StringUtils.EMPTY;
150         	String taskDescription = taskObj != null && !HrConstants.TASK_DEFAULT_DESP.equals(taskObj.getDescription()) ? taskObj.getDescription() : StringUtils.EMPTY;
151         	
152         	builder.append(workAreaDescription).append(" : $").append(compensationRate).append(" Rcd ").append(jobNumber).append(" ").append(department);
153         	if (StringUtils.isNotBlank(taskDescription)) {
154         		builder.append(" ").append(taskDescription);
155         	}
156         }
157         
158         return builder.toString();
159     }*/
160 
161     /**
162      * Constructs a list of Day Spans for the pay calendar entry provided. You
163      * must also provide a DateTimeZone so that we can create relative boundaries.
164      *
165      * @param calendarEntry
166      * @param timeZone
167      * @return
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       * Compares and confirms if the start of the day is at midnight or on a virtual day boundary
214       * returns true if at midnight false otherwise(assuming 24 hr days)
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      * Creates a Timestamp object using Jodatime as an intermediate data structure
223      * from the provided date and time string. (From the form POST and javascript
224      * formats)
225      *
226      * @param dateStr (the format is 01/01/2011)
227      * @param timeStr (the format is 8:0)
228      * @return Timestamp
229      */
230     public static DateTime convertDateStringToDateTime(String dateStr, String timeStr) {
231         // the date/time format is defined in tk.calendar.js. For now, the format is 11/17/2010 8:0
232         String[] date = dateStr.split("/");
233         String[] time = timeStr.split(":");
234 
235         DateTimeZone dtz = DateTimeZone.forID(HrServiceLocator.getTimezoneService().getTargetUserTimezone());
236 
237         // this is from the jodattime javadoc:
238         // DateTime(int year, int monthOfYear, int dayOfMonth, int hourOfDay, int minuteOfHour, int secondOfMinute, int millisOfSecond, DateTimeZone zone)
239         // Noted that the month value is the actual month which is different than the java date object where the month value is the current month minus 1.
240         // I tried to use the actual month in the code as much as I can to reduce the convertions.
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         // the date/time format is defined in tk.calendar.js. For now, the format is 11/17/2010 8:0
254         String[] date = dateStr.split("/");
255         String[] time = timeStr.split(":");
256 
257         // this is from the jodattime javadoc:
258         // DateTime(int year, int monthOfYear, int dayOfMonth, int hourOfDay, int minuteOfHour, int secondOfMinute, int millisOfSecond, DateTimeZone zone)
259         // Noted that the month value is the actual month which is different than the java date object where the month value is the current month minus 1.
260         // I tried to use the actual month in the code as much as I can to reduce the convertions.
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         // the date/time format is defined in tk.calendar.js. For now, the format is 11/17/2010 8:0
274         String[] date = dateStr.split("/");
275         String[] time = timeStr.split(":");
276 
277         // this is from the jodattime javadoc:
278         // DateTime(int year, int monthOfYear, int dayOfMonth, int hourOfDay, int minuteOfHour, int secondOfMinute, int millisOfSecond, DateTimeZone zone)
279         // Noted that the month value is the actual month which is different than the java date object where the month value is the current month minus 1.
280         // I tried to use the actual month in the code as much as I can to reduce the convertions.
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         // Check for IPv6 addresses - Not sure what to do with them at this point.
294         // TODO: IPv6 - I see these on my local machine.
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         // Check for IPv6 addresses - Not sure what to do with them at this point.
313         // TODO: IPv6 - I see these on my local machine.
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     //Used to preserve active row fetching based on max(timestamp)
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      * Method to obtain the timezone offset string for the specified date time.
392      *
393      * Examples:
394      *
395      * -0500
396      * -0800
397      *
398      * @param date
399      * @return
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      * Get the session timeout time. If it's not defined in the (local) config file, give it a default time.
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      * Returns effectiveDate "from" date that's passed in through dateString.
510      * 
511 	 * The "from" dateString can either be from the format "fromDate..toDate" or ">=fromDate", otherwise an empty string is returned.
512 	 * 
513      * @param dateString
514      * @return
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      * Returns effectiveDate "to" date that's passed in through dateString.
530      * 
531      * The "to" dateString can either be from the format "fromDate..toDate" or "<=toDate", otherwise an empty string is returned.
532      *
533      * @param dateString
534      * @return
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 	 * Cleans a numerical input so that it can be successfully used with lookups
582 	 */
583 	public static BigDecimal cleanNumeric( String value ) {
584 		String cleanedValue = value.replaceAll( "[^-0-9.]", "" );
585 		// ensure only one "minus" at the beginning, if any
586 		if ( cleanedValue.lastIndexOf( '-' ) > 0 ) {
587 			if ( cleanedValue.charAt( 0 ) == '-' ) {
588 				cleanedValue = "-" + cleanedValue.replaceAll( "-", "" );
589 			} else {
590 				cleanedValue = cleanedValue.replaceAll( "-", "" );
591 			}
592 		}
593 		// ensure only one decimal in the string
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 	 * aDateTime is the dateTime we would like to display in the fromTimeZone. 
620 	 * The results of the method is the time showing in the toTimeZone considering the time difference between these two time zones
621 	 */
622 	public static DateTime convertTimeForDifferentTimeZone(DateTime aDateTime, DateTimeZone fromTimeZone, DateTimeZone toTimeZone) {
623 		if(fromTimeZone == null || toTimeZone == null || fromTimeZone.equals(toTimeZone))
624 			return aDateTime;	// no conversion is needed
625 		
626 		Long millisOfSysTime = fromTimeZone.getMillisKeepLocal(toTimeZone, aDateTime.getMillis());
627 		DateTime toTime = new DateTime(millisOfSysTime);
628 		
629 		return toTime;
630 	}
631 
632 
633 }