1*640235e2SEnji Cooper /* $NetBSD: t_parsedate.c,v 1.25 2016/06/22 15:01:38 kre Exp $ */
257718be8SEnji Cooper /*-
3*640235e2SEnji Cooper  * Copyright (c) 2010, 2015 The NetBSD Foundation, Inc.
457718be8SEnji Cooper  * All rights reserved.
557718be8SEnji Cooper  *
657718be8SEnji Cooper  * Redistribution and use in source and binary forms, with or without
757718be8SEnji Cooper  * modification, are permitted provided that the following conditions
857718be8SEnji Cooper  * are met:
957718be8SEnji Cooper  *
1057718be8SEnji Cooper  * 1. Redistributions of source code must retain the above copyright
1157718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer.
1257718be8SEnji Cooper  * 2. Redistributions in binary form must reproduce the above copyright
1357718be8SEnji Cooper  *    notice, this list of conditions and the following disclaimer in
1457718be8SEnji Cooper  *    the documentation and/or other materials provided with the
1557718be8SEnji Cooper  *    distribution.
1657718be8SEnji Cooper  *
1757718be8SEnji Cooper  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
1857718be8SEnji Cooper  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
1957718be8SEnji Cooper  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
2057718be8SEnji Cooper  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
2157718be8SEnji Cooper  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
2257718be8SEnji Cooper  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
2357718be8SEnji Cooper  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
2457718be8SEnji Cooper  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
2557718be8SEnji Cooper  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
2657718be8SEnji Cooper  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
2757718be8SEnji Cooper  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2857718be8SEnji Cooper  * SUCH DAMAGE.
2957718be8SEnji Cooper  */
3057718be8SEnji Cooper 
3157718be8SEnji Cooper #include <sys/cdefs.h>
32*640235e2SEnji Cooper __RCSID("$NetBSD: t_parsedate.c,v 1.25 2016/06/22 15:01:38 kre Exp $");
3357718be8SEnji Cooper 
3457718be8SEnji Cooper #include <atf-c.h>
3557718be8SEnji Cooper #include <errno.h>
36*640235e2SEnji Cooper #include <stdio.h>
3757718be8SEnji Cooper #include <stdlib.h>
3857718be8SEnji Cooper #include <time.h>
3957718be8SEnji Cooper #include <util.h>
4057718be8SEnji Cooper 
41*640235e2SEnji Cooper /*
42*640235e2SEnji Cooper  * ANY is used as a placeholder for values that do not need to be
43*640235e2SEnji Cooper  * checked.  The actual value is arbitrary.  We don't use -1
44*640235e2SEnji Cooper  * because some tests might want to use -1 as a literal value.
45*640235e2SEnji Cooper  */
46*640235e2SEnji Cooper #define ANY -30215
47*640235e2SEnji Cooper 
48*640235e2SEnji Cooper /* parsecheck --
49*640235e2SEnji Cooper  * call parsedate(), then call time_to_tm() on the result,
50*640235e2SEnji Cooper  * and check that year/month/day/hour/minute/second are as expected.
51*640235e2SEnji Cooper  *
52*640235e2SEnji Cooper  * time_to_tm should usually be localtime_r or gmtime_r.
53*640235e2SEnji Cooper  *
54*640235e2SEnji Cooper  * Don't check values specified as ANY.
55*640235e2SEnji Cooper  */
56*640235e2SEnji Cooper static void
parsecheck(const char * datestr,const time_t * reftime,const int * zoff,struct tm * time_to_tm (const time_t *,struct tm *),int year,int month,int day,int hour,int minute,int second)57*640235e2SEnji Cooper parsecheck(const char *datestr, const time_t *reftime, const int *zoff,
58*640235e2SEnji Cooper 	struct tm * time_to_tm(const time_t *, struct tm *),
59*640235e2SEnji Cooper 	int year, int month, int day, int hour, int minute, int second)
60*640235e2SEnji Cooper {
61*640235e2SEnji Cooper 	time_t t;
62*640235e2SEnji Cooper 	struct tm tm;
63*640235e2SEnji Cooper 	char argstr[128];
64*640235e2SEnji Cooper 
65*640235e2SEnji Cooper 	/*
66*640235e2SEnji Cooper 	 * printable version of the args.
67*640235e2SEnji Cooper 	 *
68*640235e2SEnji Cooper 	 * Note that printf("%.*d", 0, 0)) prints nothing at all,
69*640235e2SEnji Cooper 	 * while printf("%.*d", 1, val) prints the value as usual.
70*640235e2SEnji Cooper 	 */
71*640235e2SEnji Cooper 	snprintf(argstr, sizeof(argstr), "%s%s%s, %s%.*jd, %s%.*d",
72*640235e2SEnji Cooper 		/* NULL or \"<datestr>\" */
73*640235e2SEnji Cooper 		(datestr ? "\"" : ""),
74*640235e2SEnji Cooper 		(datestr ? datestr : "NULL"),
75*640235e2SEnji Cooper 		(datestr ? "\"" : ""),
76*640235e2SEnji Cooper 		/* NULL or *reftime */
77*640235e2SEnji Cooper 		(reftime ? "" : "NULL"),
78*640235e2SEnji Cooper 		(reftime ? 1 : 0),
79*640235e2SEnji Cooper 		(reftime ? (intmax_t)*reftime : (intmax_t)0),
80*640235e2SEnji Cooper 		/* NULL or *zoff */
81*640235e2SEnji Cooper 		(zoff ? "" : "NULL"),
82*640235e2SEnji Cooper 		(zoff ? 1 : 0),
83*640235e2SEnji Cooper 		(zoff ? *zoff : 0));
84*640235e2SEnji Cooper 
85*640235e2SEnji Cooper 	ATF_CHECK_MSG((t = parsedate(datestr, reftime, zoff)) != -1,
86*640235e2SEnji Cooper 	    "parsedate(%s) returned -1\n", argstr);
87*640235e2SEnji Cooper 	ATF_CHECK(time_to_tm(&t, &tm) != NULL);
88*640235e2SEnji Cooper 	if (year != ANY)
89*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_year + 1900 == year,
90*640235e2SEnji Cooper 		    "parsedate(%s) expected year %d got %d (+1900)\n",
91*640235e2SEnji Cooper 		    argstr, year, (int)tm.tm_year);
92*640235e2SEnji Cooper 	if (month != ANY)
93*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_mon + 1 == month,
94*640235e2SEnji Cooper 		    "parsedate(%s) expected month %d got %d (+1)\n",
95*640235e2SEnji Cooper 		    argstr, month, (int)tm.tm_mon);
96*640235e2SEnji Cooper 	if (day != ANY)
97*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_mday == day,
98*640235e2SEnji Cooper 		    "parsedate(%s) expected day %d got %d\n",
99*640235e2SEnji Cooper 		    argstr, day, (int)tm.tm_mday);
100*640235e2SEnji Cooper 	if (hour != ANY)
101*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_hour == hour,
102*640235e2SEnji Cooper 		    "parsedate(%s) expected hour %d got %d\n",
103*640235e2SEnji Cooper 		    argstr, hour, (int)tm.tm_hour);
104*640235e2SEnji Cooper 	if (minute != ANY)
105*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_min == minute,
106*640235e2SEnji Cooper 		    "parsedate(%s) expected minute %d got %d\n",
107*640235e2SEnji Cooper 		    argstr, minute, (int)tm.tm_min);
108*640235e2SEnji Cooper 	if (second != ANY)
109*640235e2SEnji Cooper 		ATF_CHECK_MSG(tm.tm_sec == second,
110*640235e2SEnji Cooper 		    "parsedate(%s) expected second %d got %d\n",
111*640235e2SEnji Cooper 		    argstr, second, (int)tm.tm_sec);
112*640235e2SEnji Cooper }
113*640235e2SEnji Cooper 
11457718be8SEnji Cooper ATF_TC(dates);
11557718be8SEnji Cooper 
ATF_TC_HEAD(dates,tc)11657718be8SEnji Cooper ATF_TC_HEAD(dates, tc)
11757718be8SEnji Cooper {
11857718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test unambiguous dates"
11957718be8SEnji Cooper 	    " (PR lib/44255)");
12057718be8SEnji Cooper }
12157718be8SEnji Cooper 
ATF_TC_BODY(dates,tc)12257718be8SEnji Cooper ATF_TC_BODY(dates, tc)
12357718be8SEnji Cooper {
12457718be8SEnji Cooper 
125*640235e2SEnji Cooper 	parsecheck("9/10/69", NULL, NULL, localtime_r,
126*640235e2SEnji Cooper 		2069, 9, 10, 0, 0, 0); /* year < 70: add 2000 */
127*640235e2SEnji Cooper 	parsecheck("9/10/70", NULL, NULL, localtime_r,
128*640235e2SEnji Cooper 		1970, 9, 10, 0, 0, 0); /* 70 <= year < 100: add 1900 */
129*640235e2SEnji Cooper 	parsecheck("69-09-10", NULL, NULL, localtime_r,
130*640235e2SEnji Cooper 		69, 9, 10, 0, 0, 0); /* ISO8601 year remains unchanged */
131*640235e2SEnji Cooper 	parsecheck("70-09-10", NULL, NULL, localtime_r,
132*640235e2SEnji Cooper 		70, 9, 10, 0, 0, 0); /* ISO8601 year remains unchanged */
133*640235e2SEnji Cooper 	parsecheck("2006-11-17", NULL, NULL, localtime_r,
134*640235e2SEnji Cooper 		2006, 11, 17, 0, 0, 0);
135*640235e2SEnji Cooper 	parsecheck("10/1/2000", NULL, NULL, localtime_r,
136*640235e2SEnji Cooper 		2000, 10, 1, 0, 0, 0); /* month/day/year */
137*640235e2SEnji Cooper 	parsecheck("20 Jun 1994", NULL, NULL, localtime_r,
138*640235e2SEnji Cooper 		1994, 6, 20, 0, 0, 0);
139*640235e2SEnji Cooper 	parsecheck("97 September 2", NULL, NULL, localtime_r,
140*640235e2SEnji Cooper 		1997, 9, 2, 0, 0, 0);
141*640235e2SEnji Cooper 	parsecheck("23jun2001", NULL, NULL, localtime_r,
142*640235e2SEnji Cooper 		2001, 6, 23, 0, 0, 0);
143*640235e2SEnji Cooper 	parsecheck("1-sep-06", NULL, NULL, localtime_r,
144*640235e2SEnji Cooper 		2006, 9, 1, 0, 0, 0);
145*640235e2SEnji Cooper 	parsecheck("1/11", NULL, NULL, localtime_r,
146*640235e2SEnji Cooper 		ANY, 1, 11, 0, 0, 0); /* month/day */
147*640235e2SEnji Cooper 	parsecheck("1500-01-02", NULL, NULL, localtime_r,
148*640235e2SEnji Cooper 		1500, 1, 2, 0, 0, 0);
149*640235e2SEnji Cooper 	parsecheck("9999-12-21", NULL, NULL, localtime_r,
150*640235e2SEnji Cooper 		9999, 12, 21, 0, 0, 0);
151*640235e2SEnji Cooper 	parsecheck("2015.12.07.08.07.35", NULL, NULL, localtime_r,
152*640235e2SEnji Cooper 		2015, 12, 7, 8, 7, 35);
15357718be8SEnji Cooper }
15457718be8SEnji Cooper 
15557718be8SEnji Cooper ATF_TC(times);
15657718be8SEnji Cooper 
ATF_TC_HEAD(times,tc)15757718be8SEnji Cooper ATF_TC_HEAD(times, tc)
15857718be8SEnji Cooper {
15957718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test times"
16057718be8SEnji Cooper 	    " (PR lib/44255)");
16157718be8SEnji Cooper }
16257718be8SEnji Cooper 
ATF_TC_BODY(times,tc)16357718be8SEnji Cooper ATF_TC_BODY(times, tc)
16457718be8SEnji Cooper {
16557718be8SEnji Cooper 
166*640235e2SEnji Cooper 	parsecheck("10:01", NULL, NULL, localtime_r,
167*640235e2SEnji Cooper 		ANY, ANY, ANY, 10, 1, 0);
168*640235e2SEnji Cooper 	parsecheck("10:12pm", NULL, NULL, localtime_r,
169*640235e2SEnji Cooper 		ANY, ANY, ANY, 22, 12, 0);
170*640235e2SEnji Cooper 	parsecheck("12:11:01.000012", NULL, NULL, localtime_r,
171*640235e2SEnji Cooper 		ANY, ANY, ANY, 12, 11, 1);
172*640235e2SEnji Cooper 	parsecheck("12:21-0500", NULL, NULL, gmtime_r,
173*640235e2SEnji Cooper 		ANY, ANY, ANY, 12+5, 21, 0);
174*640235e2SEnji Cooper 	/* numeric zones not permitted with am/pm ... */
175*640235e2SEnji Cooper 	parsecheck("7 a.m. ICT", NULL, NULL, gmtime_r,
176*640235e2SEnji Cooper 		ANY, ANY, ANY, 7-7, 0, 0);
177*640235e2SEnji Cooper 	parsecheck("midnight", NULL, NULL, localtime_r,
178*640235e2SEnji Cooper 		ANY, ANY, ANY, 0, 0, 0);
179*640235e2SEnji Cooper 	parsecheck("mn", NULL, NULL, localtime_r,
180*640235e2SEnji Cooper 		ANY, ANY, ANY, 0, 0, 0);
181*640235e2SEnji Cooper 	parsecheck("noon", NULL, NULL, localtime_r,
182*640235e2SEnji Cooper 		ANY, ANY, ANY, 12, 0, 0);
183*640235e2SEnji Cooper }
184*640235e2SEnji Cooper 
185*640235e2SEnji Cooper ATF_TC(dsttimes);
186*640235e2SEnji Cooper 
ATF_TC_HEAD(dsttimes,tc)187*640235e2SEnji Cooper ATF_TC_HEAD(dsttimes, tc)
188*640235e2SEnji Cooper {
189*640235e2SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test DST transition times"
190*640235e2SEnji Cooper 	    " (PR lib/47916)");
191*640235e2SEnji Cooper }
192*640235e2SEnji Cooper 
ATF_TC_BODY(dsttimes,tc)193*640235e2SEnji Cooper ATF_TC_BODY(dsttimes, tc)
194*640235e2SEnji Cooper {
195*640235e2SEnji Cooper 	struct tm tm;
196*640235e2SEnji Cooper 	time_t t;
197*640235e2SEnji Cooper 	int tzoff;
198*640235e2SEnji Cooper 
199*640235e2SEnji Cooper 	putenv(__UNCONST("TZ=EST"));
200*640235e2SEnji Cooper 	tzset();
201*640235e2SEnji Cooper 	parsecheck("12:0", NULL, NULL, localtime_r,
202*640235e2SEnji Cooper 		ANY, ANY, ANY, 12, 0, 0);
203*640235e2SEnji Cooper 
204*640235e2SEnji Cooper 	putenv(__UNCONST("TZ=Asia/Tokyo"));
205*640235e2SEnji Cooper 	tzset();
206*640235e2SEnji Cooper 	parsecheck("12:0", NULL, NULL, localtime_r,
207*640235e2SEnji Cooper 		ANY, ANY, ANY, 12, 0, 0);
208*640235e2SEnji Cooper 
209*640235e2SEnji Cooper 	/*
210*640235e2SEnji Cooper 	 * When the effective local time is Tue Jul  9 13:21:53 BST 2013,
211*640235e2SEnji Cooper 	 * check mktime("14:00")
212*640235e2SEnji Cooper 	 */
213*640235e2SEnji Cooper 	putenv(__UNCONST("TZ=Europe/London"));
214*640235e2SEnji Cooper 	tzset();
215*640235e2SEnji Cooper 	tm = (struct tm){
216*640235e2SEnji Cooper 		.tm_year = 2013-1900, .tm_mon = 7-1, .tm_mday = 9,
217*640235e2SEnji Cooper 		.tm_hour = 13, .tm_min = 21, .tm_sec = 53,
218*640235e2SEnji Cooper 		.tm_isdst = 0 };
219*640235e2SEnji Cooper 	t = mktime(&tm);
220*640235e2SEnji Cooper 	ATF_CHECK(t != (time_t)-1);
221*640235e2SEnji Cooper 	parsecheck("14:00", &t, NULL, localtime_r,
222*640235e2SEnji Cooper 		2013, 7, 9, 14, 0, 0);
223*640235e2SEnji Cooper 	tzoff = -60; /* British Summer Time */
224*640235e2SEnji Cooper 	parsecheck("14:00", &t, &tzoff, localtime_r,
225*640235e2SEnji Cooper 		2013, 7, 9, 14, 0, 0);
22657718be8SEnji Cooper }
22757718be8SEnji Cooper 
22857718be8SEnji Cooper ATF_TC(relative);
22957718be8SEnji Cooper 
ATF_TC_HEAD(relative,tc)23057718be8SEnji Cooper ATF_TC_HEAD(relative, tc)
23157718be8SEnji Cooper {
23257718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test relative items"
23357718be8SEnji Cooper 	    " (PR lib/44255)");
23457718be8SEnji Cooper }
23557718be8SEnji Cooper 
ATF_TC_BODY(relative,tc)23657718be8SEnji Cooper ATF_TC_BODY(relative, tc)
23757718be8SEnji Cooper {
238*640235e2SEnji Cooper 	struct tm tm;
239*640235e2SEnji Cooper 	time_t now;
240*640235e2SEnji Cooper 
241*640235e2SEnji Cooper #define REL_CHECK(s, now, tm) do {					\
242*640235e2SEnji Cooper 	time_t p, q;							\
243*640235e2SEnji Cooper 	char nb[30], pb[30], qb[30];					\
244*640235e2SEnji Cooper 	p = parsedate(s, &now, NULL);					\
245*640235e2SEnji Cooper 	q = mktime(&tm);						\
246*640235e2SEnji Cooper 	ATF_CHECK_EQ_MSG(p, q,						\
247*640235e2SEnji Cooper 	    "From %jd (%24.24s) using \"%s\", obtained %jd (%24.24s); expected %jd (%24.24s)", \
248*640235e2SEnji Cooper 	    (uintmax_t)now, ctime_r(&now, nb),				\
249*640235e2SEnji Cooper 	    s, (uintmax_t)p, ctime_r(&p, pb), (uintmax_t)q, 		\
250*640235e2SEnji Cooper 	    ctime_r(&q, qb));						\
251*640235e2SEnji Cooper     } while (/*CONSTCOND*/0)
252*640235e2SEnji Cooper 
253*640235e2SEnji Cooper #define isleap(yr) (((yr) & 3) == 0 && (((yr) % 100) != 0 ||		\
254*640235e2SEnji Cooper 			((1900+(yr)) % 400) == 0))
25557718be8SEnji Cooper 
25657718be8SEnji Cooper 	ATF_CHECK(parsedate("-1 month", NULL, NULL) != -1);
25757718be8SEnji Cooper 	ATF_CHECK(parsedate("last friday", NULL, NULL) != -1);
25857718be8SEnji Cooper 	ATF_CHECK(parsedate("one week ago", NULL, NULL) != -1);
25957718be8SEnji Cooper 	ATF_CHECK(parsedate("this thursday", NULL, NULL) != -1);
26057718be8SEnji Cooper 	ATF_CHECK(parsedate("next sunday", NULL, NULL) != -1);
26157718be8SEnji Cooper 	ATF_CHECK(parsedate("+2 years", NULL, NULL) != -1);
262*640235e2SEnji Cooper 
263*640235e2SEnji Cooper 	/*
264*640235e2SEnji Cooper 	 * Test relative to a number of fixed dates.  Avoid the
265*640235e2SEnji Cooper 	 * edges of the time_t range to avert under- or overflow
266*640235e2SEnji Cooper 	 * of the relative date, and use a prime step for maximum
267*640235e2SEnji Cooper 	 * coverage of different times of day/week/month/year.
268*640235e2SEnji Cooper 	 */
269*640235e2SEnji Cooper 	for (now = 0x00FFFFFF; now < 0xFF000000; now += 3777779) {
270*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
271*640235e2SEnji Cooper 		tm.tm_mday--;
272*640235e2SEnji Cooper 		/* "yesterday" leaves time untouched */
273*640235e2SEnji Cooper 		tm.tm_isdst = -1;
274*640235e2SEnji Cooper 		REL_CHECK("yesterday", now, tm);
275*640235e2SEnji Cooper 
276*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
277*640235e2SEnji Cooper 		tm.tm_mday++;
278*640235e2SEnji Cooper 		/* as does "tomorrow" */
279*640235e2SEnji Cooper 		tm.tm_isdst = -1;
280*640235e2SEnji Cooper 		REL_CHECK("tomorrow", now, tm);
281*640235e2SEnji Cooper 
282*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
283*640235e2SEnji Cooper 		if (tm.tm_wday > 4)
284*640235e2SEnji Cooper 			tm.tm_mday += 7;
285*640235e2SEnji Cooper 		tm.tm_mday += 4 - tm.tm_wday;
286*640235e2SEnji Cooper 		/* if a day name is mentioned, it means midnight (by default) */
287*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = tm.tm_hour = 0;
288*640235e2SEnji Cooper 		tm.tm_isdst = -1;
289*640235e2SEnji Cooper 		REL_CHECK("this thursday", now, tm);
290*640235e2SEnji Cooper 
291*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
292*640235e2SEnji Cooper 		tm.tm_mday += 14 - (tm.tm_wday ? tm.tm_wday : 7);
293*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = tm.tm_hour = 0;
294*640235e2SEnji Cooper 		tm.tm_isdst = -1;
295*640235e2SEnji Cooper 		REL_CHECK("next sunday", now, tm);
296*640235e2SEnji Cooper 
297*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
298*640235e2SEnji Cooper 		if (tm.tm_wday <= 5)
299*640235e2SEnji Cooper 			tm.tm_mday -= 7;
300*640235e2SEnji Cooper 		tm.tm_mday += 5 - tm.tm_wday;
301*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = 0;
302*640235e2SEnji Cooper 		tm.tm_hour = 16;
303*640235e2SEnji Cooper 		tm.tm_isdst = -1;
304*640235e2SEnji Cooper 		REL_CHECK("last friday 4 p.m.", now, tm);
305*640235e2SEnji Cooper 
306*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
307*640235e2SEnji Cooper 		tm.tm_mday += 14;
308*640235e2SEnji Cooper 		if (tm.tm_wday > 3)
309*640235e2SEnji Cooper 			tm.tm_mday += 7;
310*640235e2SEnji Cooper 		tm.tm_mday += 3 - tm.tm_wday;
311*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = 0;
312*640235e2SEnji Cooper 		tm.tm_hour = 3;
313*640235e2SEnji Cooper 		tm.tm_isdst = -1;
314*640235e2SEnji Cooper 		REL_CHECK("we fortnight 3 a.m.", now, tm);
315*640235e2SEnji Cooper 
316*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
317*640235e2SEnji Cooper 		tm.tm_min -= 5;
318*640235e2SEnji Cooper 		tm.tm_isdst = -1;
319*640235e2SEnji Cooper 		REL_CHECK("5 minutes ago", now, tm);
320*640235e2SEnji Cooper 
321*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
322*640235e2SEnji Cooper 		tm.tm_hour++;
323*640235e2SEnji Cooper 		tm.tm_min += 37;
324*640235e2SEnji Cooper 		tm.tm_isdst = -1;
325*640235e2SEnji Cooper 		REL_CHECK("97 minutes", now, tm);
326*640235e2SEnji Cooper 
327*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
328*640235e2SEnji Cooper 		tm.tm_mon++;
329*640235e2SEnji Cooper 		if (tm.tm_mon == 1 &&
330*640235e2SEnji Cooper 		    tm.tm_mday > 28 + isleap(tm.tm_year))
331*640235e2SEnji Cooper 			tm.tm_mday = 28 + isleap(tm.tm_year);
332*640235e2SEnji Cooper 		else if ((tm.tm_mon == 3 || tm.tm_mon == 5 ||
333*640235e2SEnji Cooper 		    tm.tm_mon == 8 || tm.tm_mon == 10) && tm.tm_mday == 31)
334*640235e2SEnji Cooper 			tm.tm_mday = 30;
335*640235e2SEnji Cooper 		tm.tm_isdst = -1;
336*640235e2SEnji Cooper 		REL_CHECK("month", now, tm);
337*640235e2SEnji Cooper 
338*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
339*640235e2SEnji Cooper 		tm.tm_mon += 2;		/* "next" means add 2 ... */
340*640235e2SEnji Cooper 		if (tm.tm_mon == 13 &&
341*640235e2SEnji Cooper 		    tm.tm_mday > 28 + isleap(tm.tm_year + 1))
342*640235e2SEnji Cooper 			tm.tm_mday = 28 + isleap(tm.tm_year + 1);
343*640235e2SEnji Cooper 		else if (tm.tm_mon == 8 && tm.tm_mday == 31)
344*640235e2SEnji Cooper 			tm.tm_mday = 30;
345*640235e2SEnji Cooper 		tm.tm_isdst = -1;
346*640235e2SEnji Cooper 		REL_CHECK("next month", now, tm);
347*640235e2SEnji Cooper 
348*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
349*640235e2SEnji Cooper 		tm.tm_mon--;
350*640235e2SEnji Cooper 		if (tm.tm_mon == 1 &&
351*640235e2SEnji Cooper 		    tm.tm_mday > 28 + isleap(tm.tm_year))
352*640235e2SEnji Cooper 			tm.tm_mday = 28 + isleap(tm.tm_year);
353*640235e2SEnji Cooper 		else if ((tm.tm_mon == 3 || tm.tm_mon == 5 ||
354*640235e2SEnji Cooper 		    tm.tm_mon == 8 || tm.tm_mon == 10) && tm.tm_mday == 31)
355*640235e2SEnji Cooper 			tm.tm_mday = 30;
356*640235e2SEnji Cooper 		tm.tm_isdst = -1;
357*640235e2SEnji Cooper 		REL_CHECK("last month", now, tm);
358*640235e2SEnji Cooper 
359*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
360*640235e2SEnji Cooper 		tm.tm_mon += 6;
361*640235e2SEnji Cooper 		if (tm.tm_mon == 13 &&
362*640235e2SEnji Cooper 		    tm.tm_mday > 28 + isleap(tm.tm_year + 1))
363*640235e2SEnji Cooper 			tm.tm_mday = 28 + isleap(tm.tm_year + 1);
364*640235e2SEnji Cooper 		else if ((tm.tm_mon == 15 || tm.tm_mon == 17 ||
365*640235e2SEnji Cooper 		    tm.tm_mon == 8 || tm.tm_mon == 10) && tm.tm_mday == 31)
366*640235e2SEnji Cooper 			tm.tm_mday = 30;
367*640235e2SEnji Cooper 		tm.tm_mday += 2;
368*640235e2SEnji Cooper 		tm.tm_isdst = -1;
369*640235e2SEnji Cooper 		REL_CHECK("+6 months 2 days", now, tm);
370*640235e2SEnji Cooper 
371*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
372*640235e2SEnji Cooper 		tm.tm_mon -= 9;
373*640235e2SEnji Cooper 		if (tm.tm_mon == 1 && tm.tm_mday > 28 + isleap(tm.tm_year))
374*640235e2SEnji Cooper 			tm.tm_mday = 28 + isleap(tm.tm_year);
375*640235e2SEnji Cooper 		else if ((tm.tm_mon == -9 || tm.tm_mon == -7 ||
376*640235e2SEnji Cooper 		    tm.tm_mon == -2) && tm.tm_mday == 31)
377*640235e2SEnji Cooper 			tm.tm_mday = 30;
378*640235e2SEnji Cooper 		tm.tm_isdst = -1;
379*640235e2SEnji Cooper 		REL_CHECK("9 months ago", now, tm);
380*640235e2SEnji Cooper 
381*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
382*640235e2SEnji Cooper 		if (tm.tm_wday <= 2)
383*640235e2SEnji Cooper 			tm.tm_mday -= 7;
384*640235e2SEnji Cooper 		tm.tm_mday += 2 - tm.tm_wday;
385*640235e2SEnji Cooper 		tm.tm_isdst = -1;
386*640235e2SEnji Cooper 		tm.tm_hour = tm.tm_min = tm.tm_sec = 0;
387*640235e2SEnji Cooper 		REL_CHECK("1 week ago Tu", now, tm);
388*640235e2SEnji Cooper 
389*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
390*640235e2SEnji Cooper 		tm.tm_isdst = -1;
391*640235e2SEnji Cooper 		tm.tm_mday++;
392*640235e2SEnji Cooper 		tm.tm_hour = tm.tm_min = tm.tm_sec = 0;
393*640235e2SEnji Cooper 		REL_CHECK("midnight tomorrow", now, tm);
394*640235e2SEnji Cooper 
395*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
396*640235e2SEnji Cooper 		tm.tm_isdst = -1;
397*640235e2SEnji Cooper 		tm.tm_mday++;
398*640235e2SEnji Cooper 		tm.tm_hour = tm.tm_min = tm.tm_sec = 0;
399*640235e2SEnji Cooper 		REL_CHECK("tomorrow midnight", now, tm);
400*640235e2SEnji Cooper 
401*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
402*640235e2SEnji Cooper 		tm.tm_isdst = -1;
403*640235e2SEnji Cooper 		tm.tm_mday++;
404*640235e2SEnji Cooper 		tm.tm_hour = 12;
405*640235e2SEnji Cooper 		tm.tm_min = tm.tm_sec = 0;
406*640235e2SEnji Cooper 		REL_CHECK("noon tomorrow", now, tm);
407*640235e2SEnji Cooper 
408*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
409*640235e2SEnji Cooper 		if (tm.tm_wday > 2)
410*640235e2SEnji Cooper 			tm.tm_mday += 7;
411*640235e2SEnji Cooper 		tm.tm_mday += 2 - tm.tm_wday;
412*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = tm.tm_hour = 0;
413*640235e2SEnji Cooper 		tm.tm_isdst = -1;
414*640235e2SEnji Cooper 		REL_CHECK("midnight Tuesday", now, tm);
415*640235e2SEnji Cooper 
416*640235e2SEnji Cooper 		ATF_CHECK(localtime_r(&now, &tm) != NULL);
417*640235e2SEnji Cooper 		if (tm.tm_wday > 2 + 1)
418*640235e2SEnji Cooper 			tm.tm_mday += 7;
419*640235e2SEnji Cooper 		tm.tm_mday += 2 - tm.tm_wday;
420*640235e2SEnji Cooper 		tm.tm_mday++;	/* xxx midnight --> the next day */
421*640235e2SEnji Cooper 		tm.tm_sec = tm.tm_min = tm.tm_hour = 0;
422*640235e2SEnji Cooper 		tm.tm_isdst = -1;
423*640235e2SEnji Cooper 		REL_CHECK("Tuesday midnight", now, tm);
424*640235e2SEnji Cooper 	}
42557718be8SEnji Cooper }
42657718be8SEnji Cooper 
42757718be8SEnji Cooper ATF_TC(atsecs);
42857718be8SEnji Cooper 
ATF_TC_HEAD(atsecs,tc)42957718be8SEnji Cooper ATF_TC_HEAD(atsecs, tc)
43057718be8SEnji Cooper {
43157718be8SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test seconds past the epoch");
43257718be8SEnji Cooper }
43357718be8SEnji Cooper 
ATF_TC_BODY(atsecs,tc)43457718be8SEnji Cooper ATF_TC_BODY(atsecs, tc)
43557718be8SEnji Cooper {
43657718be8SEnji Cooper 	int tzoff;
43757718be8SEnji Cooper 
43857718be8SEnji Cooper 	/* "@0" -> (time_t)0, regardless of timezone */
43957718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, NULL) == (time_t)0);
44057718be8SEnji Cooper 	putenv(__UNCONST("TZ=Europe/Berlin"));
44157718be8SEnji Cooper 	tzset();
44257718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, NULL) == (time_t)0);
44357718be8SEnji Cooper 	putenv(__UNCONST("TZ=America/New_York"));
44457718be8SEnji Cooper 	tzset();
44557718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, NULL) == (time_t)0);
44657718be8SEnji Cooper 	tzoff = 0;
44757718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, &tzoff) == (time_t)0);
44857718be8SEnji Cooper 	tzoff = 3600;
44957718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, &tzoff) == (time_t)0);
45057718be8SEnji Cooper 	tzoff = -3600;
45157718be8SEnji Cooper 	ATF_CHECK(parsedate("@0", NULL, &tzoff) == (time_t)0);
45257718be8SEnji Cooper 
45357718be8SEnji Cooper 	/* -1 or other negative numbers are not errors */
45457718be8SEnji Cooper 	errno = 0;
45557718be8SEnji Cooper 	ATF_CHECK(parsedate("@-1", NULL, &tzoff) == (time_t)-1 && errno == 0);
45657718be8SEnji Cooper 	ATF_CHECK(parsedate("@-2", NULL, &tzoff) == (time_t)-2 && errno == 0);
45757718be8SEnji Cooper 
45857718be8SEnji Cooper 	/* junk is an error */
45957718be8SEnji Cooper 	errno = 0;
46057718be8SEnji Cooper 	ATF_CHECK(parsedate("@junk", NULL, NULL) == (time_t)-1 && errno != 0);
46157718be8SEnji Cooper }
46257718be8SEnji Cooper 
463*640235e2SEnji Cooper ATF_TC(zones);
464*640235e2SEnji Cooper 
ATF_TC_HEAD(zones,tc)465*640235e2SEnji Cooper ATF_TC_HEAD(zones, tc)
466*640235e2SEnji Cooper {
467*640235e2SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test parsing dates with zones");
468*640235e2SEnji Cooper }
469*640235e2SEnji Cooper 
ATF_TC_BODY(zones,tc)470*640235e2SEnji Cooper ATF_TC_BODY(zones, tc)
471*640235e2SEnji Cooper {
472*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 UTC", NULL, NULL, gmtime_r,
473*640235e2SEnji Cooper 		2015, 12, 6, 16, 11, 48);
474*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 UT", NULL, NULL, gmtime_r,
475*640235e2SEnji Cooper 		2015, 12, 6, 16, 11, 48);
476*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 GMT", NULL, NULL, gmtime_r,
477*640235e2SEnji Cooper 		2015, 12, 6, 16, 11, 48);
478*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 +0000", NULL, NULL, gmtime_r,
479*640235e2SEnji Cooper 		2015, 12, 6, 16, 11, 48);
480*640235e2SEnji Cooper 
481*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 -0500", NULL, NULL, gmtime_r,
482*640235e2SEnji Cooper 		2015, 12, 6, 21, 11, 48);
483*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 EST", NULL, NULL, gmtime_r,
484*640235e2SEnji Cooper 		2015, 12, 6, 21, 11, 48);
485*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 EDT", NULL, NULL, gmtime_r,
486*640235e2SEnji Cooper 		2015, 12, 6, 20, 11, 48);
487*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 +0500", NULL, NULL, gmtime_r,
488*640235e2SEnji Cooper 		2015, 12, 6, 11, 11, 48);
489*640235e2SEnji Cooper 
490*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 +1000", NULL, NULL, gmtime_r,
491*640235e2SEnji Cooper 		2015, 12, 6, 6, 11, 48);
492*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 AEST", NULL, NULL, gmtime_r,
493*640235e2SEnji Cooper 		2015, 12, 6, 6, 11, 48);
494*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 -1000", NULL, NULL, gmtime_r,
495*640235e2SEnji Cooper 		2015, 12, 7, 2, 11, 48);
496*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 HST", NULL, NULL, gmtime_r,
497*640235e2SEnji Cooper 		2015, 12, 7, 2, 11, 48);
498*640235e2SEnji Cooper 
499*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 AWST", NULL, NULL, gmtime_r,
500*640235e2SEnji Cooper 		2015, 12, 6, 8, 11, 48);
501*640235e2SEnji Cooper 	parsecheck("2015-12-06 16:11:48 NZDT", NULL, NULL, gmtime_r,
502*640235e2SEnji Cooper 		2015, 12, 6, 3, 11, 48);
503*640235e2SEnji Cooper 
504*640235e2SEnji Cooper         parsecheck("Sun, 6 Dec 2015 09:43:16 -0500", NULL, NULL, gmtime_r,
505*640235e2SEnji Cooper 		2015, 12, 6, 14, 43, 16);
506*640235e2SEnji Cooper 	parsecheck("Mon Dec  7 03:13:31 ICT 2015", NULL, NULL, gmtime_r,
507*640235e2SEnji Cooper 		2015, 12, 6, 20, 13, 31);
508*640235e2SEnji Cooper 	/* the day name is ignored when a day of month (etc) is given... */
509*640235e2SEnji Cooper 	parsecheck("Sat Dec  7 03:13:31 ICT 2015", NULL, NULL, gmtime_r,
510*640235e2SEnji Cooper 		2015, 12, 6, 20, 13, 31);
511*640235e2SEnji Cooper 
512*640235e2SEnji Cooper 
513*640235e2SEnji Cooper 	parsecheck("2015-12-06 12:00:00 IDLW", NULL, NULL, gmtime_r,
514*640235e2SEnji Cooper 		2015, 12, 7, 0, 0, 0);
515*640235e2SEnji Cooper 	parsecheck("2015-12-06 12:00:00 IDLE", NULL, NULL, gmtime_r,
516*640235e2SEnji Cooper 		2015, 12, 6, 0, 0, 0);
517*640235e2SEnji Cooper 
518*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:17:33 NFT", NULL, NULL, gmtime_r,
519*640235e2SEnji Cooper 		2015, 12, 7, 0, 47, 33);
520*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:17:33 ACST", NULL, NULL, gmtime_r,
521*640235e2SEnji Cooper 		2015, 12, 6, 11, 47, 33);
522*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:17:33 +0717", NULL, NULL, gmtime_r,
523*640235e2SEnji Cooper 		2015, 12, 6, 14, 0, 33);
524*640235e2SEnji Cooper 
525*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 Z", NULL, NULL, gmtime_r,
526*640235e2SEnji Cooper 		2015, 12, 6, 21, 21, 21);
527*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 A", NULL, NULL, gmtime_r,
528*640235e2SEnji Cooper 		2015, 12, 6, 22, 21, 21);
529*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 G", NULL, NULL, gmtime_r,
530*640235e2SEnji Cooper 		2015, 12, 7, 4, 21, 21);
531*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 M", NULL, NULL, gmtime_r,
532*640235e2SEnji Cooper 		2015, 12, 7, 9, 21, 21);
533*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 N", NULL, NULL, gmtime_r,
534*640235e2SEnji Cooper 		2015, 12, 6, 20, 21, 21);
535*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 T", NULL, NULL, gmtime_r,
536*640235e2SEnji Cooper 		2015, 12, 6, 14, 21, 21);
537*640235e2SEnji Cooper 	parsecheck("2015-12-06 21:21:21 Y", NULL, NULL, gmtime_r,
538*640235e2SEnji Cooper 		2015, 12, 6, 9, 21, 21);
539*640235e2SEnji Cooper 
540*640235e2SEnji Cooper }
541*640235e2SEnji Cooper 
542*640235e2SEnji Cooper ATF_TC(gibberish);
543*640235e2SEnji Cooper 
ATF_TC_HEAD(gibberish,tc)544*640235e2SEnji Cooper ATF_TC_HEAD(gibberish, tc)
545*640235e2SEnji Cooper {
546*640235e2SEnji Cooper 	atf_tc_set_md_var(tc, "descr", "Test (not) parsing nonsense");
547*640235e2SEnji Cooper }
548*640235e2SEnji Cooper 
ATF_TC_BODY(gibberish,tc)549*640235e2SEnji Cooper ATF_TC_BODY(gibberish, tc)
550*640235e2SEnji Cooper {
551*640235e2SEnji Cooper 	errno = 0;
552*640235e2SEnji Cooper 	ATF_CHECK(parsedate("invalid nonsense", NULL, NULL) == (time_t)-1
553*640235e2SEnji Cooper 	    && errno != 0);
554*640235e2SEnji Cooper 	errno = 0;
555*640235e2SEnji Cooper 	ATF_CHECK(parsedate("12th day of Christmas", NULL, NULL) == (time_t)-1
556*640235e2SEnji Cooper 	    && errno != 0);
557*640235e2SEnji Cooper 	errno = 0;
558*640235e2SEnji Cooper 	ATF_CHECK(parsedate("2015-31-07 15:00", NULL, NULL) == (time_t)-1
559*640235e2SEnji Cooper 	    && errno != 0);
560*640235e2SEnji Cooper 	errno = 0;
561*640235e2SEnji Cooper 	ATF_CHECK(parsedate("2015-02-29 10:01", NULL, NULL) == (time_t)-1
562*640235e2SEnji Cooper 	    && errno != 0);
563*640235e2SEnji Cooper 	errno = 0;
564*640235e2SEnji Cooper 	ATF_CHECK(parsedate("2015-12-06 24:01", NULL, NULL) == (time_t)-1
565*640235e2SEnji Cooper 	    && errno != 0);
566*640235e2SEnji Cooper 	errno = 0;
567*640235e2SEnji Cooper 	ATF_CHECK(parsedate("2015-12-06 14:61", NULL, NULL) == (time_t)-1
568*640235e2SEnji Cooper 	    && errno != 0);
569*640235e2SEnji Cooper }
570*640235e2SEnji Cooper 
ATF_TP_ADD_TCS(tp)57157718be8SEnji Cooper ATF_TP_ADD_TCS(tp)
57257718be8SEnji Cooper {
57357718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, dates);
57457718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, times);
575*640235e2SEnji Cooper 	ATF_TP_ADD_TC(tp, dsttimes);
57657718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, relative);
57757718be8SEnji Cooper 	ATF_TP_ADD_TC(tp, atsecs);
578*640235e2SEnji Cooper 	ATF_TP_ADD_TC(tp, zones);
579*640235e2SEnji Cooper 	ATF_TP_ADD_TC(tp, gibberish);
58057718be8SEnji Cooper 
58157718be8SEnji Cooper 	return atf_no_error();
58257718be8SEnji Cooper }
583*640235e2SEnji Cooper 
584