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