|  1 |     | 
     | 
  |  2 |     | 
     | 
  |  3 |     | 
     | 
  |  4 |     | 
     | 
  |  5 |     | 
     | 
  |  6 |     | 
     | 
  |  7 |     | 
     | 
  |  8 |     | 
     | 
  |  9 |     | 
     | 
  |  10 |     | 
     | 
  |  11 |     | 
     | 
  |  12 |     | 
     | 
  |  13 |     | 
     | 
  |  14 |     | 
     | 
  |  15 |     | 
     | 
  |  16 |     | 
   package org.kuali.rice.core.framework.persistence.jdbc.sql;  | 
  |  17 |     | 
     | 
  |  18 |     | 
   import java.sql.Timestamp;  | 
  |  19 |     | 
   import java.util.Arrays;  | 
  |  20 |     | 
   import java.util.Calendar;  | 
  |  21 |     | 
   import java.util.Collection;  | 
  |  22 |     | 
   import java.util.Collections;  | 
  |  23 |     | 
   import java.util.HashMap;  | 
  |  24 |     | 
   import java.util.Map;  | 
  |  25 |     | 
   import java.util.regex.Matcher;  | 
  |  26 |     | 
   import java.util.regex.Pattern;  | 
  |  27 |     | 
     | 
  |  28 |     | 
   import org.apache.commons.lang.StringUtils;  | 
  |  29 |     | 
   import org.kuali.rice.core.LogicalOperator;  | 
  |  30 |     | 
     | 
  |  31 |     | 
     | 
  |  32 |     | 
     | 
  |  33 |     | 
     | 
  |  34 |     | 
     | 
  |  35 |     | 
     | 
  |  36 |     | 
     | 
  |  37 |     | 
   public final class SQLUtils { | 
  |  38 |     | 
             | 
  |  39 |    0 |            private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(SQLUtils.class);  | 
  |  40 |     | 
     | 
  |  41 |     | 
           private static final String DATE_REGEX_SMALL_TWO_DIGIT_YEAR = "^\\d{1,2}/\\d{1,2}/\\d{2}$|^\\d{1,2}-\\d{1,2}-\\d{2}$";  | 
  |  42 |     | 
           private static final String DATE_REGEX_SMALL_TWO_DIGIT_YEAR_SPLIT = "(\\d{1,2})[/,-](\\d{1,2})[/,-](\\d{2})"; | 
  |  43 |     | 
           private static final String DATE_REGEX_SMALL_FOUR_DIGIT_YEAR = "^\\d{1,2}/\\d{1,2}/\\d{4}$|^\\d{1,2}-\\d{1,2}-\\d{4}$";  | 
  |  44 |     | 
           private static final String DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_SPLIT = "(\\d{1,2})[/,-](\\d{1,2})[/,-](\\d{4})"; | 
  |  45 |     | 
             | 
  |  46 |     | 
           private static final String DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST = "^\\d{4}/\\d{1,2}/\\d{1,2}$|^\\d{4}-\\d{1,2}-\\d{1,2}$";  | 
  |  47 |     | 
           private static final String DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST_SPLIT = "(\\d{4})[/,-](\\d{1,2})[/,-](\\d{1,2})"; | 
  |  48 |     | 
             | 
  |  49 |     | 
           private static final String DATE_REGEX_WHOLENUM_SMALL = "^\\d{6}$";  | 
  |  50 |     | 
           private static final String DATE_REGEX_WHOLENUM_SMALL_SPLIT = "(\\d{2})(\\d{2})(\\d{2})"; | 
  |  51 |     | 
           private static final String DATE_REGEX_WHOLENUM_LARGE = "^\\d{8}$";  | 
  |  52 |     | 
           private static final String DATE_REGEX_WHOLENUM_LARGE_SPLIT = "(\\d{2})(\\d{2})(\\d{4})"; | 
  |  53 |     | 
     | 
  |  54 |     | 
           private static final String TIME_REGEX = "([0-1][0-9]|2[0-3]):([0-5][0-9]):([0-5][0-9])";  | 
  |  55 |     | 
             | 
  |  56 |     | 
             | 
  |  57 |    0 |            private static final Collection<String> DOCUMENT_SEARCH_DATE_VALIDATION_REGEX_EXPRESSIONS = Collections.unmodifiableCollection(Arrays.asList(DATE_REGEX_SMALL_FOUR_DIGIT_YEAR, DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST));  | 
  |  58 |     | 
             | 
  |  59 |     | 
             | 
  |  60 |     | 
           private static final Map<String, String> REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION;  | 
  |  61 |     | 
           static { | 
  |  62 |    0 |                    final Map<String, String> map = new HashMap<String, String>();  | 
  |  63 |     | 
                     | 
  |  64 |    0 |                    map.put(DATE_REGEX_SMALL_TWO_DIGIT_YEAR, DATE_REGEX_SMALL_TWO_DIGIT_YEAR_SPLIT);  | 
  |  65 |    0 |                    map.put(DATE_REGEX_SMALL_FOUR_DIGIT_YEAR, DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_SPLIT);  | 
  |  66 |    0 |                    map.put(DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST, DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST_SPLIT);  | 
  |  67 |    0 |                    map.put(DATE_REGEX_WHOLENUM_SMALL, DATE_REGEX_WHOLENUM_SMALL_SPLIT);  | 
  |  68 |    0 |                    map.put(DATE_REGEX_WHOLENUM_LARGE,DATE_REGEX_WHOLENUM_LARGE_SPLIT);  | 
  |  69 |    0 |                    REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION = Collections.unmodifiableMap(map);  | 
  |  70 |    0 |            }  | 
  |  71 |     | 
             | 
  |  72 |    0 |            private SQLUtils() { | 
  |  73 |    0 |                    throw new UnsupportedOperationException("do not call"); | 
  |  74 |     | 
           }  | 
  |  75 |     | 
             | 
  |  76 |     | 
         | 
  |  77 |     | 
     | 
  |  78 |     | 
     | 
  |  79 |     | 
     | 
  |  80 |     | 
     | 
  |  81 |     | 
     | 
  |  82 |     | 
     | 
  |  83 |     | 
       public static String getSqlFormattedDate(String date) { | 
  |  84 |    0 |            DateComponent dc = formatDateToDateComponent(date, REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION.keySet());  | 
  |  85 |    0 |            if (dc == null) { | 
  |  86 |    0 |                return null;  | 
  |  87 |     | 
           }  | 
  |  88 |    0 |            return dc.getYear() + "/" + dc.getMonth() + "/" + dc.getDate();  | 
  |  89 |     | 
       }  | 
  |  90 |     | 
         | 
  |  91 |     | 
       public static Timestamp convertStringDateToTimestamp(String dateWithoutTime) { | 
  |  92 |    0 |            Pattern p = Pattern.compile(TIME_REGEX);  | 
  |  93 |    0 |            Matcher util = p.matcher(dateWithoutTime);  | 
  |  94 |    0 |            if (util.find()) { | 
  |  95 |    0 |                dateWithoutTime = StringUtils.substringBeforeLast(dateWithoutTime, " ");  | 
  |  96 |     | 
           }  | 
  |  97 |    0 |            DateComponent formattedDate = formatDateToDateComponent(dateWithoutTime, REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION.keySet());  | 
  |  98 |    0 |            if (formattedDate == null) { | 
  |  99 |    0 |                return null;  | 
  |  100 |     | 
           }  | 
  |  101 |    0 |            Calendar c = Calendar.getInstance();  | 
  |  102 |    0 |            c.clear();  | 
  |  103 |    0 |            c.set(Calendar.MONTH, Integer.valueOf(formattedDate.getMonth()).intValue() - 1);  | 
  |  104 |    0 |            c.set(Calendar.DATE, Integer.valueOf(formattedDate.getDate()).intValue());  | 
  |  105 |    0 |            c.set(Calendar.YEAR, Integer.valueOf(formattedDate.getYear()).intValue());  | 
  |  106 |    0 |            return convertCalendar(c);  | 
  |  107 |     | 
       }  | 
  |  108 |     | 
     | 
  |  109 |     | 
       private static class DateComponent { | 
  |  110 |     | 
           protected String month;  | 
  |  111 |     | 
           protected String date;  | 
  |  112 |     | 
           protected String year;  | 
  |  113 |     | 
     | 
  |  114 |    0 |            public DateComponent(String year, String month, String date) { | 
  |  115 |    0 |                this.month = month;  | 
  |  116 |    0 |                this.date = date;  | 
  |  117 |    0 |                this.year = year;  | 
  |  118 |    0 |            }  | 
  |  119 |     | 
     | 
  |  120 |     | 
           public String getDate() { | 
  |  121 |    0 |                return date;  | 
  |  122 |     | 
           }  | 
  |  123 |     | 
     | 
  |  124 |     | 
           public String getMonth() { | 
  |  125 |    0 |                return month;  | 
  |  126 |     | 
           }  | 
  |  127 |     | 
     | 
  |  128 |     | 
           public String getYear() { | 
  |  129 |    0 |                return year;  | 
  |  130 |     | 
           }  | 
  |  131 |     | 
       }  | 
  |  132 |     | 
         | 
  |  133 |     | 
         | 
  |  134 |     | 
     | 
  |  135 |     | 
     | 
  |  136 |     | 
     | 
  |  137 |     | 
     | 
  |  138 |     | 
     | 
  |  139 |     | 
     | 
  |  140 |     | 
       public static String getEntryFormattedDate(String date) { | 
  |  141 |    0 |            Pattern p = Pattern.compile(TIME_REGEX);  | 
  |  142 |    0 |            Matcher util = p.matcher(date);  | 
  |  143 |    0 |            if (util.find()) { | 
  |  144 |    0 |                date = StringUtils.substringBeforeLast(date, " ");  | 
  |  145 |     | 
           }  | 
  |  146 |    0 |            DateComponent dc = formatDateToDateComponent(date, DOCUMENT_SEARCH_DATE_VALIDATION_REGEX_EXPRESSIONS);  | 
  |  147 |    0 |            if (dc == null) { | 
  |  148 |    0 |                return null;  | 
  |  149 |     | 
           }  | 
  |  150 |    0 |            return dc.getMonth() + "/" + dc.getDate() + "/" + dc.getYear();  | 
  |  151 |     | 
       }  | 
  |  152 |     | 
     | 
  |  153 |     | 
       private static DateComponent formatDateToDateComponent(String date, Collection<String> regularExpressionList) { | 
  |  154 |    0 |            String matchingRegexExpression = null;  | 
  |  155 |    0 |            for (String string : regularExpressionList) { | 
  |  156 |    0 |                String matchRegex = string;  | 
  |  157 |    0 |                if (!REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION.containsKey(matchRegex)) { | 
  |  158 |    0 |                    String errorMsg = "";  | 
  |  159 |    0 |                    LOG.error("formatDateToDateComponent(String,List) " + errorMsg); | 
  |  160 |     | 
     | 
  |  161 |     | 
               }  | 
  |  162 |    0 |                Pattern p = Pattern.compile(matchRegex);  | 
  |  163 |    0 |                if ((p.matcher(date)).matches()) { | 
  |  164 |    0 |                    matchingRegexExpression = matchRegex;  | 
  |  165 |    0 |                    break;  | 
  |  166 |     | 
               }  | 
  |  167 |    0 |            }  | 
  |  168 |     | 
     | 
  |  169 |    0 |            if (matchingRegexExpression == null) { | 
  |  170 |    0 |                String errorMsg = "formatDate(String,List) Date string given '" + date + "' is not valid according to Workflow defaults.  Returning null value.";  | 
  |  171 |    0 |                if (StringUtils.isNotBlank(date)) { | 
  |  172 |    0 |                    LOG.warn(errorMsg);  | 
  |  173 |     | 
               } else { | 
  |  174 |    0 |                    LOG.debug(errorMsg);  | 
  |  175 |     | 
               }  | 
  |  176 |    0 |                return null;  | 
  |  177 |     | 
           }  | 
  |  178 |    0 |            String regexSplitExpression = REGEX_EXPRESSION_MAP_TO_REGEX_SPLIT_EXPRESSION.get(matchingRegexExpression);  | 
  |  179 |     | 
     | 
  |  180 |     | 
             | 
  |  181 |     | 
             | 
  |  182 |    0 |            Pattern p = Pattern.compile(regexSplitExpression);  | 
  |  183 |    0 |            Matcher util = p.matcher(date);  | 
  |  184 |    0 |            util.matches();  | 
  |  185 |    0 |            if (regexSplitExpression.equals(DATE_REGEX_SMALL_TWO_DIGIT_YEAR_SPLIT)) { | 
  |  186 |    0 |                StringBuffer yearBuf = new StringBuffer();  | 
  |  187 |    0 |                StringBuffer monthBuf = new StringBuffer();  | 
  |  188 |    0 |                StringBuffer dateBuf = new StringBuffer();  | 
  |  189 |    0 |                Integer year = Integer.valueOf(util.group(3));  | 
  |  190 |     | 
     | 
  |  191 |    0 |                if (year.intValue() <= 50) { | 
  |  192 |    0 |                    yearBuf.append("20").append(util.group(3)); | 
  |  193 |    0 |                } else if (util.group(3).length() < 3) { | 
  |  194 |    0 |                    yearBuf.append("19").append(util.group(3)); | 
  |  195 |     | 
               } else { | 
  |  196 |    0 |                    yearBuf.append(util.group(3));  | 
  |  197 |     | 
               }  | 
  |  198 |     | 
     | 
  |  199 |    0 |                if (util.group(1).length() < 2) { | 
  |  200 |    0 |                    monthBuf.append("0").append(util.group(1)); | 
  |  201 |     | 
               } else { | 
  |  202 |    0 |                    monthBuf.append(util.group(1));  | 
  |  203 |     | 
               }  | 
  |  204 |     | 
     | 
  |  205 |    0 |                if (util.group(2).length() < 2) { | 
  |  206 |    0 |                    dateBuf.append("0").append(util.group(2)); | 
  |  207 |     | 
               } else { | 
  |  208 |    0 |                    dateBuf.append(util.group(2));  | 
  |  209 |     | 
               }  | 
  |  210 |     | 
     | 
  |  211 |    0 |                return new DateComponent(yearBuf.toString(), monthBuf.toString(), dateBuf.toString());  | 
  |  212 |     | 
     | 
  |  213 |     | 
                 | 
  |  214 |    0 |            } else if (regexSplitExpression.equals(DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_SPLIT)) { | 
  |  215 |    0 |                StringBuffer yearBuf = new StringBuffer(util.group(3));  | 
  |  216 |    0 |                StringBuffer monthBuf = new StringBuffer();  | 
  |  217 |    0 |                StringBuffer dateBuf = new StringBuffer();  | 
  |  218 |     | 
     | 
  |  219 |    0 |                if (util.group(1).length() < 2) { | 
  |  220 |    0 |                    monthBuf.append("0").append(util.group(1)); | 
  |  221 |     | 
               } else { | 
  |  222 |    0 |                    monthBuf.append(util.group(1));  | 
  |  223 |     | 
               }  | 
  |  224 |     | 
     | 
  |  225 |    0 |                if (util.group(2).length() < 2) { | 
  |  226 |    0 |                    dateBuf.append("0").append(util.group(2)); | 
  |  227 |     | 
               } else { | 
  |  228 |    0 |                    dateBuf.append(util.group(2));  | 
  |  229 |     | 
               }  | 
  |  230 |     | 
     | 
  |  231 |    0 |                return new DateComponent(yearBuf.toString(), monthBuf.toString(), dateBuf.toString());  | 
  |  232 |     | 
     | 
  |  233 |     | 
                 | 
  |  234 |    0 |            } else if (regexSplitExpression.equals(DATE_REGEX_SMALL_FOUR_DIGIT_YEAR_FIRST_SPLIT)) { | 
  |  235 |    0 |                StringBuffer yearBuf = new StringBuffer(util.group(1));  | 
  |  236 |    0 |                StringBuffer monthBuf = new StringBuffer();  | 
  |  237 |    0 |                StringBuffer dateBuf = new StringBuffer();  | 
  |  238 |     | 
     | 
  |  239 |    0 |                if (util.group(2).length() < 2) { | 
  |  240 |    0 |                    monthBuf.append("0").append(util.group(2)); | 
  |  241 |     | 
               } else { | 
  |  242 |    0 |                    monthBuf.append(util.group(2));  | 
  |  243 |     | 
               }  | 
  |  244 |     | 
     | 
  |  245 |    0 |                if (util.group(3).length() < 2) { | 
  |  246 |    0 |                    dateBuf.append("0").append(util.group(3)); | 
  |  247 |     | 
               } else { | 
  |  248 |    0 |                    dateBuf.append(util.group(3));  | 
  |  249 |     | 
               }  | 
  |  250 |     | 
     | 
  |  251 |    0 |                return new DateComponent(yearBuf.toString(), monthBuf.toString(), dateBuf.toString());  | 
  |  252 |     | 
     | 
  |  253 |     | 
                 | 
  |  254 |    0 |            } else if (regexSplitExpression.equals(DATE_REGEX_WHOLENUM_LARGE_SPLIT)) { | 
  |  255 |    0 |                return new DateComponent(util.group(3), util.group(1), util.group(2));  | 
  |  256 |     | 
     | 
  |  257 |     | 
                 | 
  |  258 |    0 |            } else if (regexSplitExpression.equals(DATE_REGEX_WHOLENUM_SMALL_SPLIT)) { | 
  |  259 |    0 |                StringBuffer yearBuf = new StringBuffer();  | 
  |  260 |    0 |                Integer year = Integer.valueOf(util.group(3));  | 
  |  261 |     | 
     | 
  |  262 |    0 |                if (year.intValue() < 50) { | 
  |  263 |    0 |                    yearBuf.append("20"); | 
  |  264 |     | 
               } else { | 
  |  265 |    0 |                    yearBuf.append("19"); | 
  |  266 |     | 
               }  | 
  |  267 |    0 |                yearBuf.append(util.group(3));  | 
  |  268 |    0 |                return new DateComponent(yearBuf.toString(), util.group(1), util.group(2));  | 
  |  269 |     | 
           } else { | 
  |  270 |    0 |                LOG.warn("formatDate(String,List) Date string given '" + date + "' is not valid according to Workflow defaults.  Returning null value."); | 
  |  271 |    0 |                return null;  | 
  |  272 |     | 
           }  | 
  |  273 |     | 
       }  | 
  |  274 |     | 
         | 
  |  275 |     | 
       public static Calendar convertTimestamp(Timestamp timestamp) { | 
  |  276 |    0 |            if (timestamp == null) { | 
  |  277 |    0 |                return null;  | 
  |  278 |     | 
           }  | 
  |  279 |    0 |            Calendar calendar = Calendar.getInstance();  | 
  |  280 |    0 |            calendar.setTime(timestamp);  | 
  |  281 |    0 |            return calendar;  | 
  |  282 |     | 
       }  | 
  |  283 |     | 
     | 
  |  284 |     | 
       public static Timestamp convertCalendar(Calendar calendar) { | 
  |  285 |    0 |            if (calendar == null) { | 
  |  286 |    0 |                return null;  | 
  |  287 |     | 
           }  | 
  |  288 |    0 |            return new Timestamp(calendar.getTimeInMillis());  | 
  |  289 |     | 
       }  | 
  |  290 |     | 
        | 
  |  291 |     | 
           public static String cleanDate(String string) {                 | 
  |  292 |    0 |            for (LogicalOperator op : LogicalOperator.RANGE_CHARACTERS) { | 
  |  293 |    0 |                string = StringUtils.replace(string, op.op(), "");  | 
  |  294 |     | 
           }  | 
  |  295 |    0 |            return string;  | 
  |  296 |     | 
       }  | 
  |  297 |     | 
        | 
  |  298 |     | 
           public static String cleanNumericOfValidOperators(String string){ | 
  |  299 |    0 |                    for (LogicalOperator op : LogicalOperator.RANGE_CHARACTERS) { | 
  |  300 |    0 |                string = StringUtils.replace(string, op.op(), "");  | 
  |  301 |     | 
           }  | 
  |  302 |    0 |                    string = StringUtils.replace(string, LogicalOperator.OR.op(), "");  | 
  |  303 |    0 |                    string = StringUtils.replace(string, LogicalOperator.AND.op(), "");  | 
  |  304 |    0 |                    string = StringUtils.replace(string, LogicalOperator.NOT.op(), "");  | 
  |  305 |     | 
     | 
  |  306 |    0 |                    return string;  | 
  |  307 |     | 
           }  | 
  |  308 |     | 
             | 
  |  309 |     | 
         | 
  |  310 |     | 
     | 
  |  311 |     | 
     | 
  |  312 |     | 
     | 
  |  313 |     | 
     | 
  |  314 |     | 
     | 
  |  315 |     | 
       public static String cleanString(String string) { | 
  |  316 |    0 |            for (LogicalOperator op : LogicalOperator.QUERY_CHARACTERS) { | 
  |  317 |    0 |                string = StringUtils.replace(string, op.op(), "");  | 
  |  318 |     | 
           }  | 
  |  319 |    0 |            return string;  | 
  |  320 |     | 
       }  | 
  |  321 |     | 
   }  |