|         |      1 /* | 
|         |      2 ** 2003 October 31 | 
|         |      3 ** | 
|         |      4 ** The author disclaims copyright to this source code.  In place of | 
|         |      5 ** a legal notice, here is a blessing: | 
|         |      6 ** | 
|         |      7 **    May you do good and not evil. | 
|         |      8 **    May you find forgiveness for yourself and forgive others. | 
|         |      9 **    May you share freely, never taking more than you give. | 
|         |     10 ** | 
|         |     11 ************************************************************************* | 
|         |     12 ** This file contains the C functions that implement date and time | 
|         |     13 ** functions for SQLite.   | 
|         |     14 ** | 
|         |     15 ** There is only one exported symbol in this file - the function | 
|         |     16 ** sqlite3RegisterDateTimeFunctions() found at the bottom of the file. | 
|         |     17 ** All other code has file scope. | 
|         |     18 ** | 
|         |     19 ** $Id: date.cpp 1282 2008-11-13 09:31:33Z LarsPson $ | 
|         |     20 ** | 
|         |     21 ** SQLite processes all times and dates as Julian Day numbers.  The | 
|         |     22 ** dates and times are stored as the number of days since noon | 
|         |     23 ** in Greenwich on November 24, 4714 B.C. according to the Gregorian | 
|         |     24 ** calendar system.  | 
|         |     25 ** | 
|         |     26 ** 1970-01-01 00:00:00 is JD 2440587.5 | 
|         |     27 ** 2000-01-01 00:00:00 is JD 2451544.5 | 
|         |     28 ** | 
|         |     29 ** This implemention requires years to be expressed as a 4-digit number | 
|         |     30 ** which means that only dates between 0000-01-01 and 9999-12-31 can | 
|         |     31 ** be represented, even though julian day numbers allow a much wider | 
|         |     32 ** range of dates. | 
|         |     33 ** | 
|         |     34 ** The Gregorian calendar system is used for all dates and times, | 
|         |     35 ** even those that predate the Gregorian calendar.  Historians usually | 
|         |     36 ** use the Julian calendar for dates prior to 1582-10-15 and for some | 
|         |     37 ** dates afterwards, depending on locale.  Beware of this difference. | 
|         |     38 ** | 
|         |     39 ** The conversion algorithms are implemented based on descriptions | 
|         |     40 ** in the following text: | 
|         |     41 ** | 
|         |     42 **      Jean Meeus | 
|         |     43 **      Astronomical Algorithms, 2nd Edition, 1998 | 
|         |     44 **      ISBM 0-943396-61-1 | 
|         |     45 **      Willmann-Bell, Inc | 
|         |     46 **      Richmond, Virginia (USA) | 
|         |     47 */ | 
|         |     48 #include "sqliteInt.h" | 
|         |     49 #include <ctype.h> | 
|         |     50 #include <stdlib.h> | 
|         |     51 #include <assert.h> | 
|         |     52 #include <time.h> | 
|         |     53  | 
|         |     54 #ifndef SQLITE_OMIT_DATETIME_FUNCS | 
|         |     55  | 
|         |     56 /* | 
|         |     57 ** A structure for holding a single date and time. | 
|         |     58 */ | 
|         |     59 typedef struct DateTime DateTime; | 
|         |     60 struct DateTime { | 
|         |     61   double rJD;      /* The julian day number */ | 
|         |     62   int Y, M, D;     /* Year, month, and day */ | 
|         |     63   int h, m;        /* Hour and minutes */ | 
|         |     64   int tz;          /* Timezone offset in minutes */ | 
|         |     65   double s;        /* Seconds */ | 
|         |     66   char validYMD;   /* True if Y,M,D are valid */ | 
|         |     67   char validHMS;   /* True if h,m,s are valid */ | 
|         |     68   char validJD;    /* True if rJD is valid */ | 
|         |     69   char validTZ;    /* True if tz is valid */ | 
|         |     70 }; | 
|         |     71  | 
|         |     72  | 
|         |     73 /* | 
|         |     74 ** Convert zDate into one or more integers.  Additional arguments | 
|         |     75 ** come in groups of 5 as follows: | 
|         |     76 ** | 
|         |     77 **       N       number of digits in the integer | 
|         |     78 **       min     minimum allowed value of the integer | 
|         |     79 **       max     maximum allowed value of the integer | 
|         |     80 **       nextC   first character after the integer | 
|         |     81 **       pVal    where to write the integers value. | 
|         |     82 ** | 
|         |     83 ** Conversions continue until one with nextC==0 is encountered. | 
|         |     84 ** The function returns the number of successful conversions. | 
|         |     85 */ | 
|         |     86 static int getDigits(const char *zDate, ...){ | 
|         |     87   va_list ap; | 
|         |     88   int val; | 
|         |     89   int N; | 
|         |     90   int min; | 
|         |     91   int max; | 
|         |     92   int nextC; | 
|         |     93   int *pVal; | 
|         |     94   int cnt = 0; | 
|         |     95   va_start(ap, zDate); | 
|         |     96   do{ | 
|         |     97     N = va_arg(ap, int); | 
|         |     98     min = va_arg(ap, int); | 
|         |     99     max = va_arg(ap, int); | 
|         |    100     nextC = va_arg(ap, int); | 
|         |    101     pVal = va_arg(ap, int*); | 
|         |    102     val = 0; | 
|         |    103     while( N-- ){ | 
|         |    104       if( !isdigit(*(u8*)zDate) ){ | 
|         |    105         goto end_getDigits; | 
|         |    106       } | 
|         |    107       val = val*10 + *zDate - '0'; | 
|         |    108       zDate++; | 
|         |    109     } | 
|         |    110     if( val<min || val>max || (nextC!=0 && nextC!=*zDate) ){ | 
|         |    111       goto end_getDigits; | 
|         |    112     } | 
|         |    113     *pVal = val; | 
|         |    114     zDate++; | 
|         |    115     cnt++; | 
|         |    116   }while( nextC ); | 
|         |    117 end_getDigits: | 
|         |    118   va_end(ap); | 
|         |    119   return cnt; | 
|         |    120 } | 
|         |    121  | 
|         |    122 /* | 
|         |    123 ** Read text from z[] and convert into a floating point number.  Return | 
|         |    124 ** the number of digits converted. | 
|         |    125 */ | 
|         |    126 #define getValue sqlite3AtoF | 
|         |    127  | 
|         |    128 /* | 
|         |    129 ** Parse a timezone extension on the end of a date-time. | 
|         |    130 ** The extension is of the form: | 
|         |    131 ** | 
|         |    132 **        (+/-)HH:MM | 
|         |    133 ** | 
|         |    134 ** If the parse is successful, write the number of minutes | 
|         |    135 ** of change in *pnMin and return 0.  If a parser error occurs, | 
|         |    136 ** return 0. | 
|         |    137 ** | 
|         |    138 ** A missing specifier is not considered an error. | 
|         |    139 */ | 
|         |    140 static int parseTimezone(const char *zDate, DateTime *p){ | 
|         |    141   int sgn = 0; | 
|         |    142   int nHr, nMn; | 
|         |    143   while( isspace(*(u8*)zDate) ){ zDate++; } | 
|         |    144   p->tz = 0; | 
|         |    145   if( *zDate=='-' ){ | 
|         |    146     sgn = -1; | 
|         |    147   }else if( *zDate=='+' ){ | 
|         |    148     sgn = +1; | 
|         |    149   }else{ | 
|         |    150     return *zDate!=0; | 
|         |    151   } | 
|         |    152   zDate++; | 
|         |    153   if( getDigits(zDate, 2, 0, 14, ':', &nHr, 2, 0, 59, 0, &nMn)!=2 ){ | 
|         |    154     return 1; | 
|         |    155   } | 
|         |    156   zDate += 5; | 
|         |    157   p->tz = sgn*(nMn + nHr*60); | 
|         |    158   while( isspace(*(u8*)zDate) ){ zDate++; } | 
|         |    159   return *zDate!=0; | 
|         |    160 } | 
|         |    161  | 
|         |    162 /* | 
|         |    163 ** Parse times of the form HH:MM or HH:MM:SS or HH:MM:SS.FFFF. | 
|         |    164 ** The HH, MM, and SS must each be exactly 2 digits.  The | 
|         |    165 ** fractional seconds FFFF can be one or more digits. | 
|         |    166 ** | 
|         |    167 ** Return 1 if there is a parsing error and 0 on success. | 
|         |    168 */ | 
|         |    169 static int parseHhMmSs(const char *zDate, DateTime *p){ | 
|         |    170   int h, m, s; | 
|         |    171   double ms = 0.0; | 
|         |    172   if( getDigits(zDate, 2, 0, 24, ':', &h, 2, 0, 59, 0, &m)!=2 ){ | 
|         |    173     return 1; | 
|         |    174   } | 
|         |    175   zDate += 5; | 
|         |    176   if( *zDate==':' ){ | 
|         |    177     zDate++; | 
|         |    178     if( getDigits(zDate, 2, 0, 59, 0, &s)!=1 ){ | 
|         |    179       return 1; | 
|         |    180     } | 
|         |    181     zDate += 2; | 
|         |    182     if( *zDate=='.' && isdigit((u8)zDate[1]) ){ | 
|         |    183       double rScale = 1.0; | 
|         |    184       zDate++; | 
|         |    185       while( isdigit(*(u8*)zDate) ){ | 
|         |    186         ms = ms*10.0 + *zDate - '0'; | 
|         |    187         rScale *= 10.0; | 
|         |    188         zDate++; | 
|         |    189       } | 
|         |    190       ms /= rScale; | 
|         |    191     } | 
|         |    192   }else{ | 
|         |    193     s = 0; | 
|         |    194   } | 
|         |    195   p->validJD = 0; | 
|         |    196   p->validHMS = 1; | 
|         |    197   p->h = h; | 
|         |    198   p->m = m; | 
|         |    199   p->s = s + ms; | 
|         |    200   if( parseTimezone(zDate, p) ) return 1; | 
|         |    201   p->validTZ = p->tz!=0; | 
|         |    202   return 0; | 
|         |    203 } | 
|         |    204  | 
|         |    205 /* | 
|         |    206 ** Convert from YYYY-MM-DD HH:MM:SS to julian day.  We always assume | 
|         |    207 ** that the YYYY-MM-DD is according to the Gregorian calendar. | 
|         |    208 ** | 
|         |    209 ** Reference:  Meeus page 61 | 
|         |    210 */ | 
|         |    211 static void computeJD(DateTime *p){ | 
|         |    212   int Y, M, D, A, B, X1, X2; | 
|         |    213  | 
|         |    214   if( p->validJD ) return; | 
|         |    215   if( p->validYMD ){ | 
|         |    216     Y = p->Y; | 
|         |    217     M = p->M; | 
|         |    218     D = p->D; | 
|         |    219   }else{ | 
|         |    220     Y = 2000;  /* If no YMD specified, assume 2000-Jan-01 */ | 
|         |    221     M = 1; | 
|         |    222     D = 1; | 
|         |    223   } | 
|         |    224   if( M<=2 ){ | 
|         |    225     Y--; | 
|         |    226     M += 12; | 
|         |    227   } | 
|         |    228   A = Y/100; | 
|         |    229   B = 2 - A + (A/4); | 
|         |    230   X1 = 365.25*(Y+4716); | 
|         |    231   X2 = 30.6001*(M+1); | 
|         |    232   p->rJD = X1 + X2 + D + B - 1524.5; | 
|         |    233   p->validJD = 1; | 
|         |    234   if( p->validHMS ){ | 
|         |    235     p->rJD += (p->h*3600.0 + p->m*60.0 + p->s)/86400.0; | 
|         |    236     if( p->validTZ ){ | 
|         |    237       p->rJD -= p->tz*60/86400.0; | 
|         |    238       p->validYMD = 0; | 
|         |    239       p->validHMS = 0; | 
|         |    240       p->validTZ = 0; | 
|         |    241     } | 
|         |    242   } | 
|         |    243 } | 
|         |    244  | 
|         |    245 /* | 
|         |    246 ** Parse dates of the form | 
|         |    247 ** | 
|         |    248 **     YYYY-MM-DD HH:MM:SS.FFF | 
|         |    249 **     YYYY-MM-DD HH:MM:SS | 
|         |    250 **     YYYY-MM-DD HH:MM | 
|         |    251 **     YYYY-MM-DD | 
|         |    252 ** | 
|         |    253 ** Write the result into the DateTime structure and return 0 | 
|         |    254 ** on success and 1 if the input string is not a well-formed | 
|         |    255 ** date. | 
|         |    256 */ | 
|         |    257 static int parseYyyyMmDd(const char *zDate, DateTime *p){ | 
|         |    258   int Y, M, D, neg; | 
|         |    259  | 
|         |    260   if( zDate[0]=='-' ){ | 
|         |    261     zDate++; | 
|         |    262     neg = 1; | 
|         |    263   }else{ | 
|         |    264     neg = 0; | 
|         |    265   } | 
|         |    266   if( getDigits(zDate,4,0,9999,'-',&Y,2,1,12,'-',&M,2,1,31,0,&D)!=3 ){ | 
|         |    267     return 1; | 
|         |    268   } | 
|         |    269   zDate += 10; | 
|         |    270   while( isspace(*(u8*)zDate) || 'T'==*(u8*)zDate ){ zDate++; } | 
|         |    271   if( parseHhMmSs(zDate, p)==0 ){ | 
|         |    272     /* We got the time */ | 
|         |    273   }else if( *zDate==0 ){ | 
|         |    274     p->validHMS = 0; | 
|         |    275   }else{ | 
|         |    276     return 1; | 
|         |    277   } | 
|         |    278   p->validJD = 0; | 
|         |    279   p->validYMD = 1; | 
|         |    280   p->Y = neg ? -Y : Y; | 
|         |    281   p->M = M; | 
|         |    282   p->D = D; | 
|         |    283   if( p->validTZ ){ | 
|         |    284     computeJD(p); | 
|         |    285   } | 
|         |    286   return 0; | 
|         |    287 } | 
|         |    288  | 
|         |    289 /* | 
|         |    290 ** Attempt to parse the given string into a Julian Day Number.  Return | 
|         |    291 ** the number of errors. | 
|         |    292 ** | 
|         |    293 ** The following are acceptable forms for the input string: | 
|         |    294 ** | 
|         |    295 **      YYYY-MM-DD HH:MM:SS.FFF  +/-HH:MM | 
|         |    296 **      DDDD.DD  | 
|         |    297 **      now | 
|         |    298 ** | 
|         |    299 ** In the first form, the +/-HH:MM is always optional.  The fractional | 
|         |    300 ** seconds extension (the ".FFF") is optional.  The seconds portion | 
|         |    301 ** (":SS.FFF") is option.  The year and date can be omitted as long | 
|         |    302 ** as there is a time string.  The time string can be omitted as long | 
|         |    303 ** as there is a year and date. | 
|         |    304 */ | 
|         |    305 static int parseDateOrTime( | 
|         |    306   sqlite3_context *context,  | 
|         |    307   const char *zDate,  | 
|         |    308   DateTime *p | 
|         |    309 ){ | 
|         |    310   memset(p, 0, sizeof(*p)); | 
|         |    311   if( parseYyyyMmDd(zDate,p)==0 ){ | 
|         |    312     return 0; | 
|         |    313   }else if( parseHhMmSs(zDate, p)==0 ){ | 
|         |    314     return 0; | 
|         |    315   }else if( sqlite3StrICmp(zDate,"now")==0){ | 
|         |    316     double r; | 
|         |    317     sqlite3OsCurrentTime((sqlite3_vfs *)sqlite3_user_data(context), &r); | 
|         |    318     p->rJD = r; | 
|         |    319     p->validJD = 1; | 
|         |    320     return 0; | 
|         |    321   }else if( sqlite3IsNumber(zDate, 0, SQLITE_UTF8) ){ | 
|         |    322     getValue(zDate, &p->rJD); | 
|         |    323     p->validJD = 1; | 
|         |    324     return 0; | 
|         |    325   } | 
|         |    326   return 1; | 
|         |    327 } | 
|         |    328  | 
|         |    329 /* | 
|         |    330 ** Compute the Year, Month, and Day from the julian day number. | 
|         |    331 */ | 
|         |    332 static void computeYMD(DateTime *p){ | 
|         |    333   int Z, A, B, C, D, E, X1; | 
|         |    334   if( p->validYMD ) return; | 
|         |    335   if( !p->validJD ){ | 
|         |    336     p->Y = 2000; | 
|         |    337     p->M = 1; | 
|         |    338     p->D = 1; | 
|         |    339   }else{ | 
|         |    340     Z = p->rJD + 0.5; | 
|         |    341     A = (Z - 1867216.25)/36524.25; | 
|         |    342     A = Z + 1 + A - (A/4); | 
|         |    343     B = A + 1524; | 
|         |    344     C = (B - 122.1)/365.25; | 
|         |    345     D = 365.25*C; | 
|         |    346     E = (B-D)/30.6001; | 
|         |    347     X1 = 30.6001*E; | 
|         |    348     p->D = B - D - X1; | 
|         |    349     p->M = E<14 ? E-1 : E-13; | 
|         |    350     p->Y = p->M>2 ? C - 4716 : C - 4715; | 
|         |    351   } | 
|         |    352   p->validYMD = 1; | 
|         |    353 } | 
|         |    354  | 
|         |    355 /* | 
|         |    356 ** Compute the Hour, Minute, and Seconds from the julian day number. | 
|         |    357 */ | 
|         |    358 static void computeHMS(DateTime *p){ | 
|         |    359   int Z, s; | 
|         |    360   if( p->validHMS ) return; | 
|         |    361   computeJD(p); | 
|         |    362   Z = p->rJD + 0.5; | 
|         |    363   s = (p->rJD + 0.5 - Z)*86400000.0 + 0.5; | 
|         |    364   p->s = 0.001*s; | 
|         |    365   s = p->s; | 
|         |    366   p->s -= s; | 
|         |    367   p->h = s/3600; | 
|         |    368   s -= p->h*3600; | 
|         |    369   p->m = s/60; | 
|         |    370   p->s += s - p->m*60; | 
|         |    371   p->validHMS = 1; | 
|         |    372 } | 
|         |    373  | 
|         |    374 /* | 
|         |    375 ** Compute both YMD and HMS | 
|         |    376 */ | 
|         |    377 static void computeYMD_HMS(DateTime *p){ | 
|         |    378   computeYMD(p); | 
|         |    379   computeHMS(p); | 
|         |    380 } | 
|         |    381  | 
|         |    382 /* | 
|         |    383 ** Clear the YMD and HMS and the TZ | 
|         |    384 */ | 
|         |    385 static void clearYMD_HMS_TZ(DateTime *p){ | 
|         |    386   p->validYMD = 0; | 
|         |    387   p->validHMS = 0; | 
|         |    388   p->validTZ = 0; | 
|         |    389 } | 
|         |    390  | 
|         |    391 /* | 
|         |    392 ** Compute the difference (in days) between localtime and UTC (a.k.a. GMT) | 
|         |    393 ** for the time value p where p is in UTC. | 
|         |    394 */ | 
|         |    395 static double localtimeOffset(DateTime *p){ | 
|         |    396   DateTime x, y; | 
|         |    397   time_t t; | 
|         |    398   x = *p; | 
|         |    399   computeYMD_HMS(&x); | 
|         |    400   if( x.Y<1971 || x.Y>=2038 ){ | 
|         |    401     x.Y = 2000; | 
|         |    402     x.M = 1; | 
|         |    403     x.D = 1; | 
|         |    404     x.h = 0; | 
|         |    405     x.m = 0; | 
|         |    406     x.s = 0.0; | 
|         |    407   } else { | 
|         |    408     int s = x.s + 0.5; | 
|         |    409     x.s = s; | 
|         |    410   } | 
|         |    411   x.tz = 0; | 
|         |    412   x.validJD = 0; | 
|         |    413   computeJD(&x); | 
|         |    414   t = (x.rJD-2440587.5)*86400.0 + 0.5; | 
|         |    415 #ifdef HAVE_LOCALTIME_R | 
|         |    416   { | 
|         |    417     struct tm sLocal; | 
|         |    418     localtime_r(&t, &sLocal); | 
|         |    419     y.Y = sLocal.tm_year + 1900; | 
|         |    420     y.M = sLocal.tm_mon + 1; | 
|         |    421     y.D = sLocal.tm_mday; | 
|         |    422     y.h = sLocal.tm_hour; | 
|         |    423     y.m = sLocal.tm_min; | 
|         |    424     y.s = sLocal.tm_sec; | 
|         |    425   } | 
|         |    426 #else | 
|         |    427   { | 
|         |    428     struct tm *pTm; | 
|         |    429     sqlite3_mutex_enter(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)); | 
|         |    430     pTm = localtime(&t); | 
|         |    431     y.Y = pTm->tm_year + 1900; | 
|         |    432     y.M = pTm->tm_mon + 1; | 
|         |    433     y.D = pTm->tm_mday; | 
|         |    434     y.h = pTm->tm_hour; | 
|         |    435     y.m = pTm->tm_min; | 
|         |    436     y.s = pTm->tm_sec; | 
|         |    437     sqlite3_mutex_leave(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)); | 
|         |    438   } | 
|         |    439 #endif | 
|         |    440   y.validYMD = 1; | 
|         |    441   y.validHMS = 1; | 
|         |    442   y.validJD = 0; | 
|         |    443   y.validTZ = 0; | 
|         |    444   computeJD(&y); | 
|         |    445   return y.rJD - x.rJD; | 
|         |    446 } | 
|         |    447  | 
|         |    448 /* | 
|         |    449 ** Process a modifier to a date-time stamp.  The modifiers are | 
|         |    450 ** as follows: | 
|         |    451 ** | 
|         |    452 **     NNN days | 
|         |    453 **     NNN hours | 
|         |    454 **     NNN minutes | 
|         |    455 **     NNN.NNNN seconds | 
|         |    456 **     NNN months | 
|         |    457 **     NNN years | 
|         |    458 **     start of month | 
|         |    459 **     start of year | 
|         |    460 **     start of week | 
|         |    461 **     start of day | 
|         |    462 **     weekday N | 
|         |    463 **     unixepoch | 
|         |    464 **     localtime | 
|         |    465 **     utc | 
|         |    466 ** | 
|         |    467 ** Return 0 on success and 1 if there is any kind of error. | 
|         |    468 */ | 
|         |    469 static int parseModifier(const char *zMod, DateTime *p){ | 
|         |    470   int rc = 1; | 
|         |    471   int n; | 
|         |    472   double r; | 
|         |    473   char *z, zBuf[30]; | 
|         |    474   z = zBuf; | 
|         |    475   for(n=0; n<sizeof(zBuf)-1 && zMod[n]; n++){ | 
|         |    476     z[n] = tolower(zMod[n]); | 
|         |    477   } | 
|         |    478   z[n] = 0; | 
|         |    479   switch( z[0] ){ | 
|         |    480     case 'l': { | 
|         |    481       /*    localtime | 
|         |    482       ** | 
|         |    483       ** Assuming the current time value is UTC (a.k.a. GMT), shift it to | 
|         |    484       ** show local time. | 
|         |    485       */ | 
|         |    486       if( strcmp(z, "localtime")==0 ){ | 
|         |    487         computeJD(p); | 
|         |    488         p->rJD += localtimeOffset(p); | 
|         |    489         clearYMD_HMS_TZ(p); | 
|         |    490         rc = 0; | 
|         |    491       } | 
|         |    492       break; | 
|         |    493     } | 
|         |    494     case 'u': { | 
|         |    495       /* | 
|         |    496       **    unixepoch | 
|         |    497       ** | 
|         |    498       ** Treat the current value of p->rJD as the number of | 
|         |    499       ** seconds since 1970.  Convert to a real julian day number. | 
|         |    500       */ | 
|         |    501       if( strcmp(z, "unixepoch")==0 && p->validJD ){ | 
|         |    502         p->rJD = p->rJD/86400.0 + 2440587.5; | 
|         |    503         clearYMD_HMS_TZ(p); | 
|         |    504         rc = 0; | 
|         |    505       }else if( strcmp(z, "utc")==0 ){ | 
|         |    506         double c1; | 
|         |    507         computeJD(p); | 
|         |    508         c1 = localtimeOffset(p); | 
|         |    509         p->rJD -= c1; | 
|         |    510         clearYMD_HMS_TZ(p); | 
|         |    511         p->rJD += c1 - localtimeOffset(p); | 
|         |    512         rc = 0; | 
|         |    513       } | 
|         |    514       break; | 
|         |    515     } | 
|         |    516     case 'w': { | 
|         |    517       /* | 
|         |    518       **    weekday N | 
|         |    519       ** | 
|         |    520       ** Move the date to the same time on the next occurrence of | 
|         |    521       ** weekday N where 0==Sunday, 1==Monday, and so forth.  If the | 
|         |    522       ** date is already on the appropriate weekday, this is a no-op. | 
|         |    523       */ | 
|         |    524       if( strncmp(z, "weekday ", 8)==0 && getValue(&z[8],&r)>0 | 
|         |    525                  && (n=r)==r && n>=0 && r<7 ){ | 
|         |    526         int Z; | 
|         |    527         computeYMD_HMS(p); | 
|         |    528         p->validTZ = 0; | 
|         |    529         p->validJD = 0; | 
|         |    530         computeJD(p); | 
|         |    531         Z = p->rJD + 1.5; | 
|         |    532         Z %= 7; | 
|         |    533         if( Z>n ) Z -= 7; | 
|         |    534         p->rJD += n - Z; | 
|         |    535         clearYMD_HMS_TZ(p); | 
|         |    536         rc = 0; | 
|         |    537       } | 
|         |    538       break; | 
|         |    539     } | 
|         |    540     case 's': { | 
|         |    541       /* | 
|         |    542       **    start of TTTTT | 
|         |    543       ** | 
|         |    544       ** Move the date backwards to the beginning of the current day, | 
|         |    545       ** or month or year. | 
|         |    546       */ | 
|         |    547       if( strncmp(z, "start of ", 9)!=0 ) break; | 
|         |    548       z += 9; | 
|         |    549       computeYMD(p); | 
|         |    550       p->validHMS = 1; | 
|         |    551       p->h = p->m = 0; | 
|         |    552       p->s = 0.0; | 
|         |    553       p->validTZ = 0; | 
|         |    554       p->validJD = 0; | 
|         |    555       if( strcmp(z,"month")==0 ){ | 
|         |    556         p->D = 1; | 
|         |    557         rc = 0; | 
|         |    558       }else if( strcmp(z,"year")==0 ){ | 
|         |    559         computeYMD(p); | 
|         |    560         p->M = 1; | 
|         |    561         p->D = 1; | 
|         |    562         rc = 0; | 
|         |    563       }else if( strcmp(z,"day")==0 ){ | 
|         |    564         rc = 0; | 
|         |    565       } | 
|         |    566       break; | 
|         |    567     } | 
|         |    568     case '+': | 
|         |    569     case '-': | 
|         |    570     case '0': | 
|         |    571     case '1': | 
|         |    572     case '2': | 
|         |    573     case '3': | 
|         |    574     case '4': | 
|         |    575     case '5': | 
|         |    576     case '6': | 
|         |    577     case '7': | 
|         |    578     case '8': | 
|         |    579     case '9': { | 
|         |    580       n = getValue(z, &r); | 
|         |    581       assert( n>=1 ); | 
|         |    582       if( z[n]==':' ){ | 
|         |    583         /* A modifier of the form (+|-)HH:MM:SS.FFF adds (or subtracts) the | 
|         |    584         ** specified number of hours, minutes, seconds, and fractional seconds | 
|         |    585         ** to the time.  The ".FFF" may be omitted.  The ":SS.FFF" may be | 
|         |    586         ** omitted. | 
|         |    587         */ | 
|         |    588         const char *z2 = z; | 
|         |    589         DateTime tx; | 
|         |    590         int day; | 
|         |    591         if( !isdigit(*(u8*)z2) ) z2++; | 
|         |    592         memset(&tx, 0, sizeof(tx)); | 
|         |    593         if( parseHhMmSs(z2, &tx) ) break; | 
|         |    594         computeJD(&tx); | 
|         |    595         tx.rJD -= 0.5; | 
|         |    596         day = (int)tx.rJD; | 
|         |    597         tx.rJD -= day; | 
|         |    598         if( z[0]=='-' ) tx.rJD = -tx.rJD; | 
|         |    599         computeJD(p); | 
|         |    600         clearYMD_HMS_TZ(p); | 
|         |    601         p->rJD += tx.rJD; | 
|         |    602         rc = 0; | 
|         |    603         break; | 
|         |    604       } | 
|         |    605       z += n; | 
|         |    606       while( isspace(*(u8*)z) ) z++; | 
|         |    607       n = strlen(z); | 
|         |    608       if( n>10 || n<3 ) break; | 
|         |    609       if( z[n-1]=='s' ){ z[n-1] = 0; n--; } | 
|         |    610       computeJD(p); | 
|         |    611       rc = 0; | 
|         |    612       if( n==3 && strcmp(z,"day")==0 ){ | 
|         |    613         p->rJD += r; | 
|         |    614       }else if( n==4 && strcmp(z,"hour")==0 ){ | 
|         |    615         p->rJD += r/24.0; | 
|         |    616       }else if( n==6 && strcmp(z,"minute")==0 ){ | 
|         |    617         p->rJD += r/(24.0*60.0); | 
|         |    618       }else if( n==6 && strcmp(z,"second")==0 ){ | 
|         |    619         p->rJD += r/(24.0*60.0*60.0); | 
|         |    620       }else if( n==5 && strcmp(z,"month")==0 ){ | 
|         |    621         int x, y; | 
|         |    622         computeYMD_HMS(p); | 
|         |    623         p->M += r; | 
|         |    624         x = p->M>0 ? (p->M-1)/12 : (p->M-12)/12; | 
|         |    625         p->Y += x; | 
|         |    626         p->M -= x*12; | 
|         |    627         p->validJD = 0; | 
|         |    628         computeJD(p); | 
|         |    629         y = r; | 
|         |    630         if( y!=r ){ | 
|         |    631           p->rJD += (r - y)*30.0; | 
|         |    632         } | 
|         |    633       }else if( n==4 && strcmp(z,"year")==0 ){ | 
|         |    634         computeYMD_HMS(p); | 
|         |    635         p->Y += r; | 
|         |    636         p->validJD = 0; | 
|         |    637         computeJD(p); | 
|         |    638       }else{ | 
|         |    639         rc = 1; | 
|         |    640       } | 
|         |    641       clearYMD_HMS_TZ(p); | 
|         |    642       break; | 
|         |    643     } | 
|         |    644     default: { | 
|         |    645       break; | 
|         |    646     } | 
|         |    647   } | 
|         |    648   return rc; | 
|         |    649 } | 
|         |    650  | 
|         |    651 /* | 
|         |    652 ** Process time function arguments.  argv[0] is a date-time stamp. | 
|         |    653 ** argv[1] and following are modifiers.  Parse them all and write | 
|         |    654 ** the resulting time into the DateTime structure p.  Return 0 | 
|         |    655 ** on success and 1 if there are any errors. | 
|         |    656 */ | 
|         |    657 static int isDate( | 
|         |    658   sqlite3_context *context,  | 
|         |    659   int argc,  | 
|         |    660   sqlite3_value **argv,  | 
|         |    661   DateTime *p | 
|         |    662 ){ | 
|         |    663   int i; | 
|         |    664   const unsigned char *z; | 
|         |    665   if( argc==0 ) return 1; | 
|         |    666   z = sqlite3_value_text(argv[0]); | 
|         |    667   if( !z || parseDateOrTime(context, (char*)z, p) ){ | 
|         |    668     return 1; | 
|         |    669   } | 
|         |    670   for(i=1; i<argc; i++){ | 
|         |    671     if( (z = sqlite3_value_text(argv[i]))==0 || parseModifier((char*)z, p) ){ | 
|         |    672       return 1; | 
|         |    673     } | 
|         |    674   } | 
|         |    675   return 0; | 
|         |    676 } | 
|         |    677  | 
|         |    678  | 
|         |    679 /* | 
|         |    680 ** The following routines implement the various date and time functions | 
|         |    681 ** of SQLite. | 
|         |    682 */ | 
|         |    683  | 
|         |    684 /* | 
|         |    685 **    julianday( TIMESTRING, MOD, MOD, ...) | 
|         |    686 ** | 
|         |    687 ** Return the julian day number of the date specified in the arguments | 
|         |    688 */ | 
|         |    689 static void juliandayFunc( | 
|         |    690   sqlite3_context *context, | 
|         |    691   int argc, | 
|         |    692   sqlite3_value **argv | 
|         |    693 ){ | 
|         |    694   DateTime x; | 
|         |    695   if( isDate(context, argc, argv, &x)==0 ){ | 
|         |    696     computeJD(&x); | 
|         |    697     sqlite3_result_double(context, x.rJD); | 
|         |    698   } | 
|         |    699 } | 
|         |    700  | 
|         |    701 /* | 
|         |    702 **    datetime( TIMESTRING, MOD, MOD, ...) | 
|         |    703 ** | 
|         |    704 ** Return YYYY-MM-DD HH:MM:SS | 
|         |    705 */ | 
|         |    706 static void datetimeFunc( | 
|         |    707   sqlite3_context *context, | 
|         |    708   int argc, | 
|         |    709   sqlite3_value **argv | 
|         |    710 ){ | 
|         |    711   DateTime x; | 
|         |    712   if( isDate(context, argc, argv, &x)==0 ){ | 
|         |    713     char zBuf[100]; | 
|         |    714     computeYMD_HMS(&x); | 
|         |    715     sqlite3_snprintf(sizeof(zBuf), zBuf, "%04d-%02d-%02d %02d:%02d:%02d", | 
|         |    716                      x.Y, x.M, x.D, x.h, x.m, (int)(x.s)); | 
|         |    717     sqlite3_result_text(context, zBuf, -1, SQLITE_TRANSIENT); | 
|         |    718   } | 
|         |    719 } | 
|         |    720  | 
|         |    721 /* | 
|         |    722 **    time( TIMESTRING, MOD, MOD, ...) | 
|         |    723 ** | 
|         |    724 ** Return HH:MM:SS | 
|         |    725 */ | 
|         |    726 static void timeFunc( | 
|         |    727   sqlite3_context *context, | 
|         |    728   int argc, | 
|         |    729   sqlite3_value **argv | 
|         |    730 ){ | 
|         |    731   DateTime x; | 
|         |    732   if( isDate(context, argc, argv, &x)==0 ){ | 
|         |    733     char zBuf[100]; | 
|         |    734     computeHMS(&x); | 
|         |    735     sqlite3_snprintf(sizeof(zBuf), zBuf, "%02d:%02d:%02d", x.h, x.m, (int)x.s); | 
|         |    736     sqlite3_result_text(context, zBuf, -1, SQLITE_TRANSIENT); | 
|         |    737   } | 
|         |    738 } | 
|         |    739  | 
|         |    740 /* | 
|         |    741 **    date( TIMESTRING, MOD, MOD, ...) | 
|         |    742 ** | 
|         |    743 ** Return YYYY-MM-DD | 
|         |    744 */ | 
|         |    745 static void dateFunc( | 
|         |    746   sqlite3_context *context, | 
|         |    747   int argc, | 
|         |    748   sqlite3_value **argv | 
|         |    749 ){ | 
|         |    750   DateTime x; | 
|         |    751   if( isDate(context, argc, argv, &x)==0 ){ | 
|         |    752     char zBuf[100]; | 
|         |    753     computeYMD(&x); | 
|         |    754     sqlite3_snprintf(sizeof(zBuf), zBuf, "%04d-%02d-%02d", x.Y, x.M, x.D); | 
|         |    755     sqlite3_result_text(context, zBuf, -1, SQLITE_TRANSIENT); | 
|         |    756   } | 
|         |    757 } | 
|         |    758  | 
|         |    759 /* | 
|         |    760 **    strftime( FORMAT, TIMESTRING, MOD, MOD, ...) | 
|         |    761 ** | 
|         |    762 ** Return a string described by FORMAT.  Conversions as follows: | 
|         |    763 ** | 
|         |    764 **   %d  day of month | 
|         |    765 **   %f  ** fractional seconds  SS.SSS | 
|         |    766 **   %H  hour 00-24 | 
|         |    767 **   %j  day of year 000-366 | 
|         |    768 **   %J  ** Julian day number | 
|         |    769 **   %m  month 01-12 | 
|         |    770 **   %M  minute 00-59 | 
|         |    771 **   %s  seconds since 1970-01-01 | 
|         |    772 **   %S  seconds 00-59 | 
|         |    773 **   %w  day of week 0-6  sunday==0 | 
|         |    774 **   %W  week of year 00-53 | 
|         |    775 **   %Y  year 0000-9999 | 
|         |    776 **   %%  % | 
|         |    777 */ | 
|         |    778 static void strftimeFunc( | 
|         |    779   sqlite3_context *context, | 
|         |    780   int argc, | 
|         |    781   sqlite3_value **argv | 
|         |    782 ){ | 
|         |    783   DateTime x; | 
|         |    784   u64 n; | 
|         |    785   int i, j; | 
|         |    786   char *z; | 
|         |    787   const char *zFmt = (const char*)sqlite3_value_text(argv[0]); | 
|         |    788   char zBuf[100]; | 
|         |    789   if( zFmt==0 || isDate(context, argc-1, argv+1, &x) ) return; | 
|         |    790   for(i=0, n=1; zFmt[i]; i++, n++){ | 
|         |    791     if( zFmt[i]=='%' ){ | 
|         |    792       switch( zFmt[i+1] ){ | 
|         |    793         case 'd': | 
|         |    794         case 'H': | 
|         |    795         case 'm': | 
|         |    796         case 'M': | 
|         |    797         case 'S': | 
|         |    798         case 'W': | 
|         |    799           n++; | 
|         |    800           /* fall thru */ | 
|         |    801         case 'w': | 
|         |    802         case '%': | 
|         |    803           break; | 
|         |    804         case 'f': | 
|         |    805           n += 8; | 
|         |    806           break; | 
|         |    807         case 'j': | 
|         |    808           n += 3; | 
|         |    809           break; | 
|         |    810         case 'Y': | 
|         |    811           n += 8; | 
|         |    812           break; | 
|         |    813         case 's': | 
|         |    814         case 'J': | 
|         |    815           n += 50; | 
|         |    816           break; | 
|         |    817         default: | 
|         |    818           return;  /* ERROR.  return a NULL */ | 
|         |    819       } | 
|         |    820       i++; | 
|         |    821     } | 
|         |    822   } | 
|         |    823   if( n<sizeof(zBuf) ){ | 
|         |    824     z = zBuf; | 
|         |    825   }else if( n>SQLITE_MAX_LENGTH ){ | 
|         |    826     sqlite3_result_error_toobig(context); | 
|         |    827     return; | 
|         |    828   }else{ | 
|         |    829     z = (char*)sqlite3_malloc( n ); | 
|         |    830     if( z==0 ) return; | 
|         |    831   } | 
|         |    832   computeJD(&x); | 
|         |    833   computeYMD_HMS(&x); | 
|         |    834   for(i=j=0; zFmt[i]; i++){ | 
|         |    835     if( zFmt[i]!='%' ){ | 
|         |    836       z[j++] = zFmt[i]; | 
|         |    837     }else{ | 
|         |    838       i++; | 
|         |    839       switch( zFmt[i] ){ | 
|         |    840         case 'd':  sqlite3_snprintf(3, &z[j],"%02d",x.D); j+=2; break; | 
|         |    841         case 'f': { | 
|         |    842           double s = x.s; | 
|         |    843           if( s>59.999 ) s = 59.999; | 
|         |    844           sqlite3_snprintf(7, &z[j],"%06.3f", s); | 
|         |    845           j += strlen(&z[j]); | 
|         |    846           break; | 
|         |    847         } | 
|         |    848         case 'H':  sqlite3_snprintf(3, &z[j],"%02d",x.h); j+=2; break; | 
|         |    849         case 'W': /* Fall thru */ | 
|         |    850         case 'j': { | 
|         |    851           int nDay;             /* Number of days since 1st day of year */ | 
|         |    852           DateTime y = x; | 
|         |    853           y.validJD = 0; | 
|         |    854           y.M = 1; | 
|         |    855           y.D = 1; | 
|         |    856           computeJD(&y); | 
|         |    857           nDay = x.rJD - y.rJD + 0.5; | 
|         |    858           if( zFmt[i]=='W' ){ | 
|         |    859             int wd;   /* 0=Monday, 1=Tuesday, ... 6=Sunday */ | 
|         |    860             wd = ((int)(x.rJD+0.5)) % 7; | 
|         |    861             sqlite3_snprintf(3, &z[j],"%02d",(nDay+7-wd)/7); | 
|         |    862             j += 2; | 
|         |    863           }else{ | 
|         |    864             sqlite3_snprintf(4, &z[j],"%03d",nDay+1); | 
|         |    865             j += 3; | 
|         |    866           } | 
|         |    867           break; | 
|         |    868         } | 
|         |    869         case 'J': { | 
|         |    870           sqlite3_snprintf(20, &z[j],"%.16g",x.rJD); | 
|         |    871           j+=strlen(&z[j]); | 
|         |    872           break; | 
|         |    873         } | 
|         |    874         case 'm':  sqlite3_snprintf(3, &z[j],"%02d",x.M); j+=2; break; | 
|         |    875         case 'M':  sqlite3_snprintf(3, &z[j],"%02d",x.m); j+=2; break; | 
|         |    876         case 's': { | 
|         |    877           sqlite3_snprintf(30,&z[j],"%d", | 
|         |    878                            (int)((x.rJD-2440587.5)*86400.0 + 0.5)); | 
|         |    879           j += strlen(&z[j]); | 
|         |    880           break; | 
|         |    881         } | 
|         |    882         case 'S':  sqlite3_snprintf(3,&z[j],"%02d",(int)x.s); j+=2; break; | 
|         |    883         case 'w':  z[j++] = (((int)(x.rJD+1.5)) % 7) + '0'; break; | 
|         |    884         case 'Y':  sqlite3_snprintf(5,&z[j],"%04d",x.Y); j+=strlen(&z[j]);break; | 
|         |    885         case '%':  z[j++] = '%'; break; | 
|         |    886       } | 
|         |    887     } | 
|         |    888   } | 
|         |    889   z[j] = 0; | 
|         |    890   sqlite3_result_text(context, z, -1, SQLITE_TRANSIENT); | 
|         |    891   if( z!=zBuf ){ | 
|         |    892     sqlite3_free(z); | 
|         |    893   } | 
|         |    894 } | 
|         |    895  | 
|         |    896 /* | 
|         |    897 ** current_time() | 
|         |    898 ** | 
|         |    899 ** This function returns the same value as time('now'). | 
|         |    900 */ | 
|         |    901 static void ctimeFunc( | 
|         |    902   sqlite3_context *context, | 
|         |    903   int argc, | 
|         |    904   sqlite3_value **argv | 
|         |    905 ){ | 
|         |    906   sqlite3_value *pVal = sqlite3ValueNew(0); | 
|         |    907   if( pVal ){ | 
|         |    908     sqlite3ValueSetStr(pVal, -1, "now", SQLITE_UTF8, SQLITE_STATIC); | 
|         |    909     timeFunc(context, 1, &pVal); | 
|         |    910     sqlite3ValueFree(pVal); | 
|         |    911   } | 
|         |    912 } | 
|         |    913  | 
|         |    914 /* | 
|         |    915 ** current_date() | 
|         |    916 ** | 
|         |    917 ** This function returns the same value as date('now'). | 
|         |    918 */ | 
|         |    919 static void cdateFunc( | 
|         |    920   sqlite3_context *context, | 
|         |    921   int argc, | 
|         |    922   sqlite3_value **argv | 
|         |    923 ){ | 
|         |    924   sqlite3_value *pVal = sqlite3ValueNew(0); | 
|         |    925   if( pVal ){ | 
|         |    926     sqlite3ValueSetStr(pVal, -1, "now", SQLITE_UTF8, SQLITE_STATIC); | 
|         |    927     dateFunc(context, 1, &pVal); | 
|         |    928     sqlite3ValueFree(pVal); | 
|         |    929   } | 
|         |    930 } | 
|         |    931  | 
|         |    932 /* | 
|         |    933 ** current_timestamp() | 
|         |    934 ** | 
|         |    935 ** This function returns the same value as datetime('now'). | 
|         |    936 */ | 
|         |    937 static void ctimestampFunc( | 
|         |    938   sqlite3_context *context, | 
|         |    939   int argc, | 
|         |    940   sqlite3_value **argv | 
|         |    941 ){ | 
|         |    942   sqlite3_value *pVal = sqlite3ValueNew(0); | 
|         |    943   if( pVal ){ | 
|         |    944     sqlite3ValueSetStr(pVal, -1, "now", SQLITE_UTF8, SQLITE_STATIC); | 
|         |    945     datetimeFunc(context, 1, &pVal); | 
|         |    946     sqlite3ValueFree(pVal); | 
|         |    947   } | 
|         |    948 } | 
|         |    949 #endif /* !defined(SQLITE_OMIT_DATETIME_FUNCS) */ | 
|         |    950  | 
|         |    951 #ifdef SQLITE_OMIT_DATETIME_FUNCS | 
|         |    952 /* | 
|         |    953 ** If the library is compiled to omit the full-scale date and time | 
|         |    954 ** handling (to get a smaller binary), the following minimal version | 
|         |    955 ** of the functions current_time(), current_date() and current_timestamp() | 
|         |    956 ** are included instead. This is to support column declarations that | 
|         |    957 ** include "DEFAULT CURRENT_TIME" etc. | 
|         |    958 ** | 
|         |    959 ** This function uses the C-library functions time(), gmtime() | 
|         |    960 ** and strftime(). The format string to pass to strftime() is supplied | 
|         |    961 ** as the user-data for the function. | 
|         |    962 */ | 
|         |    963 static void currentTimeFunc( | 
|         |    964   sqlite3_context *context, | 
|         |    965   int argc, | 
|         |    966   sqlite3_value **argv | 
|         |    967 ){ | 
|         |    968   time_t t; | 
|         |    969   char *zFormat = (char *)sqlite3_user_data(context); | 
|         |    970   char zBuf[20]; | 
|         |    971  | 
|         |    972   time(&t); | 
|         |    973 #ifdef SQLITE_TEST | 
|         |    974   { | 
|         |    975     extern int sqlite3_current_time;  /* See os_XXX.c */ | 
|         |    976     if( sqlite3_current_time ){ | 
|         |    977       t = sqlite3_current_time; | 
|         |    978     } | 
|         |    979   } | 
|         |    980 #endif | 
|         |    981  | 
|         |    982 #ifdef HAVE_GMTIME_R | 
|         |    983   { | 
|         |    984     struct tm sNow; | 
|         |    985     gmtime_r(&t, &sNow); | 
|         |    986     strftime(zBuf, 20, zFormat, &sNow); | 
|         |    987   } | 
|         |    988 #else | 
|         |    989   { | 
|         |    990     struct tm *pTm; | 
|         |    991     sqlite3_mutex_enter(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)); | 
|         |    992     pTm = gmtime(&t); | 
|         |    993     strftime(zBuf, 20, zFormat, pTm); | 
|         |    994     sqlite3_mutex_leave(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)); | 
|         |    995   } | 
|         |    996 #endif | 
|         |    997  | 
|         |    998   sqlite3_result_text(context, zBuf, -1, SQLITE_TRANSIENT); | 
|         |    999 } | 
|         |   1000 #endif | 
|         |   1001  | 
|         |   1002 /* | 
|         |   1003 ** This function registered all of the above C functions as SQL | 
|         |   1004 ** functions.  This should be the only routine in this file with | 
|         |   1005 ** external linkage. | 
|         |   1006 */ | 
|         |   1007 void sqlite3RegisterDateTimeFunctions(sqlite3 *db){ | 
|         |   1008 #ifndef SQLITE_OMIT_DATETIME_FUNCS | 
|         |   1009   static const struct { | 
|         |   1010      char *zName; | 
|         |   1011      int nArg; | 
|         |   1012      void (*xFunc)(sqlite3_context*,int,sqlite3_value**); | 
|         |   1013   } aFuncs[] = { | 
|         |   1014     { "julianday", -1, juliandayFunc   }, | 
|         |   1015     { "date",      -1, dateFunc        }, | 
|         |   1016     { "time",      -1, timeFunc        }, | 
|         |   1017     { "datetime",  -1, datetimeFunc    }, | 
|         |   1018     { "strftime",  -1, strftimeFunc    }, | 
|         |   1019     { "current_time",       0, ctimeFunc      }, | 
|         |   1020     { "current_timestamp",  0, ctimestampFunc }, | 
|         |   1021     { "current_date",       0, cdateFunc      }, | 
|         |   1022   }; | 
|         |   1023   int i; | 
|         |   1024  | 
|         |   1025   for(i=0; i<sizeof(aFuncs)/sizeof(aFuncs[0]); i++){ | 
|         |   1026     sqlite3CreateFunc(db, aFuncs[i].zName, aFuncs[i].nArg, | 
|         |   1027         SQLITE_UTF8, (void *)(db->pVfs), aFuncs[i].xFunc, 0, 0); | 
|         |   1028   } | 
|         |   1029 #else | 
|         |   1030   static const struct { | 
|         |   1031      char *zName; | 
|         |   1032      char *zFormat; | 
|         |   1033   } aFuncs[] = { | 
|         |   1034     { "current_time", "%H:%M:%S" }, | 
|         |   1035     { "current_date", "%Y-%m-%d" }, | 
|         |   1036     { "current_timestamp", "%Y-%m-%d %H:%M:%S" } | 
|         |   1037   }; | 
|         |   1038   int i; | 
|         |   1039  | 
|         |   1040   for(i=0; i<sizeof(aFuncs)/sizeof(aFuncs[0]); i++){ | 
|         |   1041     sqlite3CreateFunc(db, aFuncs[i].zName, 0, SQLITE_UTF8,  | 
|         |   1042         aFuncs[i].zFormat, currentTimeFunc, 0, 0); | 
|         |   1043   } | 
|         |   1044 #endif | 
|         |   1045 } |