util_date.c
上传用户:xqtpzdz
上传日期:2022-05-21
资源大小:1764k
文件大小:12k
源码类别:

xml/soap/webservice

开发平台:

Visual C++

  1. /* ====================================================================
  2.  * The Apache Software License, Version 1.1
  3.  *
  4.  * Copyright (c) 2000 The Apache Software Foundation.  All rights
  5.  * reserved.
  6.  *
  7.  * Redistribution and use in source and binary forms, with or without
  8.  * modification, are permitted provided that the following conditions
  9.  * are met:
  10.  *
  11.  * 1. Redistributions of source code must retain the above copyright
  12.  *    notice, this list of conditions and the following disclaimer.
  13.  *
  14.  * 2. Redistributions in binary form must reproduce the above copyright
  15.  *    notice, this list of conditions and the following disclaimer in
  16.  *    the documentation and/or other materials provided with the
  17.  *    distribution.
  18.  *
  19.  * 3. The end-user documentation included with the redistribution,
  20.  *    if any, must include the following acknowledgment:
  21.  *       "This product includes software developed by the
  22.  *        Apache Software Foundation (http://www.apache.org/)."
  23.  *    Alternately, this acknowledgment may appear in the software itself,
  24.  *    if and wherever such third-party acknowledgments normally appear.
  25.  *
  26.  * 4. The names "Apache" and "Apache Software Foundation" must
  27.  *    not be used to endorse or promote products derived from this
  28.  *    software without prior written permission. For written
  29.  *    permission, please contact apache@apache.org.
  30.  *
  31.  * 5. Products derived from this software may not be called "Apache",
  32.  *    nor may "Apache" appear in their name, without prior written
  33.  *    permission of the Apache Software Foundation.
  34.  *
  35.  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
  36.  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  37.  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  38.  * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
  39.  * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  40.  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  41.  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  42.  * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  43.  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  44.  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  45.  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  46.  * SUCH DAMAGE.
  47.  * ====================================================================
  48.  *
  49.  * This software consists of voluntary contributions made by many
  50.  * individuals on behalf of the Apache Software Foundation.  For more
  51.  * information on the Apache Software Foundation, please see
  52.  * <http://www.apache.org/>.
  53.  *
  54.  * Portions of this software are based upon public domain software
  55.  * originally written at the National Center for Supercomputing Applications,
  56.  * University of Illinois, Urbana-Champaign.
  57.  */
  58. /*
  59.  * util_date.c: date parsing utility routines
  60.  *     These routines are (hopefully) platform-independent.
  61.  *
  62.  * 27 Oct 1996  Roy Fielding
  63.  *     Extracted (with many modifications) from mod_proxy.c and
  64.  *     tested with over 50,000 randomly chosen valid date strings
  65.  *     and several hundred variations of invalid date strings.
  66.  *
  67.  */
  68. /*#include "ap_config.h"*/
  69. #include "util_date.h"
  70. #include "ap_ctype.h"
  71. #include <string.h>
  72. #include <stdio.h>
  73. /*
  74.  * Compare a string to a mask
  75.  * Mask characters (arbitrary maximum is 256 characters, just in case):
  76.  *   @ - uppercase letter
  77.  *   $ - lowercase letter
  78.  *   & - hex digit
  79.  *   # - digit
  80.  *   ~ - digit or space
  81.  *   * - swallow remaining characters
  82.  *  <x> - exact match for any other character
  83.  */
  84. int ap_checkmask(const char *data, const char *mask)
  85. {
  86.   int i;
  87.   char d;
  88.   for (i = 0; i < 256; i++)
  89.   {
  90.     d = data[i];
  91.     switch (mask[i])
  92.     {
  93.      case '':
  94.        return (d == '');
  95.      case '*':
  96.        return 1;
  97.      case '@':
  98.        if (!ap_isupper(d)) return 0;
  99.        break;
  100.      case '$':
  101.        if (!ap_islower(d)) return 0;
  102.        break;
  103.      case '#':
  104.        if (!ap_isdigit(d))
  105.          return 0;
  106.        break;
  107.      case '&':
  108.        if (!ap_isxdigit(d))
  109.          return 0;
  110.        break;
  111.      case '~':
  112.        if ((d != ' ') && !ap_isdigit(d))
  113.          return 0;
  114.        break;
  115.      default:
  116.        if (mask[i] != d)
  117.          return 0;
  118.        break;
  119.     }
  120.   }
  121.   return 0; /* We only get here if mask is corrupted (exceeds 256) */
  122. }
  123. /*
  124.  * tm2sec converts a GMT tm structure into the number of seconds since
  125.  * 1st January 1970 UT.  Note that we ignore tm_wday, tm_yday, and tm_dst.
  126.  *
  127.  * The return value is always a valid time_t value -- (time_t)0 is returned
  128.  * if the input date is outside that capable of being represented by time(),
  129.  * i.e., before Thu, 01 Jan 1970 00:00:00 for all systems and
  130.  * beyond 2038 for 32bit systems.
  131.  *
  132.  * This routine is intended to be very fast, much faster than mktime().
  133.  */
  134. time_t ap_tm2sec(const struct tm * t)
  135. {
  136.     int year;
  137.     time_t days;
  138.     static const int dayoffset[12] =
  139.     {306, 337, 0, 31, 61, 92, 122, 153, 184, 214, 245, 275};
  140.     year = t->tm_year;
  141.     if (year < 70 || ((sizeof(time_t) <= 4) && (year >= 138)))
  142. return BAD_DATE;
  143.     /* shift new year to 1st March in order to make leap year calc easy */
  144.     if (t->tm_mon < 2)
  145. year--;
  146.     /* Find number of days since 1st March 1900 (in the Gregorian calendar). */
  147.     days = year * 365 + year / 4 - year / 100 + (year / 100 + 3) / 4;
  148.     days += dayoffset[t->tm_mon] + t->tm_mday - 1;
  149.     days -= 25508; /* 1 jan 1970 is 25508 days since 1 mar 1900 */
  150.     days = ((days * 24 + t->tm_hour) * 60 + t->tm_min) * 60 + t->tm_sec;
  151.     if (days < 0)
  152. return BAD_DATE; /* must have overflowed */
  153.     else
  154. return days; /* must be a valid time */
  155. }
  156. /*
  157.  * Parses an HTTP date in one of three standard forms:
  158.  *
  159.  *     Sun, 06 Nov 1994 08:49:37 GMT  ; RFC 822, updated by RFC 1123
  160.  *     Sunday, 06-Nov-94 08:49:37 GMT ; RFC 850, obsoleted by RFC 1036
  161.  *     Sun Nov  6 08:49:37 1994       ; ANSI C's asctime() format
  162.  *
  163.  * and returns the time_t number of seconds since 1 Jan 1970 GMT, or
  164.  * 0 if this would be out of range or if the date is invalid.
  165.  *
  166.  * The restricted HTTP syntax is
  167.  *
  168.  *     HTTP-date    = rfc1123-date | rfc850-date | asctime-date
  169.  *
  170.  *     rfc1123-date = wkday "," SP date1 SP time SP "GMT"
  171.  *     rfc850-date  = weekday "," SP date2 SP time SP "GMT"
  172.  *     asctime-date = wkday SP date3 SP time SP 4DIGIT
  173.  *
  174.  *     date1        = 2DIGIT SP month SP 4DIGIT
  175.  *                    ; day month year (e.g., 02 Jun 1982)
  176.  *     date2        = 2DIGIT "-" month "-" 2DIGIT
  177.  *                    ; day-month-year (e.g., 02-Jun-82)
  178.  *     date3        = month SP ( 2DIGIT | ( SP 1DIGIT ))
  179.  *                    ; month day (e.g., Jun  2)
  180.  *
  181.  *     time         = 2DIGIT ":" 2DIGIT ":" 2DIGIT
  182.  *                    ; 00:00:00 - 23:59:59
  183.  *
  184.  *     wkday        = "Mon" | "Tue" | "Wed"
  185.  *                  | "Thu" | "Fri" | "Sat" | "Sun"
  186.  *
  187.  *     weekday      = "Monday" | "Tuesday" | "Wednesday"
  188.  *                  | "Thursday" | "Friday" | "Saturday" | "Sunday"
  189.  *
  190.  *     month        = "Jan" | "Feb" | "Mar" | "Apr"
  191.  *                  | "May" | "Jun" | "Jul" | "Aug"
  192.  *                  | "Sep" | "Oct" | "Nov" | "Dec"
  193.  *
  194.  * However, for the sake of robustness (and Netscapeness), we ignore the
  195.  * weekday and anything after the time field (including the timezone).
  196.  *
  197.  * This routine is intended to be very fast; 10x faster than using sscanf.
  198.  *
  199.  * Originally from Andrew Daviel <andrew@vancouver-webpages.com>, 29 Jul 96
  200.  * but many changes since then.
  201.  *
  202.  */
  203. time_t ap_parseHTTPdate(const char *date)
  204. {
  205.     struct tm ds;
  206.     int mint, mon;
  207.     const char *monstr, *timstr;
  208.     static const int months[12] =
  209.     {
  210. ('J' << 16) | ('a' << 8) | 'n', ('F' << 16) | ('e' << 8) | 'b',
  211. ('M' << 16) | ('a' << 8) | 'r', ('A' << 16) | ('p' << 8) | 'r',
  212. ('M' << 16) | ('a' << 8) | 'y', ('J' << 16) | ('u' << 8) | 'n',
  213. ('J' << 16) | ('u' << 8) | 'l', ('A' << 16) | ('u' << 8) | 'g',
  214. ('S' << 16) | ('e' << 8) | 'p', ('O' << 16) | ('c' << 8) | 't',
  215. ('N' << 16) | ('o' << 8) | 'v', ('D' << 16) | ('e' << 8) | 'c'};
  216.     if (!date)
  217. return BAD_DATE;
  218.     while (*date && ap_isspace(*date)) /* Find first non-whitespace char */
  219. ++date;
  220.     if (*date == '')
  221. return BAD_DATE;
  222.     if ((date = strchr(date, ' ')) == NULL) /* Find space after weekday */
  223. return BAD_DATE;
  224.     ++date; /* Now pointing to first char after space, which should be */
  225.     /* start of the actual date information for all 3 formats. */
  226.     if (ap_checkmask(date, "## @$$ #### ##:##:## *")) { /* RFC 1123 format */
  227. ds.tm_year = ((date[7] - '0') * 10 + (date[8] - '0') - 19) * 100;
  228. if (ds.tm_year < 0)
  229.     return BAD_DATE;
  230. ds.tm_year += ((date[9] - '0') * 10) + (date[10] - '0');
  231. ds.tm_mday = ((date[0] - '0') * 10) + (date[1] - '0');
  232. monstr = date + 3;
  233. timstr = date + 12;
  234.     }
  235.     else if (ap_checkmask(date, "# @$$ #### ##:##:## *")) { /* RFC 1123 format one digit 1 */
  236. ds.tm_year = ((date[6] - '0') * 10 + (date[7] - '0') - 19) * 100;
  237. if (ds.tm_year < 0)
  238.     return BAD_DATE;
  239. ds.tm_year += ((date[8] - '0') * 10) + (date[9] - '0');
  240. ds.tm_mday = (date[0] - '0');
  241. monstr = date + 2;
  242. timstr = date + 11;
  243.     }
  244.     else if (ap_checkmask(date, "##-@$$-## ##:##:## *")) { /* RFC 850 format  */
  245. ds.tm_year = ((date[7] - '0') * 10) + (date[8] - '0');
  246. if (ds.tm_year < 70)
  247.     ds.tm_year += 100;
  248. ds.tm_mday = ((date[0] - '0') * 10) + (date[1] - '0');
  249. monstr = date + 3;
  250. timstr = date + 10;
  251.     }
  252.     else if (ap_checkmask(date, "@$$ ~# ##:##:## ####*")) { /* asctime format  */
  253. ds.tm_year = ((date[16] - '0') * 10 + (date[17] - '0') - 19) * 100;
  254. if (ds.tm_year < 0)
  255.     return BAD_DATE;
  256. ds.tm_year += ((date[18] - '0') * 10) + (date[19] - '0');
  257. if (date[4] == ' ')
  258.     ds.tm_mday = 0;
  259. else
  260.     ds.tm_mday = (date[4] - '0') * 10;
  261. ds.tm_mday += (date[5] - '0');
  262. monstr = date;
  263. timstr = date + 7;
  264.     }
  265.     else
  266. return BAD_DATE;
  267.     if (ds.tm_mday <= 0 || ds.tm_mday > 31)
  268. return BAD_DATE;
  269.     ds.tm_hour = ((timstr[0] - '0') * 10) + (timstr[1] - '0');
  270.     ds.tm_min = ((timstr[3] - '0') * 10) + (timstr[4] - '0');
  271.     ds.tm_sec = ((timstr[6] - '0') * 10) + (timstr[7] - '0');
  272.     if ((ds.tm_hour > 23) || (ds.tm_min > 59) || (ds.tm_sec > 61))
  273. return BAD_DATE;
  274.     mint = (monstr[0] << 16) | (monstr[1] << 8) | monstr[2];
  275.     for (mon = 0; mon < 12; mon++)
  276. if (mint == months[mon])
  277.     break;
  278.     if (mon == 12)
  279. return BAD_DATE;
  280.     if ((ds.tm_mday == 31) && (mon == 3 || mon == 5 || mon == 8 || mon == 10))
  281. return BAD_DATE;
  282.     /* February gets special check for leapyear */
  283.     if ((mon == 1) &&
  284. ((ds.tm_mday > 29)
  285.  || ((ds.tm_mday == 29)
  286.      && ((ds.tm_year & 3)
  287.  || (((ds.tm_year % 100) == 0)
  288.      && (((ds.tm_year % 400) != 100)))))))
  289. return BAD_DATE;
  290.     ds.tm_mon = mon;
  291.     return ap_tm2sec(&ds);
  292. }
  293. static const char ap_month_snames[12][4] =
  294. {
  295.     "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
  296. };
  297. static const char ap_day_snames[7][4] =
  298. {
  299.     "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
  300. };
  301. /*
  302.  * Outputs an HTTP date the standard form:
  303.  *
  304.  *     Sun, 06 Nov 1994 08:49:37 GMT  ; RFC 822, updated by RFC 1123
  305.  **/
  306. char *ap_gm_timestr_822(char *output, time_t sec)
  307. {
  308.   struct tm *tms;
  309.   tms = gmtime(&sec);
  310.   /* RFC date format; as strftime '%a, %d %b %Y %T GMT' */
  311.   sprintf(output,
  312.           "%s, %.2d %s %d %.2d:%.2d:%.2d GMT",
  313.           ap_day_snames[tms->tm_wday],
  314.           tms->tm_mday, ap_month_snames[tms->tm_mon], tms->tm_year + 1900,
  315.           tms->tm_hour, tms->tm_min, tms->tm_sec);
  316.   return output;
  317. }