diff options
-rw-r--r-- | date.c | 206 |
1 files changed, 152 insertions, 54 deletions
@@ -157,75 +157,164 @@ static int match_alpha(const char *date, struct tm *tm, int *offset) return 0; } -static int match_digit(char *date, struct tm *tm, int *offset) +static int is_date(int year, int month, int day, struct tm *tm) { - char *end, c; - unsigned long num, num2, num3; - - num = strtoul(date, &end, 10); - - /* Time? num:num[:num] */ - if (num < 24 && end[0] == ':' && isdigit(end[1])) { - tm->tm_hour = num; - num = strtoul(end+1, &end, 10); - if (num < 60) { - tm->tm_min = num; - if (end[0] == ':' && isdigit(end[1])) { - num = strtoul(end+1, &end, 10); - if (num < 61) - tm->tm_sec = num; - } + if (month > 0 && month < 13 && day > 0 && day < 32) { + if (year == -1) { + tm->tm_mon = month-1; + tm->tm_mday = day; + return 1; } - return end - date; + if (year >= 1970 && year < 2100) { + year -= 1900; + } else if (year > 70 && year < 100) { + /* ok */ + } else if (year < 38) { + year += 100; + } else + return 0; + + tm->tm_mon = month-1; + tm->tm_mday = day; + tm->tm_year = year; + return 1; } + return 0; +} + +static int match_multi_number(unsigned long num, char c, char *date, char *end, struct tm *tm) +{ + long num2, num3; + + num2 = strtol(end+1, &end, 10); + num3 = -1; + if (*end == c && isdigit(end[1])) + num3 = strtol(end+1, &end, 10); - /* Year? Day of month? Numeric date-string?*/ - c = *end; + /* Time? Date? */ switch (c) { - default: - if (num > 0 && num < 32) { - tm->tm_mday = num; + case ':': + if (num3 < 0) + num3 = 0; + if (num < 25 && num2 >= 0 && num2 < 60 && num3 >= 0 && num3 <= 60) { + tm->tm_hour = num; + tm->tm_min = num2; + tm->tm_sec = num3; break; } - if (num > 1900) { - tm->tm_year = num - 1900; - break; - } - if (num > 70) { - tm->tm_year = num; - break; - } - break; + return 0; case '-': case '/': - if (num && num < 32 && isdigit(end[1])) { - num2 = strtoul(end+1, &end, 10); - if (!num2 || num2 > 31) + if (num > 70) { + /* yyyy-mm-dd? */ + if (is_date(num, num2, num3, tm)) break; - if (num > 12) { - if (num2 > 12) - break; - num3 = num; - num = num2; - num2 = num3; - } - tm->tm_mon = num - 1; - tm->tm_mday = num2; - if (*end == c && isdigit(end[1])) { - num3 = strtoul(end+1, &end, 10); - if (num3 > 1900) - num3 -= 1900; - else if (num3 < 38) - num3 += 100; - tm->tm_year = num3; - } + /* yyyy-dd-mm? */ + if (is_date(num, num3, num2, tm)) + break; + } + /* mm/dd/yy ? */ + if (is_date(num3, num2, num, tm)) + break; + /* dd/mm/yy ? */ + if (is_date(num3, num, num2, tm)) break; + return 0; + } + return end - date; +} + +/* + * We've seen a digit. Time? Year? Date? + */ +static int match_digit(char *date, struct tm *tm, int *offset) +{ + int n; + char *end; + unsigned long num; + + num = strtoul(date, &end, 10); + + /* + * Seconds since 1970? We trigger on that for anything after Jan 1, 2000 + */ + if (num > 946684800) { + time_t time = num; + if (gmtime_r(&time, tm)) + return end - date; + } + + /* + * Check for special formats: num[:-/]num[same]num + */ + switch (*end) { + case ':': + case '/': + case '-': + if (isdigit(end[1])) { + int match = match_multi_number(num, *end, date, end, tm); + if (match) + return match; + } + } + + /* + * None of the special formats? Try to guess what + * the number meant. We use the number of digits + * to make a more educated guess.. + */ + n = 0; + do { + n++; + } while (isdigit(date[n])); + + /* Four-digit year or a timezone? */ + if (n == 4) { + if (num <= 1200 && *offset == -1) { + unsigned int minutes = num % 100; + unsigned int hours = num / 100; + *offset = hours*60 + minutes; + } else if (num > 1900 && num < 2100) + tm->tm_year = num - 1900; + return n; + } + + /* + * NOTE! We will give precedence to day-of-month over month or + * year numebers in the 1-12 range. So 05 is always "mday 5", + * unless we already have a mday.. + * + * IOW, 01 Apr 05 parses as "April 1st, 2005". + */ + if (num > 0 && num < 32 && tm->tm_mday < 0) { + tm->tm_mday = num; + return n; + } + + /* Two-digit year? */ + if (n == 2 && tm->tm_year < 0) { + if (num < 10 && tm->tm_mday >= 0) { + tm->tm_year = num + 100; + return n; } + if (num >= 70) { + tm->tm_year = num; + return n; + } + } + + if (num > 0 && num < 32) { + tm->tm_mday = num; + } else if (num > 1900) { + tm->tm_year = num - 1900; + } else if (num > 70) { + tm->tm_year = num; + } else if (num > 0 && num < 13) { + tm->tm_mon = num-1; } - return end - date; - + return n; } static int match_tz(char *date, int *offp) @@ -233,10 +322,19 @@ static int match_tz(char *date, int *offp) char *end; int offset = strtoul(date+1, &end, 10); int min, hour; + int n = end - date - 1; min = offset % 100; hour = offset / 100; + /* + * Don't accept any random crap.. At least 3 digits, and + * a valid minute. We might want to check that the minutes + * are divisible by 30 or something too. + */ + if (min >= 60 || n < 3) + return 0; + offset = hour*60+min; if (*date == '-') offset = -offset; |