1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /************************************************************************
4  * COPYRIGHT:
5  * Copyright (c) 1997-2016, International Business Machines Corporation
6  * and others. All Rights Reserved.
7  ************************************************************************/
8 #include "unicode/utypes.h"
9 
10 #if !UCONFIG_NO_FORMATTING
11 
12 #include "caltest.h"
13 #include "unicode/dtfmtsym.h"
14 #include "unicode/gregocal.h"
15 #include "unicode/localpointer.h"
16 #include "hebrwcal.h"
17 #include "unicode/smpdtfmt.h"
18 #include "unicode/simpletz.h"
19 #include "dbgutil.h"
20 #include "unicode/udat.h"
21 #include "unicode/ustring.h"
22 #include "cstring.h"
23 #include "unicode/localpointer.h"
24 #include "islamcal.h"
25 
26 #define mkcstr(U) u_austrcpy(calloc(8, u_strlen(U) + 1), U)
27 
28 #define TEST_CHECK_STATUS { \
29     if (U_FAILURE(status)) { \
30         if (status == U_MISSING_RESOURCE_ERROR) { \
31             dataerrln("%s:%d: Test failure.  status=%s", __FILE__, __LINE__, u_errorName(status)); \
32         } else { \
33             errln("%s:%d: Test failure.  status=%s", __FILE__, __LINE__, u_errorName(status)); \
34         } return;}}
35 
36 #define TEST_CHECK_STATUS_LOCALE(testlocale) { \
37     if (U_FAILURE(status)) { \
38         if (status == U_MISSING_RESOURCE_ERROR) { \
39             dataerrln("%s:%d: Test failure, locale %s.  status=%s", __FILE__, __LINE__, testlocale, u_errorName(status)); \
40         } else { \
41             errln("%s:%d: Test failure, locale %s.  status=%s", __FILE__, __LINE__, testlocale, u_errorName(status)); \
42         } return;}}
43 
44 #define TEST_ASSERT(expr) {if ((expr)==FALSE) {errln("%s:%d: Test failure \n", __FILE__, __LINE__);};}
45 
46 // *****************************************************************************
47 // class CalendarTest
48 // *****************************************************************************
49 
calToStr(const Calendar & cal)50 UnicodeString CalendarTest::calToStr(const Calendar & cal)
51 {
52   UnicodeString out;
53   UErrorCode status = U_ZERO_ERROR;
54   int i;
55   UDate d;
56   for(i = 0;i<UCAL_FIELD_COUNT;i++) {
57     out += (UnicodeString("") + fieldName((UCalendarDateFields)i) + "=" +  cal.get((UCalendarDateFields)i, status) + UnicodeString(" "));
58   }
59   out += "[" + UnicodeString(cal.getType()) + "]";
60 
61   if(cal.inDaylightTime(status)) {
62     out += UnicodeString(" (in DST), zone=");
63   }
64   else {
65     out += UnicodeString(", zone=");
66   }
67 
68   UnicodeString str2;
69   out += cal.getTimeZone().getDisplayName(str2);
70   d = cal.getTime(status);
71   out += UnicodeString(" :","") + d;
72 
73   return out;
74 }
75 
runIndexedTest(int32_t index,UBool exec,const char * & name,char *)76 void CalendarTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ )
77 {
78     if (exec) logln("TestSuite TestCalendar");
79     switch (index) {
80         case 0:
81             name = "TestDOW943";
82             if (exec) {
83                 logln("TestDOW943---"); logln("");
84                 TestDOW943();
85             }
86             break;
87         case 1:
88             name = "TestClonesUnique908";
89             if (exec) {
90                 logln("TestClonesUnique908---"); logln("");
91                 TestClonesUnique908();
92             }
93             break;
94         case 2:
95             name = "TestGregorianChange768";
96             if (exec) {
97                 logln("TestGregorianChange768---"); logln("");
98                 TestGregorianChange768();
99             }
100             break;
101         case 3:
102             name = "TestDisambiguation765";
103             if (exec) {
104                 logln("TestDisambiguation765---"); logln("");
105                 TestDisambiguation765();
106             }
107             break;
108         case 4:
109             name = "TestGMTvsLocal4064654";
110             if (exec) {
111                 logln("TestGMTvsLocal4064654---"); logln("");
112                 TestGMTvsLocal4064654();
113             }
114             break;
115         case 5:
116             name = "TestAddSetOrder621";
117             if (exec) {
118                 logln("TestAddSetOrder621---"); logln("");
119                 TestAddSetOrder621();
120             }
121             break;
122         case 6:
123             name = "TestAdd520";
124             if (exec) {
125                 logln("TestAdd520---"); logln("");
126                 TestAdd520();
127             }
128             break;
129         case 7:
130             name = "TestFieldSet4781";
131             if (exec) {
132                 logln("TestFieldSet4781---"); logln("");
133                 TestFieldSet4781();
134             }
135             break;
136         case 8:
137             name = "TestSerialize337";
138             if (exec) {
139                 logln("TestSerialize337---"); logln("");
140             //  TestSerialize337();
141             }
142             break;
143         case 9:
144             name = "TestSecondsZero121";
145             if (exec) {
146                 logln("TestSecondsZero121---"); logln("");
147                 TestSecondsZero121();
148             }
149             break;
150         case 10:
151             name = "TestAddSetGet0610";
152             if (exec) {
153                 logln("TestAddSetGet0610---"); logln("");
154                 TestAddSetGet0610();
155             }
156             break;
157         case 11:
158             name = "TestFields060";
159             if (exec) {
160                 logln("TestFields060---"); logln("");
161                 TestFields060();
162             }
163             break;
164         case 12:
165             name = "TestEpochStartFields";
166             if (exec) {
167                 logln("TestEpochStartFields---"); logln("");
168                 TestEpochStartFields();
169             }
170             break;
171         case 13:
172             name = "TestDOWProgression";
173             if (exec) {
174                 logln("TestDOWProgression---"); logln("");
175                 TestDOWProgression();
176             }
177             break;
178         case 14:
179             name = "TestGenericAPI";
180             if (exec) {
181                 logln("TestGenericAPI---"); logln("");
182                 TestGenericAPI();
183             }
184             break;
185         case 15:
186             name = "TestAddRollExtensive";
187             if (exec) {
188                 logln("TestAddRollExtensive---"); logln("");
189                 TestAddRollExtensive();
190             }
191             break;
192         case 16:
193             name = "TestDOW_LOCALandYEAR_WOY";
194             if (exec) {
195                 logln("TestDOW_LOCALandYEAR_WOY---"); logln("");
196                 TestDOW_LOCALandYEAR_WOY();
197             }
198             break;
199         case 17:
200             name = "TestWOY";
201             if (exec) {
202                 logln("TestWOY---"); logln("");
203                 TestWOY();
204             }
205             break;
206         case 18:
207             name = "TestRog";
208             if (exec) {
209                 logln("TestRog---"); logln("");
210                 TestRog();
211             }
212             break;
213         case 19:
214            name = "TestYWOY";
215             if (exec) {
216                 logln("TestYWOY---"); logln("");
217                 TestYWOY();
218             }
219             break;
220         case 20:
221           name = "TestJD";
222           if(exec) {
223             logln("TestJD---"); logln("");
224             TestJD();
225           }
226           break;
227         case 21:
228           name = "TestDebug";
229           if(exec) {
230             logln("TestDebug---"); logln("");
231             TestDebug();
232           }
233           break;
234         case 22:
235           name = "Test6703";
236           if(exec) {
237             logln("Test6703---"); logln("");
238             Test6703();
239           }
240           break;
241         case 23:
242           name = "Test3785";
243           if(exec) {
244             logln("Test3785---"); logln("");
245             Test3785();
246           }
247           break;
248         case 24:
249           name = "Test1624";
250           if(exec) {
251             logln("Test1624---"); logln("");
252             Test1624();
253           }
254           break;
255         case 25:
256           name = "TestTimeStamp";
257           if(exec) {
258             logln("TestTimeStamp---"); logln("");
259             TestTimeStamp();
260           }
261           break;
262         case 26:
263           name = "TestISO8601";
264           if(exec) {
265             logln("TestISO8601---"); logln("");
266             TestISO8601();
267           }
268           break;
269         case 27:
270           name = "TestAmbiguousWallTimeAPIs";
271           if(exec) {
272             logln("TestAmbiguousWallTimeAPIs---"); logln("");
273             TestAmbiguousWallTimeAPIs();
274           }
275           break;
276         case 28:
277           name = "TestRepeatedWallTime";
278           if(exec) {
279             logln("TestRepeatedWallTime---"); logln("");
280             TestRepeatedWallTime();
281           }
282           break;
283         case 29:
284           name = "TestSkippedWallTime";
285           if(exec) {
286             logln("TestSkippedWallTime---"); logln("");
287             TestSkippedWallTime();
288           }
289           break;
290         case 30:
291           name = "TestCloneLocale";
292           if(exec) {
293             logln("TestCloneLocale---"); logln("");
294             TestCloneLocale();
295           }
296           break;
297         case 31:
298           name = "TestIslamicUmAlQura";
299           if(exec) {
300             logln("TestIslamicUmAlQura---"); logln("");
301             TestIslamicUmAlQura();
302           }
303           break;
304         case 32:
305           name = "TestIslamicTabularDates";
306           if(exec) {
307             logln("TestIslamicTabularDates---"); logln("");
308             TestIslamicTabularDates();
309           }
310           break;
311         case 33:
312           name = "TestHebrewMonthValidation";
313           if(exec) {
314             logln("TestHebrewMonthValidation---"); logln("");
315             TestHebrewMonthValidation();
316           }
317           break;
318         case 34:
319           name = "TestWeekData";
320           if(exec) {
321             logln("TestWeekData---"); logln("");
322             TestWeekData();
323           }
324           break;
325         case 35:
326           name = "TestAddAcrossZoneTransition";
327           if(exec) {
328             logln("TestAddAcrossZoneTransition---"); logln("");
329             TestAddAcrossZoneTransition();
330           }
331           break;
332         case 36:
333           name = "TestChineseCalendarMapping";
334           if(exec) {
335             logln("TestChineseCalendarMapping---"); logln("");
336             TestChineseCalendarMapping();
337           }
338           break;
339         default: name = ""; break;
340     }
341 }
342 
343 // ---------------------------------------------------------------------------------
344 
fieldName(UCalendarDateFields f)345 UnicodeString CalendarTest::fieldName(UCalendarDateFields f) {
346     switch (f) {
347 #define FIELD_NAME_STR(x) case x: return (#x+5)
348       FIELD_NAME_STR( UCAL_ERA );
349       FIELD_NAME_STR( UCAL_YEAR );
350       FIELD_NAME_STR( UCAL_MONTH );
351       FIELD_NAME_STR( UCAL_WEEK_OF_YEAR );
352       FIELD_NAME_STR( UCAL_WEEK_OF_MONTH );
353       FIELD_NAME_STR( UCAL_DATE );
354       FIELD_NAME_STR( UCAL_DAY_OF_YEAR );
355       FIELD_NAME_STR( UCAL_DAY_OF_WEEK );
356       FIELD_NAME_STR( UCAL_DAY_OF_WEEK_IN_MONTH );
357       FIELD_NAME_STR( UCAL_AM_PM );
358       FIELD_NAME_STR( UCAL_HOUR );
359       FIELD_NAME_STR( UCAL_HOUR_OF_DAY );
360       FIELD_NAME_STR( UCAL_MINUTE );
361       FIELD_NAME_STR( UCAL_SECOND );
362       FIELD_NAME_STR( UCAL_MILLISECOND );
363       FIELD_NAME_STR( UCAL_ZONE_OFFSET );
364       FIELD_NAME_STR( UCAL_DST_OFFSET );
365       FIELD_NAME_STR( UCAL_YEAR_WOY );
366       FIELD_NAME_STR( UCAL_DOW_LOCAL );
367       FIELD_NAME_STR( UCAL_EXTENDED_YEAR );
368       FIELD_NAME_STR( UCAL_JULIAN_DAY );
369       FIELD_NAME_STR( UCAL_MILLISECONDS_IN_DAY );
370 #undef FIELD_NAME_STR
371     default:
372         return UnicodeString("") + ((int32_t)f);
373     }
374 }
375 
376 /**
377  * Test various API methods for API completeness.
378  */
379 void
TestGenericAPI()380 CalendarTest::TestGenericAPI()
381 {
382     UErrorCode status = U_ZERO_ERROR;
383     UDate d;
384     UnicodeString str;
385     UBool eq = FALSE,b4 = FALSE,af = FALSE;
386 
387     UDate when = date(90, UCAL_APRIL, 15);
388 
389     UnicodeString tzid("TestZone");
390     int32_t tzoffset = 123400;
391 
392     SimpleTimeZone *zone = new SimpleTimeZone(tzoffset, tzid);
393     Calendar *cal = Calendar::createInstance(zone->clone(), status);
394     if (failure(status, "Calendar::createInstance #1", TRUE)) return;
395 
396     if (*zone != cal->getTimeZone()) errln("FAIL: Calendar::getTimeZone failed");
397 
398     Calendar *cal2 = Calendar::createInstance(cal->getTimeZone(), status);
399     if (failure(status, "Calendar::createInstance #2")) return;
400     cal->setTime(when, status);
401     cal2->setTime(when, status);
402     if (failure(status, "Calendar::setTime")) return;
403 
404     if (!(*cal == *cal2)) errln("FAIL: Calendar::operator== failed");
405     if ((*cal != *cal2))  errln("FAIL: Calendar::operator!= failed");
406     if (!cal->equals(*cal2, status) ||
407         cal->before(*cal2, status) ||
408         cal->after(*cal2, status) ||
409         U_FAILURE(status)) errln("FAIL: equals/before/after failed");
410 
411     logln(UnicodeString("cal=")  +cal->getTime(status)  + UnicodeString(calToStr(*cal)));
412     logln(UnicodeString("cal2=")  +cal2->getTime(status)  + UnicodeString(calToStr(*cal2)));
413     logln("cal2->setTime(when+1000)");
414     cal2->setTime(when + 1000, status);
415     logln(UnicodeString("cal2=")  +cal2->getTime(status)  + UnicodeString(calToStr(*cal2)));
416 
417     if (failure(status, "Calendar::setTime")) return;
418     if (cal->equals(*cal2, status) ||
419         cal2->before(*cal, status) ||
420         cal->after(*cal2, status) ||
421         U_FAILURE(status)) errln("FAIL: equals/before/after failed after setTime(+1000)");
422 
423     logln("cal->roll(UCAL_SECOND)");
424     cal->roll(UCAL_SECOND, (UBool) TRUE, status);
425     logln(UnicodeString("cal=")  +cal->getTime(status)  + UnicodeString(calToStr(*cal)));
426     cal->roll(UCAL_SECOND, (int32_t)0, status);
427     logln(UnicodeString("cal=")  +cal->getTime(status)  + UnicodeString(calToStr(*cal)));
428     if (failure(status, "Calendar::roll")) return;
429 
430     if (!(eq=cal->equals(*cal2, status)) ||
431         (b4=cal->before(*cal2, status)) ||
432         (af=cal->after(*cal2, status)) ||
433         U_FAILURE(status)) {
434       errln("FAIL: equals[%c]/before[%c]/after[%c] failed after roll 1 second [should be T/F/F]",
435             eq?'T':'F',
436             b4?'T':'F',
437             af?'T':'F');
438       logln(UnicodeString("cal=")  +cal->getTime(status)  + UnicodeString(calToStr(*cal)));
439       logln(UnicodeString("cal2=")  +cal2->getTime(status)  + UnicodeString(calToStr(*cal2)));
440     }
441 
442     // Roll back to January
443     cal->roll(UCAL_MONTH, (int32_t)(1 + UCAL_DECEMBER - cal->get(UCAL_MONTH, status)), status);
444     if (failure(status, "Calendar::roll")) return;
445     if (cal->equals(*cal2, status) ||
446         cal2->before(*cal, status) ||
447         cal->after(*cal2, status) ||
448         U_FAILURE(status)) errln("FAIL: equals/before/after failed after rollback to January");
449 
450     TimeZone *z = cal->orphanTimeZone();
451     if (z->getID(str) != tzid ||
452         z->getRawOffset() != tzoffset)
453         errln("FAIL: orphanTimeZone failed");
454 
455     int32_t i;
456     for (i=0; i<2; ++i)
457     {
458         UBool lenient = ( i > 0 );
459         cal->setLenient(lenient);
460         if (lenient != cal->isLenient()) errln("FAIL: setLenient/isLenient failed");
461         // Later: Check for lenient behavior
462     }
463 
464     for (i=UCAL_SUNDAY; i<=UCAL_SATURDAY; ++i)
465     {
466         cal->setFirstDayOfWeek((UCalendarDaysOfWeek)i);
467         if (cal->getFirstDayOfWeek() != i) errln("FAIL: set/getFirstDayOfWeek failed");
468         UErrorCode aStatus = U_ZERO_ERROR;
469         if (cal->getFirstDayOfWeek(aStatus) != i || U_FAILURE(aStatus)) errln("FAIL: getFirstDayOfWeek(status) failed");
470     }
471 
472     for (i=1; i<=7; ++i)
473     {
474         cal->setMinimalDaysInFirstWeek((uint8_t)i);
475         if (cal->getMinimalDaysInFirstWeek() != i) errln("FAIL: set/getFirstDayOfWeek failed");
476     }
477 
478     for (i=0; i<UCAL_FIELD_COUNT; ++i)
479     {
480         if (cal->getMinimum((UCalendarDateFields)i) > cal->getGreatestMinimum((UCalendarDateFields)i))
481             errln(UnicodeString("FAIL: getMinimum larger than getGreatestMinimum for field ") + i);
482         if (cal->getLeastMaximum((UCalendarDateFields)i) > cal->getMaximum((UCalendarDateFields)i))
483             errln(UnicodeString("FAIL: getLeastMaximum larger than getMaximum for field ") + i);
484         if (cal->getMinimum((UCalendarDateFields)i) >= cal->getMaximum((UCalendarDateFields)i))
485             errln(UnicodeString("FAIL: getMinimum not less than getMaximum for field ") + i);
486     }
487 
488     cal->adoptTimeZone(TimeZone::createDefault());
489     cal->clear();
490     cal->set(1984, 5, 24);
491     if (cal->getTime(status) != date(84, 5, 24) || U_FAILURE(status))
492         errln("FAIL: Calendar::set(3 args) failed");
493 
494     cal->clear();
495     cal->set(1985, 3, 2, 11, 49);
496     if (cal->getTime(status) != date(85, 3, 2, 11, 49) || U_FAILURE(status))
497         errln("FAIL: Calendar::set(5 args) failed");
498 
499     cal->clear();
500     cal->set(1995, 9, 12, 1, 39, 55);
501     if (cal->getTime(status) != date(95, 9, 12, 1, 39, 55) || U_FAILURE(status))
502         errln("FAIL: Calendar::set(6 args) failed");
503 
504     cal->getTime(status);
505     if (failure(status, "Calendar::getTime")) return;
506     for (i=0; i<UCAL_FIELD_COUNT; ++i)
507     {
508         switch(i) {
509             case UCAL_YEAR: case UCAL_MONTH: case UCAL_DATE:
510             case UCAL_HOUR_OF_DAY: case UCAL_MINUTE: case UCAL_SECOND:
511             case UCAL_EXTENDED_YEAR:
512               if (!cal->isSet((UCalendarDateFields)i)) errln("FAIL: Calendar::isSet F, should be T " + fieldName((UCalendarDateFields)i));
513                 break;
514             default:
515               if (cal->isSet((UCalendarDateFields)i)) errln("FAIL: Calendar::isSet = T, should be F  " + fieldName((UCalendarDateFields)i));
516         }
517         cal->clear((UCalendarDateFields)i);
518         if (cal->isSet((UCalendarDateFields)i)) errln("FAIL: Calendar::clear/isSet failed " + fieldName((UCalendarDateFields)i));
519     }
520 
521     if(cal->getActualMinimum(Calendar::SECOND, status) != 0){
522         errln("Calendar is suppose to return 0 for getActualMinimum");
523     }
524 
525     Calendar *cal3 = Calendar::createInstance(status);
526     cal3->roll(Calendar::SECOND, (int32_t)0, status);
527     if (failure(status, "Calendar::roll(EDateFields, int32_t, UErrorCode)")) return;
528 
529     delete cal;
530     delete cal2;
531     delete cal3;
532 
533     int32_t count;
534     const Locale* loc = Calendar::getAvailableLocales(count);
535     if (count < 1 || loc == 0)
536     {
537         dataerrln("FAIL: getAvailableLocales failed");
538     }
539     else
540     {
541         for (i=0; i<count; ++i)
542         {
543             cal = Calendar::createInstance(loc[i], status);
544             if (U_FAILURE(status)) {
545                 errcheckln(status, UnicodeString("FAIL: Calendar::createInstance #3, locale ") +  loc[i].getName() + " , error " + u_errorName(status));
546                 return;
547             }
548             delete cal;
549         }
550     }
551 
552     cal = Calendar::createInstance(TimeZone::createDefault(), Locale::getEnglish(), status);
553     if (failure(status, "Calendar::createInstance #4")) return;
554     delete cal;
555 
556     cal = Calendar::createInstance(*zone, Locale::getEnglish(), status);
557     if (failure(status, "Calendar::createInstance #5")) return;
558     delete cal;
559 
560     GregorianCalendar *gc = new GregorianCalendar(*zone, status);
561     if (failure(status, "new GregorianCalendar")) return;
562     delete gc;
563 
564     gc = new GregorianCalendar(Locale::getEnglish(), status);
565     if (failure(status, "new GregorianCalendar")) return;
566     delete gc;
567 
568     gc = new GregorianCalendar(Locale::getEnglish(), status);
569     delete gc;
570 
571     gc = new GregorianCalendar(*zone, Locale::getEnglish(), status);
572     if (failure(status, "new GregorianCalendar")) return;
573     delete gc;
574 
575     gc = new GregorianCalendar(zone, status);
576     if (failure(status, "new GregorianCalendar")) return;
577     delete gc;
578 
579     gc = new GregorianCalendar(1998, 10, 14, 21, 43, status);
580     if (gc->getTime(status) != (d =date(98, 10, 14, 21, 43) )|| U_FAILURE(status))
581       errln("FAIL: new GregorianCalendar(ymdhm) failed with " + UnicodeString(u_errorName(status)) + ",  cal="  + gc->getTime(status)  + UnicodeString(calToStr(*gc)) + ", d=" + d);
582     else
583       logln(UnicodeString("GOOD: cal=")  +gc->getTime(status)  + UnicodeString(calToStr(*gc)) + ", d=" + d);
584     delete gc;
585 
586     gc = new GregorianCalendar(1998, 10, 14, 21, 43, 55, status);
587     if (gc->getTime(status) != (d=date(98, 10, 14, 21, 43, 55)) || U_FAILURE(status))
588       errln("FAIL: new GregorianCalendar(ymdhms) failed with " + UnicodeString(u_errorName(status)));
589 
590     GregorianCalendar gc2(Locale::getEnglish(), status);
591     if (failure(status, "new GregorianCalendar")) return;
592     gc2 = *gc;
593     if (gc2 != *gc || !(gc2 == *gc)) errln("FAIL: GregorianCalendar assignment/operator==/operator!= failed");
594     delete gc;
595     delete z;
596 
597     /* Code coverage for Calendar class. */
598     cal = Calendar::createInstance(status);
599     if (failure(status, "Calendar::createInstance #6")) {
600         return;
601     }else {
602         ((Calendar *)cal)->roll(UCAL_HOUR, (int32_t)100, status);
603         ((Calendar *)cal)->clear(UCAL_HOUR);
604 #if !UCONFIG_NO_SERVICE
605         URegistryKey key = cal->registerFactory(NULL, status);
606         cal->unregister(key, status);
607 #endif
608     }
609     delete cal;
610 
611     status = U_ZERO_ERROR;
612     cal = Calendar::createInstance(Locale("he_IL@calendar=hebrew"), status);
613     if (failure(status, "Calendar::createInstance #7")) {
614         return;
615     } else {
616         cal->roll(Calendar::MONTH, (int32_t)100, status);
617     }
618 
619     LocalPointer<StringEnumeration> values(
620         Calendar::getKeywordValuesForLocale("calendar", Locale("he"), FALSE, status));
621     if (values.isNull() || U_FAILURE(status)) {
622         dataerrln("FAIL: Calendar::getKeywordValuesForLocale(he): %s", u_errorName(status));
623     } else {
624         UBool containsHebrew = FALSE;
625         const char *charValue;
626         int32_t valueLength;
627         while ((charValue = values->next(&valueLength, status)) != NULL) {
628             if (valueLength == 6 && strcmp(charValue, "hebrew") == 0) {
629                 containsHebrew = TRUE;
630             }
631         }
632         if (!containsHebrew) {
633             errln("Calendar::getKeywordValuesForLocale(he)->next() does not contain \"hebrew\"");
634         }
635 
636         values->reset(status);
637         containsHebrew = FALSE;
638         UnicodeString hebrew = UNICODE_STRING_SIMPLE("hebrew");
639         const UChar *ucharValue;
640         while ((ucharValue = values->unext(&valueLength, status)) != NULL) {
641             UnicodeString value(FALSE, ucharValue, valueLength);
642             if (value == hebrew) {
643                 containsHebrew = TRUE;
644             }
645         }
646         if (!containsHebrew) {
647             errln("Calendar::getKeywordValuesForLocale(he)->unext() does not contain \"hebrew\"");
648         }
649 
650         values->reset(status);
651         containsHebrew = FALSE;
652         const UnicodeString *stringValue;
653         while ((stringValue = values->snext(status)) != NULL) {
654             if (*stringValue == hebrew) {
655                 containsHebrew = TRUE;
656             }
657         }
658         if (!containsHebrew) {
659             errln("Calendar::getKeywordValuesForLocale(he)->snext() does not contain \"hebrew\"");
660         }
661     }
662     delete cal;
663 }
664 
665 // -------------------------------------
666 
667 /**
668  * This test confirms the correct behavior of add when incrementing
669  * through subsequent days.
670  */
671 void
TestRog()672 CalendarTest::TestRog()
673 {
674     UErrorCode status = U_ZERO_ERROR;
675     GregorianCalendar* gc = new GregorianCalendar(status);
676     if (failure(status, "new GregorianCalendar", TRUE)) return;
677     int32_t year = 1997, month = UCAL_APRIL, date = 1;
678     gc->set(year, month, date);
679     gc->set(UCAL_HOUR_OF_DAY, 23);
680     gc->set(UCAL_MINUTE, 0);
681     gc->set(UCAL_SECOND, 0);
682     gc->set(UCAL_MILLISECOND, 0);
683     for (int32_t i = 0; i < 9; i++, gc->add(UCAL_DATE, 1, status)) {
684         if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
685         if (gc->get(UCAL_YEAR, status) != year ||
686             gc->get(UCAL_MONTH, status) != month ||
687             gc->get(UCAL_DATE, status) != (date + i)) errln("FAIL: Date wrong");
688         if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
689     }
690     delete gc;
691 }
692 
693 // -------------------------------------
694 
695 /**
696  * Test the handling of the day of the week, checking for correctness and
697  * for correct minimum and maximum values.
698  */
699 void
TestDOW943()700 CalendarTest::TestDOW943()
701 {
702     dowTest(FALSE);
703     dowTest(TRUE);
704 }
705 
dowTest(UBool lenient)706 void CalendarTest::dowTest(UBool lenient)
707 {
708     UErrorCode status = U_ZERO_ERROR;
709     GregorianCalendar* cal = new GregorianCalendar(status);
710     if (failure(status, "new GregorianCalendar", TRUE)) return;
711     logln("cal - Aug 12, 1997\n");
712     cal->set(1997, UCAL_AUGUST, 12);
713     cal->getTime(status);
714     if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
715     logln((lenient?UnicodeString("LENIENT0: "):UnicodeString("nonlenient0: ")) + UnicodeString(calToStr(*cal)));
716     cal->setLenient(lenient);
717     logln("cal - Dec 1, 1996\n");
718     cal->set(1996, UCAL_DECEMBER, 1);
719     logln((lenient?UnicodeString("LENIENT: "):UnicodeString("nonlenient: ")) + UnicodeString(calToStr(*cal)));
720     int32_t dow = cal->get(UCAL_DAY_OF_WEEK, status);
721     if (U_FAILURE(status)) { errln("Calendar::get failed [%s]", u_errorName(status)); return; }
722     int32_t min = cal->getMinimum(UCAL_DAY_OF_WEEK);
723     int32_t max = cal->getMaximum(UCAL_DAY_OF_WEEK);
724     if (dow < min ||
725         dow > max) errln(UnicodeString("FAIL: Day of week ") + (int32_t)dow + " out of range");
726     if (dow != UCAL_SUNDAY) errln("FAIL: Day of week should be SUNDAY[%d] not %d", UCAL_SUNDAY, dow);
727     if (min != UCAL_SUNDAY ||
728         max != UCAL_SATURDAY) errln("FAIL: Min/max bad");
729     delete cal;
730 }
731 
732 // -------------------------------------
733 
734 /**
735  * Confirm that cloned Calendar objects do not inadvertently share substructures.
736  */
737 void
TestClonesUnique908()738 CalendarTest::TestClonesUnique908()
739 {
740     UErrorCode status = U_ZERO_ERROR;
741     Calendar *c = Calendar::createInstance(status);
742     if (failure(status, "Calendar::createInstance", TRUE)) return;
743     Calendar *d = (Calendar*) c->clone();
744     c->set(UCAL_MILLISECOND, 123);
745     d->set(UCAL_MILLISECOND, 456);
746     if (c->get(UCAL_MILLISECOND, status) != 123 ||
747         d->get(UCAL_MILLISECOND, status) != 456) {
748         errln("FAIL: Clones share fields");
749     }
750     if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
751     delete c;
752     delete d;
753 }
754 
755 // -------------------------------------
756 
757 /**
758  * Confirm that the Gregorian cutoff value works as advertised.
759  */
760 void
TestGregorianChange768()761 CalendarTest::TestGregorianChange768()
762 {
763     UBool b;
764     UErrorCode status = U_ZERO_ERROR;
765     UnicodeString str;
766     GregorianCalendar* c = new GregorianCalendar(status);
767     if (failure(status, "new GregorianCalendar", TRUE)) return;
768     logln(UnicodeString("With cutoff ") + dateToString(c->getGregorianChange(), str));
769     b = c->isLeapYear(1800);
770     logln(UnicodeString(" isLeapYear(1800) = ") + (b ? "true" : "false"));
771     logln(UnicodeString(" (should be FALSE)"));
772     if (b) errln("FAIL");
773     c->setGregorianChange(date(0, 0, 1), status);
774     if (U_FAILURE(status)) { errln("GregorianCalendar::setGregorianChange failed"); return; }
775     logln(UnicodeString("With cutoff ") + dateToString(c->getGregorianChange(), str));
776     b = c->isLeapYear(1800);
777     logln(UnicodeString(" isLeapYear(1800) = ") + (b ? "true" : "false"));
778     logln(UnicodeString(" (should be TRUE)"));
779     if (!b) errln("FAIL");
780     delete c;
781 }
782 
783 // -------------------------------------
784 
785 /**
786  * Confirm the functioning of the field disambiguation algorithm.
787  */
788 void
TestDisambiguation765()789 CalendarTest::TestDisambiguation765()
790 {
791     UErrorCode status = U_ZERO_ERROR;
792     Calendar *c = Calendar::createInstance("en_US", status);
793     if (failure(status, "Calendar::createInstance", TRUE)) return;
794     c->setLenient(FALSE);
795     c->clear();
796     c->set(UCAL_YEAR, 1997);
797     c->set(UCAL_MONTH, UCAL_JUNE);
798     c->set(UCAL_DATE, 3);
799     verify765("1997 third day of June = ", c, 1997, UCAL_JUNE, 3);
800     c->clear();
801     c->set(UCAL_YEAR, 1997);
802     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
803     c->set(UCAL_MONTH, UCAL_JUNE);
804     c->set(UCAL_DAY_OF_WEEK_IN_MONTH, 1);
805     verify765("1997 first Tuesday in June = ", c, 1997, UCAL_JUNE, 3);
806     c->clear();
807     c->set(UCAL_YEAR, 1997);
808     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
809     c->set(UCAL_MONTH, UCAL_JUNE);
810     c->set(UCAL_DAY_OF_WEEK_IN_MONTH, - 1);
811     verify765("1997 last Tuesday in June = ", c, 1997, UCAL_JUNE, 24);
812 
813     status = U_ZERO_ERROR;
814     c->clear();
815     c->set(UCAL_YEAR, 1997);
816     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
817     c->set(UCAL_MONTH, UCAL_JUNE);
818     c->set(UCAL_DAY_OF_WEEK_IN_MONTH, 0);
819     c->getTime(status);
820     verify765("1997 zero-th Tuesday in June = ", status);
821 
822     c->clear();
823     c->set(UCAL_YEAR, 1997);
824     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
825     c->set(UCAL_MONTH, UCAL_JUNE);
826     c->set(UCAL_WEEK_OF_MONTH, 1);
827     verify765("1997 Tuesday in week 1 of June = ", c, 1997, UCAL_JUNE, 3);
828     c->clear();
829     c->set(UCAL_YEAR, 1997);
830     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
831     c->set(UCAL_MONTH, UCAL_JUNE);
832     c->set(UCAL_WEEK_OF_MONTH, 5);
833     verify765("1997 Tuesday in week 5 of June = ", c, 1997, UCAL_JULY, 1);
834 
835     status = U_ZERO_ERROR;
836     c->clear();
837     c->set(UCAL_YEAR, 1997);
838     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
839     c->set(UCAL_MONTH, UCAL_JUNE);
840     c->set(UCAL_WEEK_OF_MONTH, 0);
841     c->setMinimalDaysInFirstWeek(1);
842     c->getTime(status);
843     verify765("1997 Tuesday in week 0 of June = ", status);
844 
845     /* Note: The following test used to expect YEAR 1997, WOY 1 to
846      * resolve to a date in Dec 1996; that is, to behave as if
847      * YEAR_WOY were 1997.  With the addition of a new explicit
848      * YEAR_WOY field, YEAR_WOY must itself be set if that is what is
849      * desired.  Using YEAR in combination with WOY is ambiguous, and
850      * results in the first WOY/DOW day of the year satisfying the
851      * given fields (there may be up to two such days). In this case,
852      * it propertly resolves to Tue Dec 30 1997, which has a WOY value
853      * of 1 (for YEAR_WOY 1998) and a DOW of Tuesday, and falls in the
854      * _calendar_ year 1997, as specified. - aliu */
855     c->clear();
856     c->set(UCAL_YEAR_WOY, 1997); // aliu
857     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
858     c->set(UCAL_WEEK_OF_YEAR, 1);
859     verify765("1997 Tuesday in week 1 of yearWOY = ", c, 1996, UCAL_DECEMBER, 31);
860     c->clear(); // - add test for YEAR
861     c->setMinimalDaysInFirstWeek(1);
862     c->set(UCAL_YEAR, 1997);
863     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
864     c->set(UCAL_WEEK_OF_YEAR, 1);
865     verify765("1997 Tuesday in week 1 of year = ", c, 1997, UCAL_DECEMBER, 30);
866     c->clear();
867     c->set(UCAL_YEAR, 1997);
868     c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
869     c->set(UCAL_WEEK_OF_YEAR, 10);
870     verify765("1997 Tuesday in week 10 of year = ", c, 1997, UCAL_MARCH, 4);
871     //try {
872 
873     // {sfb} week 0 is no longer a valid week of year
874     /*c->clear();
875     c->set(Calendar::YEAR, 1997);
876     c->set(Calendar::DAY_OF_WEEK, Calendar::TUESDAY);
877     //c->set(Calendar::WEEK_OF_YEAR, 0);
878     c->set(Calendar::WEEK_OF_YEAR, 1);
879     verify765("1997 Tuesday in week 0 of year = ", c, 1996, Calendar::DECEMBER, 24);*/
880 
881     //}
882     //catch(IllegalArgumentException ex) {
883     //    errln("FAIL: Exception seen:");
884     //    ex.printStackTrace(log);
885     //}
886     delete c;
887 }
888 
889 // -------------------------------------
890 
891 void
verify765(const UnicodeString & msg,Calendar * c,int32_t year,int32_t month,int32_t day)892 CalendarTest::verify765(const UnicodeString& msg, Calendar* c, int32_t year, int32_t month, int32_t day)
893 {
894     UnicodeString str;
895     UErrorCode status = U_ZERO_ERROR;
896     int32_t y = c->get(UCAL_YEAR, status);
897     int32_t m = c->get(UCAL_MONTH, status);
898     int32_t d = c->get(UCAL_DATE, status);
899     if ( y == year &&
900          m == month &&
901          d == day) {
902         if (U_FAILURE(status)) { errln("FAIL: Calendar::get failed"); return; }
903         logln("PASS: " + msg + dateToString(c->getTime(status), str));
904         if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
905     }
906     else {
907         errln("FAIL: " + msg + dateToString(c->getTime(status), str) + "; expected " + (int32_t)year + "/" + (int32_t)(month + 1) + "/" + (int32_t)day +
908             "; got " + (int32_t)y + "/" + (int32_t)(m + 1) + "/" + (int32_t)d + " for Locale: " + c->getLocaleID(ULOC_ACTUAL_LOCALE,status));
909         if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
910     }
911 }
912 
913 // -------------------------------------
914 
915 void
verify765(const UnicodeString & msg,UErrorCode status)916 CalendarTest::verify765(const UnicodeString& msg/*, IllegalArgumentException e*/, UErrorCode status)
917 {
918     if (status != U_ILLEGAL_ARGUMENT_ERROR) errln("FAIL: No IllegalArgumentException for " + msg);
919     else logln("PASS: " + msg + "IllegalArgument as expected");
920 }
921 
922 // -------------------------------------
923 
924 /**
925  * Confirm that the offset between local time and GMT behaves as expected.
926  */
927 void
TestGMTvsLocal4064654()928 CalendarTest::TestGMTvsLocal4064654()
929 {
930     test4064654(1997, 1, 1, 12, 0, 0);
931     test4064654(1997, 4, 16, 18, 30, 0);
932 }
933 
934 // -------------------------------------
935 
936 void
test4064654(int32_t yr,int32_t mo,int32_t dt,int32_t hr,int32_t mn,int32_t sc)937 CalendarTest::test4064654(int32_t yr, int32_t mo, int32_t dt, int32_t hr, int32_t mn, int32_t sc)
938 {
939     UDate date;
940     UErrorCode status = U_ZERO_ERROR;
941     UnicodeString str;
942     Calendar *gmtcal = Calendar::createInstance(status);
943     if (failure(status, "Calendar::createInstance", TRUE)) return;
944     gmtcal->adoptTimeZone(TimeZone::createTimeZone("Africa/Casablanca"));
945     gmtcal->set(yr, mo - 1, dt, hr, mn, sc);
946     gmtcal->set(UCAL_MILLISECOND, 0);
947     date = gmtcal->getTime(status);
948     if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
949     logln("date = " + dateToString(date, str));
950     Calendar *cal = Calendar::createInstance(status);
951     if (U_FAILURE(status)) { errln("Calendar::createInstance failed"); return; }
952     cal->setTime(date, status);
953     if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
954     int32_t offset = cal->getTimeZone().getOffset((uint8_t)cal->get(UCAL_ERA, status),
955                                                   cal->get(UCAL_YEAR, status),
956                                                   cal->get(UCAL_MONTH, status),
957                                                   cal->get(UCAL_DATE, status),
958                                                   (uint8_t)cal->get(UCAL_DAY_OF_WEEK, status),
959                                                   cal->get(UCAL_MILLISECOND, status), status);
960     if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
961     logln("offset for " + dateToString(date, str) + "= " + (offset / 1000 / 60 / 60.0) + "hr");
962     int32_t utc = ((cal->get(UCAL_HOUR_OF_DAY, status) * 60 +
963                     cal->get(UCAL_MINUTE, status)) * 60 +
964                    cal->get(UCAL_SECOND, status)) * 1000 +
965         cal->get(UCAL_MILLISECOND, status) - offset;
966     if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
967     int32_t expected = ((hr * 60 + mn) * 60 + sc) * 1000;
968     if (utc != expected) errln(UnicodeString("FAIL: Discrepancy of ") + (utc - expected) +
969                                " millis = " + ((utc - expected) / 1000 / 60 / 60.0) + " hr");
970     delete gmtcal;
971     delete cal;
972 }
973 
974 // -------------------------------------
975 
976 /**
977  * The operations of adding and setting should not exhibit pathological
978  * dependence on the order of operations.  This test checks for this.
979  */
980 void
TestAddSetOrder621()981 CalendarTest::TestAddSetOrder621()
982 {
983     UDate d = date(97, 4, 14, 13, 23, 45);
984     UErrorCode status = U_ZERO_ERROR;
985     Calendar *cal = Calendar::createInstance(status);
986     if (failure(status, "Calendar::createInstance", TRUE)) return;
987 
988     cal->setTime(d, status);
989     if (U_FAILURE(status)) {
990         errln("Calendar::setTime failed");
991         delete cal;
992         return;
993     }
994     cal->add(UCAL_DATE, - 5, status);
995     if (U_FAILURE(status)) {
996         errln("Calendar::add failed");
997         delete cal;
998         return;
999     }
1000     cal->set(UCAL_HOUR_OF_DAY, 0);
1001     cal->set(UCAL_MINUTE, 0);
1002     cal->set(UCAL_SECOND, 0);
1003     UnicodeString s;
1004     dateToString(cal->getTime(status), s);
1005     if (U_FAILURE(status)) {
1006         errln("Calendar::getTime failed");
1007         delete cal;
1008         return;
1009     }
1010     delete cal;
1011 
1012     cal = Calendar::createInstance(status);
1013     if (U_FAILURE(status)) {
1014         errln("Calendar::createInstance failed");
1015         delete cal;
1016         return;
1017     }
1018     cal->setTime(d, status);
1019     if (U_FAILURE(status)) {
1020         errln("Calendar::setTime failed");
1021         delete cal;
1022         return;
1023     }
1024     cal->set(UCAL_HOUR_OF_DAY, 0);
1025     cal->set(UCAL_MINUTE, 0);
1026     cal->set(UCAL_SECOND, 0);
1027     cal->add(UCAL_DATE, - 5, status);
1028     if (U_FAILURE(status)) {
1029         errln("Calendar::add failed");
1030         delete cal;
1031         return;
1032     }
1033     UnicodeString s2;
1034     dateToString(cal->getTime(status), s2);
1035     if (U_FAILURE(status)) {
1036         errln("Calendar::getTime failed");
1037         delete cal;
1038         return;
1039     }
1040     if (s == s2)
1041         logln("Pass: " + s + " == " + s2);
1042     else
1043         errln("FAIL: " + s + " != " + s2);
1044     delete cal;
1045 }
1046 
1047 // -------------------------------------
1048 
1049 /**
1050  * Confirm that adding to various fields works.
1051  */
1052 void
TestAdd520()1053 CalendarTest::TestAdd520()
1054 {
1055     int32_t y = 1997, m = UCAL_FEBRUARY, d = 1;
1056     UErrorCode status = U_ZERO_ERROR;
1057     GregorianCalendar *temp = new GregorianCalendar(y, m, d, status);
1058     if (failure(status, "new GregorianCalendar", TRUE)) return;
1059     check520(temp, y, m, d);
1060     temp->add(UCAL_YEAR, 1, status);
1061     if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1062     y++;
1063     check520(temp, y, m, d);
1064     temp->add(UCAL_MONTH, 1, status);
1065     if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1066     m++;
1067     check520(temp, y, m, d);
1068     temp->add(UCAL_DATE, 1, status);
1069     if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1070     d++;
1071     check520(temp, y, m, d);
1072     temp->add(UCAL_DATE, 2, status);
1073     if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1074     d += 2;
1075     check520(temp, y, m, d);
1076     temp->add(UCAL_DATE, 28, status);
1077     if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1078     d = 1;++m;
1079     check520(temp, y, m, d);
1080     delete temp;
1081 }
1082 
1083 // -------------------------------------
1084 
1085 /**
1086  * Execute adding and rolling in GregorianCalendar extensively,
1087  */
1088 void
TestAddRollExtensive()1089 CalendarTest::TestAddRollExtensive()
1090 {
1091     int32_t maxlimit = 40;
1092     int32_t y = 1997, m = UCAL_FEBRUARY, d = 1, hr = 1, min = 1, sec = 0, ms = 0;
1093     UErrorCode status = U_ZERO_ERROR;
1094     GregorianCalendar *temp = new GregorianCalendar(y, m, d, status);
1095     if (failure(status, "new GregorianCalendar", TRUE)) return;
1096 
1097     temp->set(UCAL_HOUR, hr);
1098     temp->set(UCAL_MINUTE, min);
1099     temp->set(UCAL_SECOND, sec);
1100     temp->set(UCAL_MILLISECOND, ms);
1101     temp->setMinimalDaysInFirstWeek(1);
1102 
1103     UCalendarDateFields e;
1104 
1105     logln("Testing GregorianCalendar add...");
1106     e = UCAL_YEAR;
1107     while (e < UCAL_FIELD_COUNT) {
1108         int32_t i;
1109         int32_t limit = maxlimit;
1110         status = U_ZERO_ERROR;
1111         for (i = 0; i < limit; i++) {
1112             temp->add(e, 1, status);
1113             if (U_FAILURE(status)) { limit = i; status = U_ZERO_ERROR; }
1114         }
1115         for (i = 0; i < limit; i++) {
1116             temp->add(e, -1, status);
1117             if (U_FAILURE(status)) { errln("GregorianCalendar::add -1 failed"); return; }
1118         }
1119         check520(temp, y, m, d, hr, min, sec, ms, e);
1120 
1121         e = (UCalendarDateFields) ((int32_t) e + 1);
1122     }
1123 
1124     logln("Testing GregorianCalendar roll...");
1125     e = UCAL_YEAR;
1126     while (e < UCAL_FIELD_COUNT) {
1127         int32_t i;
1128         int32_t limit = maxlimit;
1129         status = U_ZERO_ERROR;
1130         for (i = 0; i < limit; i++) {
1131             logln(calToStr(*temp) + UnicodeString("  " ) + fieldName(e) + UnicodeString("++") );
1132             temp->roll(e, 1, status);
1133             if (U_FAILURE(status)) {
1134               logln("caltest.cpp:%d e=%d, i=%d - roll(+) err %s\n",  __LINE__, (int) e, (int) i, u_errorName(status));
1135               logln(calToStr(*temp));
1136               limit = i; status = U_ZERO_ERROR;
1137             }
1138         }
1139         for (i = 0; i < limit; i++) {
1140             logln("caltest.cpp:%d e=%d, i=%d\n",  __LINE__, (int) e, (int) i);
1141             logln(calToStr(*temp) + UnicodeString("  " ) + fieldName(e) + UnicodeString("--") );
1142             temp->roll(e, -1, status);
1143             if (U_FAILURE(status)) { errln(UnicodeString("GregorianCalendar::roll ") + CalendarTest::fieldName(e) + " count=" + UnicodeString('@'+i) + " by -1 failed with " + u_errorName(status) ); return; }
1144         }
1145         check520(temp, y, m, d, hr, min, sec, ms, e);
1146 
1147         e = (UCalendarDateFields) ((int32_t) e + 1);
1148     }
1149 
1150     delete temp;
1151 }
1152 
1153 // -------------------------------------
1154 void
check520(Calendar * c,int32_t y,int32_t m,int32_t d,int32_t hr,int32_t min,int32_t sec,int32_t ms,UCalendarDateFields field)1155 CalendarTest::check520(Calendar* c,
1156                         int32_t y, int32_t m, int32_t d,
1157                         int32_t hr, int32_t min, int32_t sec,
1158                         int32_t ms, UCalendarDateFields field)
1159 
1160 {
1161     UErrorCode status = U_ZERO_ERROR;
1162     if (c->get(UCAL_YEAR, status) != y ||
1163         c->get(UCAL_MONTH, status) != m ||
1164         c->get(UCAL_DATE, status) != d ||
1165         c->get(UCAL_HOUR, status) != hr ||
1166         c->get(UCAL_MINUTE, status) != min ||
1167         c->get(UCAL_SECOND, status) != sec ||
1168         c->get(UCAL_MILLISECOND, status) != ms) {
1169         errln(UnicodeString("U_FAILURE for field ") + (int32_t)field +
1170                 ": Expected y/m/d h:m:s:ms of " +
1171                 y + "/" + (m + 1) + "/" + d + " " +
1172               hr + ":" + min + ":" + sec + ":" + ms +
1173               "; got " + c->get(UCAL_YEAR, status) +
1174               "/" + (c->get(UCAL_MONTH, status) + 1) +
1175               "/" + c->get(UCAL_DATE, status) +
1176               " " + c->get(UCAL_HOUR, status) + ":" +
1177               c->get(UCAL_MINUTE, status) + ":" +
1178               c->get(UCAL_SECOND, status) + ":" +
1179               c->get(UCAL_MILLISECOND, status)
1180               );
1181 
1182         if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1183     }
1184     else
1185         logln(UnicodeString("Confirmed: ") + y + "/" +
1186                 (m + 1) + "/" + d + " " +
1187                 hr + ":" + min + ":" + sec + ":" + ms);
1188 }
1189 
1190 // -------------------------------------
1191 void
check520(Calendar * c,int32_t y,int32_t m,int32_t d)1192 CalendarTest::check520(Calendar* c,
1193                         int32_t y, int32_t m, int32_t d)
1194 
1195 {
1196     UErrorCode status = U_ZERO_ERROR;
1197     if (c->get(UCAL_YEAR, status) != y ||
1198         c->get(UCAL_MONTH, status) != m ||
1199         c->get(UCAL_DATE, status) != d) {
1200         errln(UnicodeString("FAILURE: Expected y/m/d of ") +
1201               y + "/" + (m + 1) + "/" + d + " " +
1202               "; got " + c->get(UCAL_YEAR, status) +
1203               "/" + (c->get(UCAL_MONTH, status) + 1) +
1204               "/" + c->get(UCAL_DATE, status)
1205               );
1206 
1207         if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1208     }
1209     else
1210         logln(UnicodeString("Confirmed: ") + y + "/" +
1211                 (m + 1) + "/" + d);
1212 }
1213 
1214 // -------------------------------------
1215 
1216 /**
1217  * Test that setting of fields works.  In particular, make sure that all instances
1218  * of GregorianCalendar don't share a static instance of the fields array.
1219  */
1220 void
TestFieldSet4781()1221 CalendarTest::TestFieldSet4781()
1222 {
1223     // try {
1224         UErrorCode status = U_ZERO_ERROR;
1225         GregorianCalendar *g = new GregorianCalendar(status);
1226         if (failure(status, "new GregorianCalendar", TRUE)) return;
1227         GregorianCalendar *g2 = new GregorianCalendar(status);
1228         if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
1229         g2->set(UCAL_HOUR, 12, status);
1230         g2->set(UCAL_MINUTE, 0, status);
1231         g2->set(UCAL_SECOND, 0, status);
1232         if (U_FAILURE(status)) { errln("Calendar::set failed"); return; }
1233         if (*g == *g2) logln("Same");
1234         else logln("Different");
1235     //}
1236         //catch(IllegalArgumentException e) {
1237         //errln("Unexpected exception seen: " + e);
1238     //}
1239         delete g;
1240         delete g2;
1241 }
1242 
1243 // -------------------------------------
1244 
1245 /* We don't support serialization on C++
1246 void
1247 CalendarTest::TestSerialize337()
1248 {
1249     Calendar cal = Calendar::getInstance();
1250     UBool ok = FALSE;
1251     try {
1252         FileOutputStream f = new FileOutputStream(FILENAME);
1253         ObjectOutput s = new ObjectOutputStream(f);
1254         s.writeObject(PREFIX);
1255         s.writeObject(cal);
1256         s.writeObject(POSTFIX);
1257         f.close();
1258         FileInputStream in = new FileInputStream(FILENAME);
1259         ObjectInputStream t = new ObjectInputStream(in);
1260         UnicodeString& pre = (UnicodeString&) t.readObject();
1261         Calendar c = (Calendar) t.readObject();
1262         UnicodeString& post = (UnicodeString&) t.readObject();
1263         in.close();
1264         ok = pre.equals(PREFIX) &&
1265             post.equals(POSTFIX) &&
1266             cal->equals(c);
1267         File fl = new File(FILENAME);
1268         fl.delete();
1269     }
1270     catch(IOException e) {
1271         errln("FAIL: Exception received:");
1272         e.printStackTrace(log);
1273     }
1274     catch(ClassNotFoundException e) {
1275         errln("FAIL: Exception received:");
1276         e.printStackTrace(log);
1277     }
1278     if (!ok) errln("Serialization of Calendar object failed.");
1279 }
1280 
1281 UnicodeString& CalendarTest::PREFIX = "abc";
1282 
1283 UnicodeString& CalendarTest::POSTFIX = "def";
1284 
1285 UnicodeString& CalendarTest::FILENAME = "tmp337.bin";
1286  */
1287 
1288 // -------------------------------------
1289 
1290 /**
1291  * Verify that the seconds of a Calendar can be zeroed out through the
1292  * expected sequence of operations.
1293  */
1294 void
TestSecondsZero121()1295 CalendarTest::TestSecondsZero121()
1296 {
1297     UErrorCode status = U_ZERO_ERROR;
1298     Calendar *cal = new GregorianCalendar(status);
1299     if (failure(status, "new GregorianCalendar", TRUE)) return;
1300     cal->setTime(Calendar::getNow(), status);
1301     if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
1302     cal->set(UCAL_SECOND, 0);
1303     if (U_FAILURE(status)) { errln("Calendar::set failed"); return; }
1304     UDate d = cal->getTime(status);
1305     if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
1306     UnicodeString s;
1307     dateToString(d, s);
1308     if (s.indexOf("DATE_FORMAT_FAILURE") >= 0) {
1309         dataerrln("Got: \"DATE_FORMAT_FAILURE\".");
1310     } else if (s.indexOf(":00 ") < 0) {
1311         errln("Expected to see :00 in " + s);
1312     }
1313     delete cal;
1314 }
1315 
1316 // -------------------------------------
1317 
1318 /**
1319  * Verify that a specific sequence of adding and setting works as expected;
1320  * it should not vary depending on when and whether the get method is
1321  * called.
1322  */
1323 void
TestAddSetGet0610()1324 CalendarTest::TestAddSetGet0610()
1325 {
1326     UnicodeString EXPECTED_0610("1993/0/5", "");
1327     UErrorCode status = U_ZERO_ERROR;
1328     {
1329         Calendar *calendar = new GregorianCalendar(status);
1330         if (failure(status, "new GregorianCalendar", TRUE)) return;
1331         calendar->set(1993, UCAL_JANUARY, 4);
1332         logln("1A) " + value(calendar));
1333         calendar->add(UCAL_DATE, 1, status);
1334         if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1335         UnicodeString v = value(calendar);
1336         logln("1B) " + v);
1337         logln("--) 1993/0/5");
1338         if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
1339         delete calendar;
1340     }
1341     {
1342         Calendar *calendar = new GregorianCalendar(1993, UCAL_JANUARY, 4, status);
1343         if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
1344         logln("2A) " + value(calendar));
1345         calendar->add(UCAL_DATE, 1, status);
1346         if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1347         UnicodeString v = value(calendar);
1348         logln("2B) " + v);
1349         logln("--) 1993/0/5");
1350         if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
1351         delete calendar;
1352     }
1353     {
1354         Calendar *calendar = new GregorianCalendar(1993, UCAL_JANUARY, 4, status);
1355         if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
1356         logln("3A) " + value(calendar));
1357         calendar->getTime(status);
1358         if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
1359         calendar->add(UCAL_DATE, 1, status);
1360         if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1361         UnicodeString v = value(calendar);
1362         logln("3B) " + v);
1363         logln("--) 1993/0/5");
1364         if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
1365         delete calendar;
1366     }
1367 }
1368 
1369 // -------------------------------------
1370 
1371 UnicodeString
value(Calendar * calendar)1372 CalendarTest::value(Calendar* calendar)
1373 {
1374     UErrorCode status = U_ZERO_ERROR;
1375     return UnicodeString("") + (int32_t)calendar->get(UCAL_YEAR, status) +
1376         "/" + (int32_t)calendar->get(UCAL_MONTH, status) +
1377         "/" + (int32_t)calendar->get(UCAL_DATE, status) +
1378         (U_FAILURE(status) ? " FAIL: Calendar::get failed" : "");
1379 }
1380 
1381 
1382 // -------------------------------------
1383 
1384 /**
1385  * Verify that various fields on a known date are set correctly.
1386  */
1387 void
TestFields060()1388 CalendarTest::TestFields060()
1389 {
1390     UErrorCode status = U_ZERO_ERROR;
1391     int32_t year = 1997;
1392     int32_t month = UCAL_OCTOBER;
1393     int32_t dDate = 22;
1394     GregorianCalendar *calendar = 0;
1395     calendar = new GregorianCalendar(year, month, dDate, status);
1396     if (failure(status, "new GregorianCalendar", TRUE)) return;
1397     for (int32_t i = 0; i < EXPECTED_FIELDS_length;) {
1398         UCalendarDateFields field = (UCalendarDateFields)EXPECTED_FIELDS[i++];
1399         int32_t expected = EXPECTED_FIELDS[i++];
1400         if (calendar->get(field, status) != expected) {
1401             errln(UnicodeString("Expected field ") + (int32_t)field + " to have value " + (int32_t)expected +
1402                   "; received " + (int32_t)calendar->get(field, status) + " instead");
1403             if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1404         }
1405     }
1406     delete calendar;
1407 }
1408 
1409 int32_t CalendarTest::EXPECTED_FIELDS[] = {
1410     UCAL_YEAR, 1997,
1411     UCAL_MONTH, UCAL_OCTOBER,
1412     UCAL_DATE, 22,
1413     UCAL_DAY_OF_WEEK, UCAL_WEDNESDAY,
1414     UCAL_DAY_OF_WEEK_IN_MONTH, 4,
1415     UCAL_DAY_OF_YEAR, 295
1416 };
1417 
1418 const int32_t CalendarTest::EXPECTED_FIELDS_length = (int32_t)(sizeof(CalendarTest::EXPECTED_FIELDS) /
1419     sizeof(CalendarTest::EXPECTED_FIELDS[0]));
1420 
1421 // -------------------------------------
1422 
1423 /**
1424  * Verify that various fields on a known date are set correctly.  In this
1425  * case, the start of the epoch (January 1 1970).
1426  */
1427 void
TestEpochStartFields()1428 CalendarTest::TestEpochStartFields()
1429 {
1430     UErrorCode status = U_ZERO_ERROR;
1431     TimeZone *z = TimeZone::createDefault();
1432     Calendar *c = Calendar::createInstance(status);
1433     if (failure(status, "Calendar::createInstance", TRUE)) return;
1434     UDate d = - z->getRawOffset();
1435     GregorianCalendar *gc = new GregorianCalendar(status);
1436     if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
1437     gc->setTimeZone(*z);
1438     gc->setTime(d, status);
1439     if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
1440     UBool idt = gc->inDaylightTime(status);
1441     if (U_FAILURE(status)) { errln("GregorianCalendar::inDaylightTime failed"); return; }
1442     if (idt) {
1443         UnicodeString str;
1444         logln("Warning: Skipping test because " + dateToString(d, str) + " is in DST.");
1445     }
1446     else {
1447         c->setTime(d, status);
1448         if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
1449         for (int32_t i = 0; i < UCAL_ZONE_OFFSET;++i) {
1450             if (c->get((UCalendarDateFields)i, status) != EPOCH_FIELDS[i])
1451                 dataerrln(UnicodeString("Expected field ") + i + " to have value " + EPOCH_FIELDS[i] +
1452                       "; saw " + c->get((UCalendarDateFields)i, status) + " instead");
1453             if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1454         }
1455         if (c->get(UCAL_ZONE_OFFSET, status) != z->getRawOffset())
1456         {
1457             errln(UnicodeString("Expected field ZONE_OFFSET to have value ") + z->getRawOffset() +
1458                   "; saw " + c->get(UCAL_ZONE_OFFSET, status) + " instead");
1459             if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1460         }
1461         if (c->get(UCAL_DST_OFFSET, status) != 0)
1462         {
1463             errln(UnicodeString("Expected field DST_OFFSET to have value 0") +
1464                   "; saw " + c->get(UCAL_DST_OFFSET, status) + " instead");
1465             if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1466         }
1467     }
1468     delete c;
1469     delete z;
1470     delete gc;
1471 }
1472 
1473 int32_t CalendarTest::EPOCH_FIELDS[] = {
1474     1, 1970, 0, 1, 1, 1, 1, 5, 1, 0, 0, 0, 0, 0, 0, - 28800000, 0
1475 };
1476 
1477 // -------------------------------------
1478 
1479 /**
1480  * Test that the days of the week progress properly when add is called repeatedly
1481  * for increments of 24 days.
1482  */
1483 void
TestDOWProgression()1484 CalendarTest::TestDOWProgression()
1485 {
1486     UErrorCode status = U_ZERO_ERROR;
1487     Calendar *cal = new GregorianCalendar(1972, UCAL_OCTOBER, 26, status);
1488     if (failure(status, "new GregorianCalendar", TRUE)) return;
1489     marchByDelta(cal, 24);
1490     delete cal;
1491 }
1492 
1493 // -------------------------------------
1494 
1495 void
TestDOW_LOCALandYEAR_WOY()1496 CalendarTest::TestDOW_LOCALandYEAR_WOY()
1497 {
1498     /* Note: I've commented out the loop_addroll tests for YEAR and
1499      * YEAR_WOY below because these two fields should NOT behave
1500      * identically when adding.  YEAR should keep the month/dom
1501      * invariant.  YEAR_WOY should keep the woy/dow invariant.  I've
1502      * added a new test that checks for this in place of the old call
1503      * to loop_addroll. - aliu */
1504     UErrorCode status = U_ZERO_ERROR;
1505     int32_t times = 20;
1506     Calendar *cal=Calendar::createInstance(Locale::getGermany(), status);
1507     if (failure(status, "Calendar::createInstance", TRUE)) return;
1508     SimpleDateFormat *sdf=new SimpleDateFormat(UnicodeString("YYYY'-W'ww-ee"), Locale::getGermany(), status);
1509     if (U_FAILURE(status)) { dataerrln("Couldn't create SimpleDateFormat - %s", u_errorName(status)); return; }
1510 
1511     // ICU no longer use localized date-time pattern characters by default.
1512     // So we set pattern chars using 'J' instead of 'Y'.
1513     DateFormatSymbols *dfs = new DateFormatSymbols(Locale::getGermany(), status);
1514     dfs->setLocalPatternChars(UnicodeString("GyMdkHmsSEDFwWahKzJeugAZvcLQq"));
1515     sdf->adoptDateFormatSymbols(dfs);
1516     sdf->applyLocalizedPattern(UnicodeString("JJJJ'-W'ww-ee"), status);
1517     if (U_FAILURE(status)) { errln("Couldn't apply localized pattern"); return; }
1518 
1519     cal->clear();
1520     cal->set(1997, UCAL_DECEMBER, 25);
1521     doYEAR_WOYLoop(cal, sdf, times, status);
1522     //loop_addroll(cal, /*sdf,*/ times, UCAL_YEAR_WOY, UCAL_YEAR,  status);
1523     yearAddTest(*cal, status); // aliu
1524     loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
1525     if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1997"); return; }
1526 
1527     cal->clear();
1528     cal->set(1998, UCAL_DECEMBER, 25);
1529     doYEAR_WOYLoop(cal, sdf, times, status);
1530     //loop_addroll(cal, /*sdf,*/ times, UCAL_YEAR_WOY, UCAL_YEAR,  status);
1531     yearAddTest(*cal, status); // aliu
1532     loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
1533     if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1998"); return; }
1534 
1535     cal->clear();
1536     cal->set(1582, UCAL_OCTOBER, 1);
1537     doYEAR_WOYLoop(cal, sdf, times, status);
1538     //loop_addroll(cal, /*sdf,*/ times, Calendar::YEAR_WOY, Calendar::YEAR,  status);
1539     yearAddTest(*cal, status); // aliu
1540     loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
1541     if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1582"); return; }
1542     delete sdf;
1543     delete cal;
1544 
1545     return;
1546 }
1547 
1548 /**
1549  * Confirm that adding a YEAR and adding a YEAR_WOY work properly for
1550  * the given Calendar at its current setting.
1551  */
yearAddTest(Calendar & cal,UErrorCode & status)1552 void CalendarTest::yearAddTest(Calendar& cal, UErrorCode& status) {
1553     /**
1554      * When adding the YEAR, the month and day should remain constant.
1555      * When adding the YEAR_WOY, the WOY and DOW should remain constant. - aliu
1556      * Examples:
1557      *  Wed Jan 14 1998 / 1998-W03-03 Add(YEAR_WOY, 1) -> Wed Jan 20 1999 / 1999-W03-03
1558      *                                Add(YEAR, 1)     -> Thu Jan 14 1999 / 1999-W02-04
1559      *  Thu Jan 14 1999 / 1999-W02-04 Add(YEAR_WOY, 1) -> Thu Jan 13 2000 / 2000-W02-04
1560      *                                Add(YEAR, 1)     -> Fri Jan 14 2000 / 2000-W02-05
1561      *  Sun Oct 31 1582 / 1582-W42-07 Add(YEAR_WOY, 1) -> Sun Oct 23 1583 / 1583-W42-07
1562      *                                Add(YEAR, 1)     -> Mon Oct 31 1583 / 1583-W44-01
1563      */
1564     int32_t y   = cal.get(UCAL_YEAR, status);
1565     int32_t mon = cal.get(UCAL_MONTH, status);
1566     int32_t day = cal.get(UCAL_DATE, status);
1567     int32_t ywy = cal.get(UCAL_YEAR_WOY, status);
1568     int32_t woy = cal.get(UCAL_WEEK_OF_YEAR, status);
1569     int32_t dow = cal.get(UCAL_DOW_LOCAL, status);
1570     UDate t = cal.getTime(status);
1571 
1572     if(U_FAILURE(status)){
1573         errln(UnicodeString("Failed to create Calendar for locale. Error: ") + UnicodeString(u_errorName(status)));
1574         return;
1575     }
1576     UnicodeString str, str2;
1577     SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"), status);
1578     fmt.setCalendar(cal);
1579 
1580     fmt.format(t, str.remove());
1581     str += ".add(YEAR, 1)    =>";
1582     cal.add(UCAL_YEAR, 1, status);
1583     int32_t y2   = cal.get(UCAL_YEAR, status);
1584     int32_t mon2 = cal.get(UCAL_MONTH, status);
1585     int32_t day2 = cal.get(UCAL_DATE, status);
1586     fmt.format(cal.getTime(status), str);
1587     if (y2 != (y+1) || mon2 != mon || day2 != day) {
1588         str += (UnicodeString)", expected year " +
1589             (y+1) + ", month " + (mon+1) + ", day " + day;
1590         errln((UnicodeString)"FAIL: " + str);
1591         logln( UnicodeString(" -> ") + CalendarTest::calToStr(cal) );
1592     } else {
1593         logln(str);
1594     }
1595 
1596     fmt.format(t, str.remove());
1597     str += ".add(YEAR_WOY, 1)=>";
1598     cal.setTime(t, status);
1599     logln( UnicodeString(" <- ") + CalendarTest::calToStr(cal) );
1600     cal.add(UCAL_YEAR_WOY, 1, status);
1601     int32_t ywy2 = cal.get(UCAL_YEAR_WOY, status);
1602     int32_t woy2 = cal.get(UCAL_WEEK_OF_YEAR, status);
1603     int32_t dow2 = cal.get(UCAL_DOW_LOCAL, status);
1604     fmt.format(cal.getTime(status), str);
1605     if (ywy2 != (ywy+1) || woy2 != woy || dow2 != dow) {
1606         str += (UnicodeString)", expected yearWOY " +
1607             (ywy+1) + ", woy " + woy + ", dowLocal " + dow;
1608         errln((UnicodeString)"FAIL: " + str);
1609         logln( UnicodeString(" -> ") + CalendarTest::calToStr(cal) );
1610     } else {
1611         logln(str);
1612     }
1613 }
1614 
1615 // -------------------------------------
1616 
loop_addroll(Calendar * cal,int times,UCalendarDateFields field,UCalendarDateFields field2,UErrorCode & errorCode)1617 void CalendarTest::loop_addroll(Calendar *cal, /*SimpleDateFormat *sdf,*/ int times, UCalendarDateFields field, UCalendarDateFields field2, UErrorCode& errorCode) {
1618     Calendar *calclone;
1619     SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"), errorCode);
1620     fmt.setCalendar(*cal);
1621     int i;
1622 
1623     for(i = 0; i<times; i++) {
1624         calclone = cal->clone();
1625         UDate start = cal->getTime(errorCode);
1626         cal->add(field,1,errorCode);
1627         if (U_FAILURE(errorCode)) { errln("Error in add"); delete calclone; return; }
1628         calclone->add(field2,1,errorCode);
1629         if (U_FAILURE(errorCode)) { errln("Error in add"); delete calclone; return; }
1630         if(cal->getTime(errorCode) != calclone->getTime(errorCode)) {
1631             UnicodeString str("FAIL: Results of add differ. "), str2;
1632             str += fmt.format(start, str2) + " ";
1633             str += UnicodeString("Add(") + fieldName(field) + ", 1) -> " +
1634                 fmt.format(cal->getTime(errorCode), str2.remove()) + "; ";
1635             str += UnicodeString("Add(") + fieldName(field2) + ", 1) -> " +
1636                 fmt.format(calclone->getTime(errorCode), str2.remove());
1637             errln(str);
1638             delete calclone;
1639             return;
1640         }
1641         delete calclone;
1642     }
1643 
1644     for(i = 0; i<times; i++) {
1645         calclone = cal->clone();
1646         cal->roll(field,(int32_t)1,errorCode);
1647         if (U_FAILURE(errorCode)) { errln("Error in roll"); delete calclone; return; }
1648         calclone->roll(field2,(int32_t)1,errorCode);
1649         if (U_FAILURE(errorCode)) { errln("Error in roll"); delete calclone; return; }
1650         if(cal->getTime(errorCode) != calclone->getTime(errorCode)) {
1651             delete calclone;
1652             errln("Results of roll differ!");
1653             return;
1654         }
1655         delete calclone;
1656     }
1657 }
1658 
1659 // -------------------------------------
1660 
1661 void
doYEAR_WOYLoop(Calendar * cal,SimpleDateFormat * sdf,int32_t times,UErrorCode & errorCode)1662 CalendarTest::doYEAR_WOYLoop(Calendar *cal, SimpleDateFormat *sdf,
1663                                     int32_t times, UErrorCode& errorCode) {
1664 
1665     UnicodeString us;
1666     UDate tst, original;
1667     Calendar *tstres = new GregorianCalendar(Locale::getGermany(), errorCode);
1668     for(int i=0; i<times; ++i) {
1669         sdf->format(Formattable(cal->getTime(errorCode),Formattable::kIsDate), us, errorCode);
1670         //logln("expected: "+us);
1671         if (U_FAILURE(errorCode)) { errln("Format error"); return; }
1672         tst=sdf->parse(us,errorCode);
1673         if (U_FAILURE(errorCode)) { errln("Parse error"); return; }
1674         tstres->clear();
1675         tstres->setTime(tst, errorCode);
1676         //logln((UnicodeString)"Parsed week of year is "+tstres->get(UCAL_WEEK_OF_YEAR, errorCode));
1677         if (U_FAILURE(errorCode)) { errln("Set time error"); return; }
1678         original = cal->getTime(errorCode);
1679         us.remove();
1680         sdf->format(Formattable(tst,Formattable::kIsDate), us, errorCode);
1681         //logln("got: "+us);
1682         if (U_FAILURE(errorCode)) { errln("Get time error"); return; }
1683         if(original!=tst) {
1684             us.remove();
1685             sdf->format(Formattable(original, Formattable::kIsDate), us, errorCode);
1686             errln("FAIL: Parsed time doesn't match with regular");
1687             logln("expected "+us + " " + calToStr(*cal));
1688             us.remove();
1689             sdf->format(Formattable(tst, Formattable::kIsDate), us, errorCode);
1690             logln("got "+us + " " + calToStr(*tstres));
1691         }
1692         tstres->clear();
1693         tstres->set(UCAL_YEAR_WOY, cal->get(UCAL_YEAR_WOY, errorCode));
1694         tstres->set(UCAL_WEEK_OF_YEAR, cal->get(UCAL_WEEK_OF_YEAR, errorCode));
1695         tstres->set(UCAL_DOW_LOCAL, cal->get(UCAL_DOW_LOCAL, errorCode));
1696         if(cal->get(UCAL_YEAR, errorCode) != tstres->get(UCAL_YEAR, errorCode)) {
1697             errln("FAIL: Different Year!");
1698             logln((UnicodeString)"Expected "+cal->get(UCAL_YEAR, errorCode));
1699             logln((UnicodeString)"Got "+tstres->get(UCAL_YEAR, errorCode));
1700             return;
1701         }
1702         if(cal->get(UCAL_DAY_OF_YEAR, errorCode) != tstres->get(UCAL_DAY_OF_YEAR, errorCode)) {
1703             errln("FAIL: Different Day Of Year!");
1704             logln((UnicodeString)"Expected "+cal->get(UCAL_DAY_OF_YEAR, errorCode));
1705             logln((UnicodeString)"Got "+tstres->get(UCAL_DAY_OF_YEAR, errorCode));
1706             return;
1707         }
1708         //logln(calToStr(*cal));
1709         cal->add(UCAL_DATE, 1, errorCode);
1710         if (U_FAILURE(errorCode)) { errln("Add error"); return; }
1711         us.remove();
1712     }
1713     delete (tstres);
1714 }
1715 // -------------------------------------
1716 
1717 void
marchByDelta(Calendar * cal,int32_t delta)1718 CalendarTest::marchByDelta(Calendar* cal, int32_t delta)
1719 {
1720     UErrorCode status = U_ZERO_ERROR;
1721     Calendar *cur = (Calendar*) cal->clone();
1722     int32_t initialDOW = cur->get(UCAL_DAY_OF_WEEK, status);
1723     if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1724     int32_t DOW, newDOW = initialDOW;
1725     do {
1726         UnicodeString str;
1727         DOW = newDOW;
1728         logln(UnicodeString("DOW = ") + DOW + "  " + dateToString(cur->getTime(status), str));
1729         if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
1730         cur->add(UCAL_DAY_OF_WEEK, delta, status);
1731         if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
1732         newDOW = cur->get(UCAL_DAY_OF_WEEK, status);
1733         if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
1734         int32_t expectedDOW = 1 + (DOW + delta - 1) % 7;
1735         if (newDOW != expectedDOW) {
1736             errln(UnicodeString("Day of week should be ") + expectedDOW + " instead of " + newDOW +
1737                   " on " + dateToString(cur->getTime(status), str));
1738             if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
1739             return;
1740         }
1741     }
1742     while (newDOW != initialDOW);
1743     delete cur;
1744 }
1745 
1746 #define CHECK(status, msg) \
1747     if (U_FAILURE(status)) { \
1748         errcheckln(status, msg); \
1749         return; \
1750     }
1751 
TestWOY(void)1752 void CalendarTest::TestWOY(void) {
1753     /*
1754       FDW = Mon, MDFW = 4:
1755          Sun Dec 26 1999, WOY 51
1756          Mon Dec 27 1999, WOY 52
1757          Tue Dec 28 1999, WOY 52
1758          Wed Dec 29 1999, WOY 52
1759          Thu Dec 30 1999, WOY 52
1760          Fri Dec 31 1999, WOY 52
1761          Sat Jan 01 2000, WOY 52 ***
1762          Sun Jan 02 2000, WOY 52 ***
1763          Mon Jan 03 2000, WOY 1
1764          Tue Jan 04 2000, WOY 1
1765          Wed Jan 05 2000, WOY 1
1766          Thu Jan 06 2000, WOY 1
1767          Fri Jan 07 2000, WOY 1
1768          Sat Jan 08 2000, WOY 1
1769          Sun Jan 09 2000, WOY 1
1770          Mon Jan 10 2000, WOY 2
1771 
1772       FDW = Mon, MDFW = 2:
1773          Sun Dec 26 1999, WOY 52
1774          Mon Dec 27 1999, WOY 1  ***
1775          Tue Dec 28 1999, WOY 1  ***
1776          Wed Dec 29 1999, WOY 1  ***
1777          Thu Dec 30 1999, WOY 1  ***
1778          Fri Dec 31 1999, WOY 1  ***
1779          Sat Jan 01 2000, WOY 1
1780          Sun Jan 02 2000, WOY 1
1781          Mon Jan 03 2000, WOY 2
1782          Tue Jan 04 2000, WOY 2
1783          Wed Jan 05 2000, WOY 2
1784          Thu Jan 06 2000, WOY 2
1785          Fri Jan 07 2000, WOY 2
1786          Sat Jan 08 2000, WOY 2
1787          Sun Jan 09 2000, WOY 2
1788          Mon Jan 10 2000, WOY 3
1789     */
1790 
1791     UnicodeString str;
1792     UErrorCode status = U_ZERO_ERROR;
1793     int32_t i;
1794 
1795     GregorianCalendar cal(status);
1796     SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy', WOY' w"), status);
1797     if (failure(status, "Cannot construct calendar/format", TRUE)) return;
1798 
1799     UCalendarDaysOfWeek fdw = (UCalendarDaysOfWeek) 0;
1800 
1801     //for (int8_t pass=2; pass<=2; ++pass) {
1802     for (int8_t pass=1; pass<=2; ++pass) {
1803         switch (pass) {
1804         case 1:
1805             fdw = UCAL_MONDAY;
1806             cal.setFirstDayOfWeek(fdw);
1807             cal.setMinimalDaysInFirstWeek(4);
1808             fmt.adoptCalendar(cal.clone());
1809             break;
1810         case 2:
1811             fdw = UCAL_MONDAY;
1812             cal.setFirstDayOfWeek(fdw);
1813             cal.setMinimalDaysInFirstWeek(2);
1814             fmt.adoptCalendar(cal.clone());
1815             break;
1816         }
1817 
1818         //for (i=2; i<=6; ++i) {
1819         for (i=0; i<16; ++i) {
1820         UDate t, t2;
1821         int32_t t_y, t_woy, t_dow;
1822         cal.clear();
1823         cal.set(1999, UCAL_DECEMBER, 26 + i);
1824         fmt.format(t = cal.getTime(status), str.remove());
1825         CHECK(status, "Fail: getTime failed");
1826         logln(UnicodeString("* ") + str);
1827         int32_t dow = cal.get(UCAL_DAY_OF_WEEK, status);
1828         int32_t woy = cal.get(UCAL_WEEK_OF_YEAR, status);
1829         int32_t year = cal.get(UCAL_YEAR, status);
1830         int32_t mon = cal.get(UCAL_MONTH, status);
1831         logln(calToStr(cal));
1832         CHECK(status, "Fail: get failed");
1833         int32_t dowLocal = dow - fdw;
1834         if (dowLocal < 0) dowLocal += 7;
1835         dowLocal++;
1836         int32_t yearWoy = year;
1837         if (mon == UCAL_JANUARY) {
1838             if (woy >= 52) --yearWoy;
1839         } else {
1840             if (woy == 1) ++yearWoy;
1841         }
1842 
1843         // Basic fields->time check y/woy/dow
1844         // Since Y/WOY is ambiguous, we do a check of the fields,
1845         // not of the specific time.
1846         cal.clear();
1847         cal.set(UCAL_YEAR, year);
1848         cal.set(UCAL_WEEK_OF_YEAR, woy);
1849         cal.set(UCAL_DAY_OF_WEEK, dow);
1850         t_y = cal.get(UCAL_YEAR, status);
1851         t_woy = cal.get(UCAL_WEEK_OF_YEAR, status);
1852         t_dow = cal.get(UCAL_DAY_OF_WEEK, status);
1853         CHECK(status, "Fail: get failed");
1854         if (t_y != year || t_woy != woy || t_dow != dow) {
1855             str = "Fail: y/woy/dow fields->time => ";
1856             fmt.format(cal.getTime(status), str);
1857             errln(str);
1858             logln(calToStr(cal));
1859             logln("[get!=set] Y%d!=%d || woy%d!=%d || dow%d!=%d\n",
1860                   t_y, year, t_woy, woy, t_dow, dow);
1861         } else {
1862           logln("y/woy/dow fields->time OK");
1863         }
1864 
1865         // Basic fields->time check y/woy/dow_local
1866         // Since Y/WOY is ambiguous, we do a check of the fields,
1867         // not of the specific time.
1868         cal.clear();
1869         cal.set(UCAL_YEAR, year);
1870         cal.set(UCAL_WEEK_OF_YEAR, woy);
1871         cal.set(UCAL_DOW_LOCAL, dowLocal);
1872         t_y = cal.get(UCAL_YEAR, status);
1873         t_woy = cal.get(UCAL_WEEK_OF_YEAR, status);
1874         t_dow = cal.get(UCAL_DOW_LOCAL, status);
1875         CHECK(status, "Fail: get failed");
1876         if (t_y != year || t_woy != woy || t_dow != dowLocal) {
1877             str = "Fail: y/woy/dow_local fields->time => ";
1878             fmt.format(cal.getTime(status), str);
1879             errln(str);
1880         }
1881 
1882         // Basic fields->time check y_woy/woy/dow
1883         cal.clear();
1884         cal.set(UCAL_YEAR_WOY, yearWoy);
1885         cal.set(UCAL_WEEK_OF_YEAR, woy);
1886         cal.set(UCAL_DAY_OF_WEEK, dow);
1887         t2 = cal.getTime(status);
1888         CHECK(status, "Fail: getTime failed");
1889         if (t != t2) {
1890             str = "Fail: y_woy/woy/dow fields->time => ";
1891             fmt.format(t2, str);
1892             errln(str);
1893             logln(calToStr(cal));
1894             logln("%.f != %.f\n", t, t2);
1895         } else {
1896           logln("y_woy/woy/dow OK");
1897         }
1898 
1899         // Basic fields->time check y_woy/woy/dow_local
1900         cal.clear();
1901         cal.set(UCAL_YEAR_WOY, yearWoy);
1902         cal.set(UCAL_WEEK_OF_YEAR, woy);
1903         cal.set(UCAL_DOW_LOCAL, dowLocal);
1904         t2 = cal.getTime(status);
1905         CHECK(status, "Fail: getTime failed");
1906         if (t != t2) {
1907             str = "Fail: y_woy/woy/dow_local fields->time => ";
1908             fmt.format(t2, str);
1909             errln(str);
1910         }
1911 
1912         logln("Testing DOW_LOCAL.. dow%d\n", dow);
1913         // Make sure DOW_LOCAL disambiguates over DOW
1914         int32_t wrongDow = dow - 3;
1915         if (wrongDow < 1) wrongDow += 7;
1916         cal.setTime(t, status);
1917         cal.set(UCAL_DAY_OF_WEEK, wrongDow);
1918         cal.set(UCAL_DOW_LOCAL, dowLocal);
1919         t2 = cal.getTime(status);
1920         CHECK(status, "Fail: set/getTime failed");
1921         if (t != t2) {
1922             str = "Fail: DOW_LOCAL fields->time => ";
1923             fmt.format(t2, str);
1924             errln(str);
1925             logln(calToStr(cal));
1926             logln("%.f :   DOW%d, DOW_LOCAL%d -> %.f\n",
1927                   t, wrongDow, dowLocal, t2);
1928         }
1929 
1930         // Make sure DOW disambiguates over DOW_LOCAL
1931         int32_t wrongDowLocal = dowLocal - 3;
1932         if (wrongDowLocal < 1) wrongDowLocal += 7;
1933         cal.setTime(t, status);
1934         cal.set(UCAL_DOW_LOCAL, wrongDowLocal);
1935         cal.set(UCAL_DAY_OF_WEEK, dow);
1936         t2 = cal.getTime(status);
1937         CHECK(status, "Fail: set/getTime failed");
1938         if (t != t2) {
1939             str = "Fail: DOW       fields->time => ";
1940             fmt.format(t2, str);
1941             errln(str);
1942         }
1943 
1944         // Make sure YEAR_WOY disambiguates over YEAR
1945         cal.setTime(t, status);
1946         cal.set(UCAL_YEAR, year - 2);
1947         cal.set(UCAL_YEAR_WOY, yearWoy);
1948         t2 = cal.getTime(status);
1949         CHECK(status, "Fail: set/getTime failed");
1950         if (t != t2) {
1951             str = "Fail: YEAR_WOY  fields->time => ";
1952             fmt.format(t2, str);
1953             errln(str);
1954         }
1955 
1956         // Make sure YEAR disambiguates over YEAR_WOY
1957         cal.setTime(t, status);
1958         cal.set(UCAL_YEAR_WOY, yearWoy - 2);
1959         cal.set(UCAL_YEAR, year);
1960         t2 = cal.getTime(status);
1961         CHECK(status, "Fail: set/getTime failed");
1962         if (t != t2) {
1963             str = "Fail: YEAR      fields->time => ";
1964             fmt.format(t2, str);
1965             errln(str);
1966         }
1967     }
1968     }
1969 
1970     /*
1971       FDW = Mon, MDFW = 4:
1972          Sun Dec 26 1999, WOY 51
1973          Mon Dec 27 1999, WOY 52
1974          Tue Dec 28 1999, WOY 52
1975          Wed Dec 29 1999, WOY 52
1976          Thu Dec 30 1999, WOY 52
1977          Fri Dec 31 1999, WOY 52
1978          Sat Jan 01 2000, WOY 52
1979          Sun Jan 02 2000, WOY 52
1980     */
1981 
1982     // Roll the DOW_LOCAL within week 52
1983     for (i=27; i<=33; ++i) {
1984         int32_t amount;
1985         for (amount=-7; amount<=7; ++amount) {
1986             str = "roll(";
1987             cal.set(1999, UCAL_DECEMBER, i);
1988             UDate t, t2;
1989             fmt.format(cal.getTime(status), str);
1990             CHECK(status, "Fail: getTime failed");
1991             str += UnicodeString(", ") + amount + ") = ";
1992 
1993             cal.roll(UCAL_DOW_LOCAL, amount, status);
1994             CHECK(status, "Fail: roll failed");
1995 
1996             t = cal.getTime(status);
1997             int32_t newDom = i + amount;
1998             while (newDom < 27) newDom += 7;
1999             while (newDom > 33) newDom -= 7;
2000             cal.set(1999, UCAL_DECEMBER, newDom);
2001             t2 = cal.getTime(status);
2002             CHECK(status, "Fail: getTime failed");
2003             fmt.format(t, str);
2004 
2005             if (t != t2) {
2006                 str.append(", exp ");
2007                 fmt.format(t2, str);
2008                 errln(str);
2009             } else {
2010                 logln(str);
2011             }
2012         }
2013     }
2014 }
2015 
TestYWOY()2016 void CalendarTest::TestYWOY()
2017 {
2018    UnicodeString str;
2019    UErrorCode status = U_ZERO_ERROR;
2020 
2021    GregorianCalendar cal(status);
2022    if (failure(status, "construct GregorianCalendar", TRUE)) return;
2023 
2024    cal.setFirstDayOfWeek(UCAL_SUNDAY);
2025    cal.setMinimalDaysInFirstWeek(1);
2026 
2027    logln("Setting:  ywoy=2004, woy=1, dow=MONDAY");
2028    cal.clear();
2029    cal.set(UCAL_YEAR_WOY,2004);
2030    cal.set(UCAL_WEEK_OF_YEAR,1);
2031    cal.set(UCAL_DAY_OF_WEEK, UCAL_MONDAY);
2032 
2033    logln(calToStr(cal));
2034    if(cal.get(UCAL_YEAR, status) != 2003) {
2035      errln("year not 2003");
2036    }
2037 
2038    logln("+ setting DOW to THURSDAY");
2039    cal.clear();
2040    cal.set(UCAL_YEAR_WOY,2004);
2041    cal.set(UCAL_WEEK_OF_YEAR,1);
2042    cal.set(UCAL_DAY_OF_WEEK, UCAL_THURSDAY);
2043 
2044    logln(calToStr(cal));
2045    if(cal.get(UCAL_YEAR, status) != 2004) {
2046      errln("year not 2004");
2047    }
2048 
2049    logln("+ setting DOW_LOCAL to 1");
2050    cal.clear();
2051    cal.set(UCAL_YEAR_WOY,2004);
2052    cal.set(UCAL_WEEK_OF_YEAR,1);
2053    cal.set(UCAL_DAY_OF_WEEK, UCAL_THURSDAY);
2054    cal.set(UCAL_DOW_LOCAL, 1);
2055 
2056    logln(calToStr(cal));
2057    if(cal.get(UCAL_YEAR, status) != 2003) {
2058      errln("year not 2003");
2059    }
2060 
2061    cal.setFirstDayOfWeek(UCAL_MONDAY);
2062    cal.setMinimalDaysInFirstWeek(4);
2063    UDate t = 946713600000.;
2064    cal.setTime(t, status);
2065    cal.set(UCAL_DAY_OF_WEEK, 4);
2066    cal.set(UCAL_DOW_LOCAL, 6);
2067    if(cal.getTime(status) != t) {
2068      logln(calToStr(cal));
2069      errln("FAIL:  DOW_LOCAL did not take precedence");
2070    }
2071 
2072 }
2073 
TestJD()2074 void CalendarTest::TestJD()
2075 {
2076   int32_t jd;
2077   static const int32_t kEpochStartAsJulianDay = 2440588;
2078   UErrorCode status = U_ZERO_ERROR;
2079   GregorianCalendar cal(status);
2080   if (failure(status, "construct GregorianCalendar", TRUE)) return;
2081   cal.setTimeZone(*TimeZone::getGMT());
2082   cal.clear();
2083   jd = cal.get(UCAL_JULIAN_DAY, status);
2084   if(jd != kEpochStartAsJulianDay) {
2085     errln("Wanted JD of %d at time=0, [epoch 1970] but got %d\n", kEpochStartAsJulianDay, jd);
2086   } else {
2087     logln("Wanted JD of %d at time=0, [epoch 1970], got %d\n", kEpochStartAsJulianDay, jd);
2088   }
2089 
2090   cal.setTime(Calendar::getNow(), status);
2091   cal.clear();
2092   cal.set(UCAL_JULIAN_DAY, kEpochStartAsJulianDay);
2093   UDate epochTime = cal.getTime(status);
2094   if(epochTime != 0) {
2095     errln("Wanted time of 0 at jd=%d, got %.1lf\n", kEpochStartAsJulianDay, epochTime);
2096   } else {
2097     logln("Wanted time of 0 at jd=%d, got %.1lf\n", kEpochStartAsJulianDay, epochTime);
2098   }
2099 
2100 }
2101 
2102 // make sure the ctestfw utilities are in sync with the Calendar
TestDebug()2103 void CalendarTest::TestDebug()
2104 {
2105     for(int32_t  t=0;t<=UDBG_ENUM_COUNT;t++) {
2106         int32_t count = udbg_enumCount((UDebugEnumType)t);
2107         if(count == -1) {
2108             logln("enumCount(%d) returned -1", count);
2109             continue;
2110         }
2111         for(int32_t i=0;i<=count;i++) {
2112             if(t<=UDBG_HIGHEST_CONTIGUOUS_ENUM && i<count) {
2113                 if( i!=udbg_enumArrayValue((UDebugEnumType)t, i)) {
2114                     errln("FAIL: udbg_enumArrayValue(%d,%d) returned %d, expected %d", t, i, udbg_enumArrayValue((UDebugEnumType)t,i), i);
2115                 }
2116             } else {
2117                 logln("Testing count+1:");
2118             }
2119                   const char *name = udbg_enumName((UDebugEnumType)t,i);
2120                   if(name==NULL) {
2121                           if(i==count || t>UDBG_HIGHEST_CONTIGUOUS_ENUM  ) {
2122                                 logln(" null name - expected.\n");
2123                           } else {
2124                                 errln("FAIL: udbg_enumName(%d,%d) returned NULL", t, i);
2125                           }
2126                           name = "(null)";
2127                   }
2128           logln("udbg_enumArrayValue(%d,%d) = %s, returned %d", t, i,
2129                       name, udbg_enumArrayValue((UDebugEnumType)t,i));
2130             logln("udbg_enumString = " + udbg_enumString((UDebugEnumType)t,i));
2131         }
2132         if(udbg_enumExpectedCount((UDebugEnumType)t) != count && t<=UDBG_HIGHEST_CONTIGUOUS_ENUM) {
2133             errln("FAIL: udbg_enumExpectedCount(%d): %d, != UCAL_FIELD_COUNT=%d ", t, udbg_enumExpectedCount((UDebugEnumType)t), count);
2134         } else {
2135             logln("udbg_ucal_fieldCount: %d, UCAL_FIELD_COUNT=udbg_enumCount %d ", udbg_enumExpectedCount((UDebugEnumType)t), count);
2136         }
2137     }
2138 }
2139 
2140 
2141 #undef CHECK
2142 
2143 // List of interesting locales
testLocaleID(int32_t i)2144 const char *CalendarTest::testLocaleID(int32_t i)
2145 {
2146   switch(i) {
2147   case 0: return "he_IL@calendar=hebrew";
2148   case 1: return "en_US@calendar=hebrew";
2149   case 2: return "fr_FR@calendar=hebrew";
2150   case 3: return "fi_FI@calendar=hebrew";
2151   case 4: return "nl_NL@calendar=hebrew";
2152   case 5: return "hu_HU@calendar=hebrew";
2153   case 6: return "nl_BE@currency=MTL;calendar=islamic";
2154   case 7: return "th_TH_TRADITIONAL@calendar=gregorian";
2155   case 8: return "ar_JO@calendar=islamic-civil";
2156   case 9: return "fi_FI@calendar=islamic";
2157   case 10: return "fr_CH@calendar=islamic-civil";
2158   case 11: return "he_IL@calendar=islamic-civil";
2159   case 12: return "hu_HU@calendar=buddhist";
2160   case 13: return "hu_HU@calendar=islamic";
2161   case 14: return "en_US@calendar=japanese";
2162   default: return NULL;
2163   }
2164 }
2165 
testLocaleCount()2166 int32_t CalendarTest::testLocaleCount()
2167 {
2168   static int32_t gLocaleCount = -1;
2169   if(gLocaleCount < 0) {
2170     int32_t i;
2171     for(i=0;testLocaleID(i) != NULL;i++) {
2172       ;
2173     }
2174     gLocaleCount = i;
2175   }
2176   return gLocaleCount;
2177 }
2178 
doMinDateOfCalendar(Calendar * adopt,UBool & isGregorian,UErrorCode & status)2179 static UDate doMinDateOfCalendar(Calendar* adopt, UBool &isGregorian, UErrorCode& status) {
2180   if(U_FAILURE(status)) return 0.0;
2181 
2182   adopt->clear();
2183   adopt->set(UCAL_EXTENDED_YEAR, adopt->getActualMinimum(UCAL_EXTENDED_YEAR, status));
2184   UDate ret = adopt->getTime(status);
2185   isGregorian = dynamic_cast<GregorianCalendar*>(adopt) != NULL;
2186   delete adopt;
2187   return ret;
2188 }
2189 
minDateOfCalendar(const Locale & locale,UBool & isGregorian,UErrorCode & status)2190 UDate CalendarTest::minDateOfCalendar(const Locale& locale, UBool &isGregorian, UErrorCode& status) {
2191   if(U_FAILURE(status)) return 0.0;
2192   return doMinDateOfCalendar(Calendar::createInstance(locale, status), isGregorian, status);
2193 }
2194 
minDateOfCalendar(const Calendar & cal,UBool & isGregorian,UErrorCode & status)2195 UDate CalendarTest::minDateOfCalendar(const Calendar& cal, UBool &isGregorian, UErrorCode& status) {
2196   if(U_FAILURE(status)) return 0.0;
2197   return doMinDateOfCalendar(cal.clone(), isGregorian, status);
2198 }
2199 
Test6703()2200 void CalendarTest::Test6703()
2201 {
2202     UErrorCode status = U_ZERO_ERROR;
2203     Calendar *cal;
2204 
2205     Locale loc1("en@calendar=fubar");
2206     cal = Calendar::createInstance(loc1, status);
2207     if (failure(status, "Calendar::createInstance", TRUE)) return;
2208     delete cal;
2209 
2210     status = U_ZERO_ERROR;
2211     Locale loc2("en");
2212     cal = Calendar::createInstance(loc2, status);
2213     if (failure(status, "Calendar::createInstance")) return;
2214     delete cal;
2215 
2216     status = U_ZERO_ERROR;
2217     Locale loc3("en@calendar=roc");
2218     cal = Calendar::createInstance(loc3, status);
2219     if (failure(status, "Calendar::createInstance")) return;
2220     delete cal;
2221 
2222     return;
2223 }
2224 
Test3785()2225 void CalendarTest::Test3785()
2226 {
2227     UErrorCode status = U_ZERO_ERROR;
2228     UnicodeString uzone = UNICODE_STRING_SIMPLE("Europe/Paris");
2229     UnicodeString exp1 = UNICODE_STRING_SIMPLE("Mon 30 Jumada II 1433 AH, 01:47:03");
2230     UnicodeString exp2 = UNICODE_STRING_SIMPLE("Mon 1 Rajab 1433 AH, 01:47:04");
2231 
2232     LocalUDateFormatPointer df(udat_open(UDAT_NONE, UDAT_NONE, "en@calendar=islamic", uzone.getTerminatedBuffer(),
2233                                          uzone.length(), NULL, 0, &status));
2234     if (df.isNull() || U_FAILURE(status)) return;
2235 
2236     UChar upattern[64];
2237     u_uastrcpy(upattern, "EEE d MMMM y G, HH:mm:ss");
2238     udat_applyPattern(df.getAlias(), FALSE, upattern, u_strlen(upattern));
2239 
2240     UChar ubuffer[1024];
2241     UDate ud0 = 1337557623000.0;
2242 
2243     status = U_ZERO_ERROR;
2244     udat_format(df.getAlias(), ud0, ubuffer, 1024, NULL, &status);
2245     if (U_FAILURE(status)) {
2246         errln("Error formatting date 1\n");
2247         return;
2248     }
2249     //printf("formatted: '%s'\n", mkcstr(ubuffer));
2250 
2251     UnicodeString act1(ubuffer);
2252     if ( act1 != exp1 ) {
2253         errln("Unexpected result from date 1 format\n");
2254     }
2255     ud0 += 1000.0; // add one second
2256 
2257     status = U_ZERO_ERROR;
2258     udat_format(df.getAlias(), ud0, ubuffer, 1024, NULL, &status);
2259     if (U_FAILURE(status)) {
2260         errln("Error formatting date 2\n");
2261         return;
2262     }
2263     //printf("formatted: '%s'\n", mkcstr(ubuffer));
2264     UnicodeString act2(ubuffer);
2265     if ( act2 != exp2 ) {
2266         errln("Unexpected result from date 2 format\n");
2267     }
2268 
2269     return;
2270 }
2271 
Test1624()2272 void CalendarTest::Test1624() {
2273     UErrorCode status = U_ZERO_ERROR;
2274     Locale loc("he_IL@calendar=hebrew");
2275     HebrewCalendar hc(loc,status);
2276 
2277     for (int32_t year = 5600; year < 5800; year++ ) {
2278 
2279         for (int32_t month = HebrewCalendar::TISHRI; month <= HebrewCalendar::ELUL; month++) {
2280             // skip the adar 1 month if year is not a leap year
2281             if (HebrewCalendar::isLeapYear(year) == FALSE && month == HebrewCalendar::ADAR_1) {
2282                 continue;
2283             }
2284             int32_t day = 15;
2285             hc.set(year,month,day);
2286             int32_t dayHC = hc.get(UCAL_DATE,status);
2287             int32_t monthHC = hc.get(UCAL_MONTH,status);
2288             int32_t yearHC = hc.get(UCAL_YEAR,status);
2289 
2290             if (failure(status, "HebrewCalendar.get()", TRUE)) continue;
2291 
2292             if (dayHC != day) {
2293                 errln(" ==> day %d incorrect, should be: %d\n",dayHC,day);
2294                 break;
2295             }
2296             if (monthHC != month) {
2297                 errln(" ==> month %d incorrect, should be: %d\n",monthHC,month);
2298                 break;
2299             }
2300             if (yearHC != year) {
2301                 errln(" ==> day %d incorrect, should be: %d\n",yearHC,year);
2302                 break;
2303             }
2304         }
2305     }
2306     return;
2307 }
2308 
TestTimeStamp()2309 void CalendarTest::TestTimeStamp() {
2310     UErrorCode status = U_ZERO_ERROR;
2311     UDate start = 0.0, time;
2312     Calendar *cal;
2313 
2314     // Create a new Gregorian Calendar.
2315     cal = Calendar::createInstance("en_US@calender=gregorian", status);
2316     if (U_FAILURE(status)) {
2317         dataerrln("Error creating Gregorian calendar.");
2318         return;
2319     }
2320 
2321     for (int i = 0; i < 20000; i++) {
2322         // Set the Gregorian Calendar to a specific date for testing.
2323         cal->set(2009, UCAL_JULY, 3, 0, 49, 46);
2324 
2325         time = cal->getTime(status);
2326         if (U_FAILURE(status)) {
2327             errln("Error calling getTime()");
2328             break;
2329         }
2330 
2331         if (i == 0) {
2332             start = time;
2333         } else {
2334             if (start != time) {
2335                 errln("start and time not equal.");
2336                 break;
2337             }
2338         }
2339     }
2340 
2341     delete cal;
2342 }
2343 
TestISO8601()2344 void CalendarTest::TestISO8601() {
2345     const char* TEST_LOCALES[] = {
2346         "en_US@calendar=iso8601",
2347         "en_US@calendar=Iso8601",
2348         "th_TH@calendar=iso8601",
2349         "ar_EG@calendar=iso8601",
2350         NULL
2351     };
2352 
2353     int32_t TEST_DATA[][3] = {
2354         {2008, 1, 2008},
2355         {2009, 1, 2009},
2356         {2010, 53, 2009},
2357         {2011, 52, 2010},
2358         {2012, 52, 2011},
2359         {2013, 1, 2013},
2360         {2014, 1, 2014},
2361         {0, 0, 0},
2362     };
2363 
2364     for (int i = 0; TEST_LOCALES[i] != NULL; i++) {
2365         UErrorCode status = U_ZERO_ERROR;
2366         Calendar *cal = Calendar::createInstance(TEST_LOCALES[i], status);
2367         if (U_FAILURE(status)) {
2368             errln("Error: Failed to create a calendar for locale: %s", TEST_LOCALES[i]);
2369             continue;
2370         }
2371         if (uprv_strcmp(cal->getType(), "gregorian") != 0) {
2372             errln("Error: Gregorian calendar is not used for locale: %s", TEST_LOCALES[i]);
2373             continue;
2374         }
2375         for (int j = 0; TEST_DATA[j][0] != 0; j++) {
2376             cal->set(TEST_DATA[j][0], UCAL_JANUARY, 1);
2377             int32_t weekNum = cal->get(UCAL_WEEK_OF_YEAR, status);
2378             int32_t weekYear = cal->get(UCAL_YEAR_WOY, status);
2379             if (U_FAILURE(status)) {
2380                 errln("Error: Failed to get week of year");
2381                 break;
2382             }
2383             if (weekNum != TEST_DATA[j][1] || weekYear != TEST_DATA[j][2]) {
2384                 errln("Error: Incorrect week of year on January 1st, %d for locale %s: Returned [weekNum=%d, weekYear=%d], Expected [weekNum=%d, weekYear=%d]",
2385                     TEST_DATA[j][0], TEST_LOCALES[i], weekNum, weekYear, TEST_DATA[j][1], TEST_DATA[j][2]);
2386             }
2387         }
2388         delete cal;
2389     }
2390 
2391 }
2392 
2393 void
TestAmbiguousWallTimeAPIs(void)2394 CalendarTest::TestAmbiguousWallTimeAPIs(void) {
2395     UErrorCode status = U_ZERO_ERROR;
2396     Calendar* cal = Calendar::createInstance(status);
2397     if (U_FAILURE(status)) {
2398         errln("Fail: Error creating a calendar instance.");
2399         return;
2400     }
2401 
2402     if (cal->getRepeatedWallTimeOption() != UCAL_WALLTIME_LAST) {
2403         errln("Fail: Default repeted time option is not UCAL_WALLTIME_LAST");
2404     }
2405     if (cal->getSkippedWallTimeOption() != UCAL_WALLTIME_LAST) {
2406         errln("Fail: Default skipped time option is not UCAL_WALLTIME_LAST");
2407     }
2408 
2409     Calendar* cal2 = cal->clone();
2410 
2411     if (*cal != *cal2) {
2412         errln("Fail: Cloned calendar != the original");
2413     }
2414     if (!cal->equals(*cal2, status)) {
2415         errln("Fail: The time of cloned calendar is not equal to the original");
2416     } else if (U_FAILURE(status)) {
2417         errln("Fail: Error equals");
2418     }
2419     status = U_ZERO_ERROR;
2420 
2421     cal2->setRepeatedWallTimeOption(UCAL_WALLTIME_FIRST);
2422     cal2->setSkippedWallTimeOption(UCAL_WALLTIME_FIRST);
2423 
2424     if (*cal == *cal2) {
2425         errln("Fail: Cloned and modified calendar == the original");
2426     }
2427     if (!cal->equals(*cal2, status)) {
2428         errln("Fail: The time of cloned calendar is not equal to the original after changing wall time options");
2429     } else if (U_FAILURE(status)) {
2430         errln("Fail: Error equals after changing wall time options");
2431     }
2432     status = U_ZERO_ERROR;
2433 
2434     if (cal2->getRepeatedWallTimeOption() != UCAL_WALLTIME_FIRST) {
2435         errln("Fail: Repeted time option is not UCAL_WALLTIME_FIRST");
2436     }
2437     if (cal2->getSkippedWallTimeOption() != UCAL_WALLTIME_FIRST) {
2438         errln("Fail: Skipped time option is not UCAL_WALLTIME_FIRST");
2439     }
2440 
2441     cal2->setRepeatedWallTimeOption(UCAL_WALLTIME_NEXT_VALID);
2442     if (cal2->getRepeatedWallTimeOption() != UCAL_WALLTIME_FIRST) {
2443         errln("Fail: Repeated wall time option was updated other than UCAL_WALLTIME_FIRST");
2444     }
2445 
2446     delete cal;
2447     delete cal2;
2448 }
2449 
2450 class CalFields {
2451 public:
2452     CalFields(int32_t year, int32_t month, int32_t day, int32_t hour, int32_t min, int32_t sec, int32_t ms = 0);
2453     CalFields(const Calendar& cal, UErrorCode& status);
2454     void setTo(Calendar& cal) const;
2455     char* toString(char* buf, int32_t len) const;
2456     UBool operator==(const CalFields& rhs) const;
2457     UBool operator!=(const CalFields& rhs) const;
2458     UBool isEquivalentTo(const Calendar& cal, UErrorCode& status) const;
2459 
2460 private:
2461     int32_t year;
2462     int32_t month;
2463     int32_t day;
2464     int32_t hour;
2465     int32_t min;
2466     int32_t sec;
2467     int32_t ms;
2468 };
2469 
CalFields(int32_t year,int32_t month,int32_t day,int32_t hour,int32_t min,int32_t sec,int32_t ms)2470 CalFields::CalFields(int32_t year, int32_t month, int32_t day, int32_t hour, int32_t min, int32_t sec, int32_t ms)
2471     : year(year), month(month), day(day), hour(hour), min(min), sec(sec), ms(ms) {
2472 }
2473 
CalFields(const Calendar & cal,UErrorCode & status)2474 CalFields::CalFields(const Calendar& cal, UErrorCode& status) {
2475     year = cal.get(UCAL_YEAR, status);
2476     month = cal.get(UCAL_MONTH, status) + 1;
2477     day = cal.get(UCAL_DAY_OF_MONTH, status);
2478     hour = cal.get(UCAL_HOUR_OF_DAY, status);
2479     min = cal.get(UCAL_MINUTE, status);
2480     sec = cal.get(UCAL_SECOND, status);
2481     ms = cal.get(UCAL_MILLISECOND, status);
2482 }
2483 
2484 void
setTo(Calendar & cal) const2485 CalFields::setTo(Calendar& cal) const {
2486     cal.clear();
2487     cal.set(year, month - 1, day, hour, min, sec);
2488     cal.set(UCAL_MILLISECOND, ms);
2489 }
2490 
2491 char*
toString(char * buf,int32_t len) const2492 CalFields::toString(char* buf, int32_t len) const {
2493     char local[32];
2494     sprintf(local, "%04d-%02d-%02d %02d:%02d:%02d.%03d", year, month, day, hour, min, sec, ms);
2495     uprv_strncpy(buf, local, len - 1);
2496     buf[len - 1] = 0;
2497     return buf;
2498 }
2499 
2500 UBool
operator ==(const CalFields & rhs) const2501 CalFields::operator==(const CalFields& rhs) const {
2502     return year == rhs.year
2503         && month == rhs.month
2504         && day == rhs.day
2505         && hour == rhs.hour
2506         && min == rhs.min
2507         && sec == rhs.sec
2508         && ms == rhs.ms;
2509 }
2510 
2511 UBool
operator !=(const CalFields & rhs) const2512 CalFields::operator!=(const CalFields& rhs) const {
2513     return !(*this == rhs);
2514 }
2515 
2516 UBool
isEquivalentTo(const Calendar & cal,UErrorCode & status) const2517 CalFields::isEquivalentTo(const Calendar& cal, UErrorCode& status) const {
2518     return year == cal.get(UCAL_YEAR, status)
2519         && month == cal.get(UCAL_MONTH, status) + 1
2520         && day == cal.get(UCAL_DAY_OF_MONTH, status)
2521         && hour == cal.get(UCAL_HOUR_OF_DAY, status)
2522         && min == cal.get(UCAL_MINUTE, status)
2523         && sec == cal.get(UCAL_SECOND, status)
2524         && ms == cal.get(UCAL_MILLISECOND, status);
2525 }
2526 
2527 typedef struct {
2528     const char*     tzid;
2529     const CalFields in;
2530     const CalFields expLastGMT;
2531     const CalFields expFirstGMT;
2532 } RepeatedWallTimeTestData;
2533 
2534 static const RepeatedWallTimeTestData RPDATA[] =
2535 {
2536     // Time zone            Input wall time                 WALLTIME_LAST in GMT            WALLTIME_FIRST in GMT
2537     {"America/New_York",    CalFields(2011,11,6,0,59,59),   CalFields(2011,11,6,4,59,59),   CalFields(2011,11,6,4,59,59)},
2538     {"America/New_York",    CalFields(2011,11,6,1,0,0),     CalFields(2011,11,6,6,0,0),     CalFields(2011,11,6,5,0,0)},
2539     {"America/New_York",    CalFields(2011,11,6,1,0,1),     CalFields(2011,11,6,6,0,1),     CalFields(2011,11,6,5,0,1)},
2540     {"America/New_York",    CalFields(2011,11,6,1,30,0),    CalFields(2011,11,6,6,30,0),    CalFields(2011,11,6,5,30,0)},
2541     {"America/New_York",    CalFields(2011,11,6,1,59,59),   CalFields(2011,11,6,6,59,59),   CalFields(2011,11,6,5,59,59)},
2542     {"America/New_York",    CalFields(2011,11,6,2,0,0),     CalFields(2011,11,6,7,0,0),     CalFields(2011,11,6,7,0,0)},
2543     {"America/New_York",    CalFields(2011,11,6,2,0,1),     CalFields(2011,11,6,7,0,1),     CalFields(2011,11,6,7,0,1)},
2544 
2545     {"Australia/Lord_Howe", CalFields(2011,4,3,1,29,59),    CalFields(2011,4,2,14,29,59),   CalFields(2011,4,2,14,29,59)},
2546     {"Australia/Lord_Howe", CalFields(2011,4,3,1,30,0),     CalFields(2011,4,2,15,0,0),     CalFields(2011,4,2,14,30,0)},
2547     {"Australia/Lord_Howe", CalFields(2011,4,3,1,45,0),     CalFields(2011,4,2,15,15,0),    CalFields(2011,4,2,14,45,0)},
2548     {"Australia/Lord_Howe", CalFields(2011,4,3,1,59,59),    CalFields(2011,4,2,15,29,59),   CalFields(2011,4,2,14,59,59)},
2549     {"Australia/Lord_Howe", CalFields(2011,4,3,2,0,0),      CalFields(2011,4,2,15,30,0),    CalFields(2011,4,2,15,30,0)},
2550     {"Australia/Lord_Howe", CalFields(2011,4,3,2,0,1),      CalFields(2011,4,2,15,30,1),    CalFields(2011,4,2,15,30,1)},
2551 
2552     {NULL,                  CalFields(0,0,0,0,0,0),         CalFields(0,0,0,0,0,0),          CalFields(0,0,0,0,0,0)}
2553 };
2554 
TestRepeatedWallTime(void)2555 void CalendarTest::TestRepeatedWallTime(void) {
2556     UErrorCode status = U_ZERO_ERROR;
2557     GregorianCalendar calGMT((const TimeZone&)*TimeZone::getGMT(), status);
2558     GregorianCalendar calDefault(status);
2559     GregorianCalendar calLast(status);
2560     GregorianCalendar calFirst(status);
2561 
2562     if (U_FAILURE(status)) {
2563         errln("Fail: Failed to create a calendar object.");
2564         return;
2565     }
2566 
2567     calLast.setRepeatedWallTimeOption(UCAL_WALLTIME_LAST);
2568     calFirst.setRepeatedWallTimeOption(UCAL_WALLTIME_FIRST);
2569 
2570     for (int32_t i = 0; RPDATA[i].tzid != NULL; i++) {
2571         char buf[32];
2572         TimeZone *tz = TimeZone::createTimeZone(RPDATA[i].tzid);
2573 
2574         // UCAL_WALLTIME_LAST
2575         status = U_ZERO_ERROR;
2576         calLast.setTimeZone(*tz);
2577         RPDATA[i].in.setTo(calLast);
2578         calGMT.setTime(calLast.getTime(status), status);
2579         CalFields outLastGMT(calGMT, status);
2580         if (U_FAILURE(status)) {
2581             errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_LAST) - ")
2582                 + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
2583         } else {
2584             if (outLastGMT != RPDATA[i].expLastGMT) {
2585                 dataerrln(UnicodeString("Fail: UCAL_WALLTIME_LAST ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
2586                     + outLastGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
2587             }
2588         }
2589 
2590         // default
2591         status = U_ZERO_ERROR;
2592         calDefault.setTimeZone(*tz);
2593         RPDATA[i].in.setTo(calDefault);
2594         calGMT.setTime(calDefault.getTime(status), status);
2595         CalFields outDefGMT(calGMT, status);
2596         if (U_FAILURE(status)) {
2597             errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (default) - ")
2598                 + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
2599         } else {
2600             if (outDefGMT != RPDATA[i].expLastGMT) {
2601                 dataerrln(UnicodeString("Fail: (default) ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
2602                     + outDefGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
2603             }
2604         }
2605 
2606         // UCAL_WALLTIME_FIRST
2607         status = U_ZERO_ERROR;
2608         calFirst.setTimeZone(*tz);
2609         RPDATA[i].in.setTo(calFirst);
2610         calGMT.setTime(calFirst.getTime(status), status);
2611         CalFields outFirstGMT(calGMT, status);
2612         if (U_FAILURE(status)) {
2613             errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_FIRST) - ")
2614                 + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
2615         } else {
2616             if (outFirstGMT != RPDATA[i].expFirstGMT) {
2617                 dataerrln(UnicodeString("Fail: UCAL_WALLTIME_FIRST ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
2618                     + outFirstGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expFirstGMT.toString(buf, sizeof(buf)) + "[GMT]");
2619             }
2620         }
2621         delete tz;
2622     }
2623 }
2624 
2625 typedef struct {
2626     const char*     tzid;
2627     const CalFields in;
2628     UBool           isValid;
2629     const CalFields expLastGMT;
2630     const CalFields expFirstGMT;
2631     const CalFields expNextAvailGMT;
2632 } SkippedWallTimeTestData;
2633 
2634 static SkippedWallTimeTestData SKDATA[] =
2635 {
2636      // Time zone           Input wall time                 valid?  WALLTIME_LAST in GMT            WALLTIME_FIRST in GMT           WALLTIME_NEXT_VALID in GMT
2637     {"America/New_York",    CalFields(2011,3,13,1,59,59),   TRUE,   CalFields(2011,3,13,6,59,59),   CalFields(2011,3,13,6,59,59),   CalFields(2011,3,13,6,59,59)},
2638     {"America/New_York",    CalFields(2011,3,13,2,0,0),     FALSE,  CalFields(2011,3,13,7,0,0),     CalFields(2011,3,13,6,0,0),     CalFields(2011,3,13,7,0,0)},
2639     {"America/New_York",    CalFields(2011,3,13,2,1,0),     FALSE,  CalFields(2011,3,13,7,1,0),     CalFields(2011,3,13,6,1,0),     CalFields(2011,3,13,7,0,0)},
2640     {"America/New_York",    CalFields(2011,3,13,2,30,0),    FALSE,  CalFields(2011,3,13,7,30,0),    CalFields(2011,3,13,6,30,0),    CalFields(2011,3,13,7,0,0)},
2641     {"America/New_York",    CalFields(2011,3,13,2,59,59),   FALSE,  CalFields(2011,3,13,7,59,59),   CalFields(2011,3,13,6,59,59),   CalFields(2011,3,13,7,0,0)},
2642     {"America/New_York",    CalFields(2011,3,13,3,0,0),     TRUE,   CalFields(2011,3,13,7,0,0),     CalFields(2011,3,13,7,0,0),     CalFields(2011,3,13,7,0,0)},
2643 
2644     {"Pacific/Apia",        CalFields(2011,12,29,23,59,59), TRUE,   CalFields(2011,12,30,9,59,59),  CalFields(2011,12,30,9,59,59),  CalFields(2011,12,30,9,59,59)},
2645     {"Pacific/Apia",        CalFields(2011,12,30,0,0,0),    FALSE,  CalFields(2011,12,30,10,0,0),   CalFields(2011,12,29,10,0,0),   CalFields(2011,12,30,10,0,0)},
2646     {"Pacific/Apia",        CalFields(2011,12,30,12,0,0),   FALSE,  CalFields(2011,12,30,22,0,0),   CalFields(2011,12,29,22,0,0),   CalFields(2011,12,30,10,0,0)},
2647     {"Pacific/Apia",        CalFields(2011,12,30,23,59,59), FALSE,  CalFields(2011,12,31,9,59,59),  CalFields(2011,12,30,9,59,59),  CalFields(2011,12,30,10,0,0)},
2648     {"Pacific/Apia",        CalFields(2011,12,31,0,0,0),    TRUE,   CalFields(2011,12,30,10,0,0),   CalFields(2011,12,30,10,0,0),   CalFields(2011,12,30,10,0,0)},
2649 
2650     {NULL,                  CalFields(0,0,0,0,0,0),         TRUE,   CalFields(0,0,0,0,0,0),         CalFields(0,0,0,0,0,0),         CalFields(0,0,0,0,0,0)}
2651 };
2652 
2653 
TestSkippedWallTime(void)2654 void CalendarTest::TestSkippedWallTime(void) {
2655     UErrorCode status = U_ZERO_ERROR;
2656     GregorianCalendar calGMT((const TimeZone&)*TimeZone::getGMT(), status);
2657     GregorianCalendar calDefault(status);
2658     GregorianCalendar calLast(status);
2659     GregorianCalendar calFirst(status);
2660     GregorianCalendar calNextAvail(status);
2661 
2662     if (U_FAILURE(status)) {
2663         errln("Fail: Failed to create a calendar object.");
2664         return;
2665     }
2666 
2667     calLast.setSkippedWallTimeOption(UCAL_WALLTIME_LAST);
2668     calFirst.setSkippedWallTimeOption(UCAL_WALLTIME_FIRST);
2669     calNextAvail.setSkippedWallTimeOption(UCAL_WALLTIME_NEXT_VALID);
2670 
2671     for (int32_t i = 0; SKDATA[i].tzid != NULL; i++) {
2672         UDate d;
2673         char buf[32];
2674         TimeZone *tz = TimeZone::createTimeZone(SKDATA[i].tzid);
2675 
2676         for (int32_t j = 0; j < 2; j++) {
2677             UBool bLenient = (j == 0);
2678 
2679             // UCAL_WALLTIME_LAST
2680             status = U_ZERO_ERROR;
2681             calLast.setLenient(bLenient);
2682             calLast.setTimeZone(*tz);
2683             SKDATA[i].in.setTo(calLast);
2684             d = calLast.getTime(status);
2685             if (bLenient || SKDATA[i].isValid) {
2686                 calGMT.setTime(d, status);
2687                 CalFields outLastGMT(calGMT, status);
2688                 if (U_FAILURE(status)) {
2689                     errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_LAST) - ")
2690                         + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2691                 } else {
2692                     if (outLastGMT != SKDATA[i].expLastGMT) {
2693                         dataerrln(UnicodeString("Fail: UCAL_WALLTIME_LAST ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
2694                             + outLastGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
2695                     }
2696                 }
2697             } else if (U_SUCCESS(status)) {
2698                 // strict, invalid wall time - must report an error
2699                 dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_LAST)") +
2700                     + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2701             }
2702 
2703             // default
2704             status = U_ZERO_ERROR;
2705             calDefault.setLenient(bLenient);
2706             calDefault.setTimeZone(*tz);
2707             SKDATA[i].in.setTo(calDefault);
2708             d = calDefault.getTime(status);
2709             if (bLenient || SKDATA[i].isValid) {
2710                 calGMT.setTime(d, status);
2711                 CalFields outDefGMT(calGMT, status);
2712                 if (U_FAILURE(status)) {
2713                     errln(UnicodeString("Fail: Failed to get/set time calDefault/calGMT (default) - ")
2714                         + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2715                 } else {
2716                     if (outDefGMT != SKDATA[i].expLastGMT) {
2717                         dataerrln(UnicodeString("Fail: (default) ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
2718                             + outDefGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
2719                     }
2720                 }
2721             } else if (U_SUCCESS(status)) {
2722                 // strict, invalid wall time - must report an error
2723                 dataerrln(UnicodeString("Fail: An error expected (default)") +
2724                     + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2725             }
2726 
2727             // UCAL_WALLTIME_FIRST
2728             status = U_ZERO_ERROR;
2729             calFirst.setLenient(bLenient);
2730             calFirst.setTimeZone(*tz);
2731             SKDATA[i].in.setTo(calFirst);
2732             d = calFirst.getTime(status);
2733             if (bLenient || SKDATA[i].isValid) {
2734                 calGMT.setTime(d, status);
2735                 CalFields outFirstGMT(calGMT, status);
2736                 if (U_FAILURE(status)) {
2737                     errln(UnicodeString("Fail: Failed to get/set time calFirst/calGMT (UCAL_WALLTIME_FIRST) - ")
2738                         + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2739                 } else {
2740                     if (outFirstGMT != SKDATA[i].expFirstGMT) {
2741                         dataerrln(UnicodeString("Fail: UCAL_WALLTIME_FIRST ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
2742                             + outFirstGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expFirstGMT.toString(buf, sizeof(buf)) + "[GMT]");
2743                     }
2744                 }
2745             } else if (U_SUCCESS(status)) {
2746                 // strict, invalid wall time - must report an error
2747                 dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_FIRST)") +
2748                     + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2749             }
2750 
2751             // UCAL_WALLTIME_NEXT_VALID
2752             status = U_ZERO_ERROR;
2753             calNextAvail.setLenient(bLenient);
2754             calNextAvail.setTimeZone(*tz);
2755             SKDATA[i].in.setTo(calNextAvail);
2756             d = calNextAvail.getTime(status);
2757             if (bLenient || SKDATA[i].isValid) {
2758                 calGMT.setTime(d, status);
2759                 CalFields outNextAvailGMT(calGMT, status);
2760                 if (U_FAILURE(status)) {
2761                     errln(UnicodeString("Fail: Failed to get/set time calNextAvail/calGMT (UCAL_WALLTIME_NEXT_VALID) - ")
2762                         + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2763                 } else {
2764                     if (outNextAvailGMT != SKDATA[i].expNextAvailGMT) {
2765                         dataerrln(UnicodeString("Fail: UCAL_WALLTIME_NEXT_VALID ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
2766                             + outNextAvailGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expNextAvailGMT.toString(buf, sizeof(buf)) + "[GMT]");
2767                     }
2768                 }
2769             } else if (U_SUCCESS(status)) {
2770                 // strict, invalid wall time - must report an error
2771                 dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_NEXT_VALID)") +
2772                     + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
2773             }
2774         }
2775 
2776         delete tz;
2777     }
2778 }
2779 
TestCloneLocale(void)2780 void CalendarTest::TestCloneLocale(void) {
2781   UErrorCode status = U_ZERO_ERROR;
2782   LocalPointer<Calendar>  cal(Calendar::createInstance(TimeZone::getGMT()->clone(),
2783                                                        Locale::createFromName("en"), status));
2784   TEST_CHECK_STATUS;
2785   Locale l0 = cal->getLocale(ULOC_VALID_LOCALE, status);
2786   TEST_CHECK_STATUS;
2787   LocalPointer<Calendar> cal2(cal->clone());
2788   Locale l = cal2->getLocale(ULOC_VALID_LOCALE, status);
2789   if(l0!=l) {
2790     errln("Error: cloned locale %s != original locale %s, status %s\n", l0.getName(), l.getName(), u_errorName(status));
2791   }
2792   TEST_CHECK_STATUS;
2793 }
2794 
setAndTestCalendar(Calendar * cal,int32_t initMonth,int32_t initDay,int32_t initYear,UErrorCode & status)2795 void CalendarTest::setAndTestCalendar(Calendar* cal, int32_t initMonth, int32_t initDay, int32_t initYear, UErrorCode& status) {
2796         cal->clear();
2797         cal->setLenient(FALSE);
2798         cal->set(initYear, initMonth, initDay);
2799         int32_t day = cal->get(UCAL_DAY_OF_MONTH, status);
2800         int32_t month = cal->get(UCAL_MONTH, status);
2801         int32_t year = cal->get(UCAL_YEAR, status);
2802         if(U_FAILURE(status))
2803             return;
2804 
2805         if(initDay != day || initMonth != month || initYear != year)
2806         {
2807             errln(" year init values:\tmonth %i\tday %i\tyear %i", initMonth, initDay, initYear);
2808             errln("values post set():\tmonth %i\tday %i\tyear %i",month, day, year);
2809         }
2810 }
2811 
setAndTestWholeYear(Calendar * cal,int32_t startYear,UErrorCode & status)2812 void CalendarTest::setAndTestWholeYear(Calendar* cal, int32_t startYear, UErrorCode& status) {
2813         for(int32_t startMonth = 0; startMonth < 12; startMonth++) {
2814             for(int32_t startDay = 1; startDay < 31; startDay++ ) {
2815                     setAndTestCalendar(cal, startMonth, startDay, startYear, status);
2816                     if(U_FAILURE(status) && startDay == 30) {
2817                         status = U_ZERO_ERROR;
2818                         continue;
2819                     }
2820                     TEST_CHECK_STATUS;
2821             }
2822         }
2823 }
2824 
2825 // =====================================================================
2826 
2827 typedef struct {
2828     int16_t  gYear;
2829     int8_t   gMon;
2830     int8_t   gDay;
2831     int16_t  uYear;
2832     int8_t   uMon;
2833     int8_t   uDay;
2834 } GregoUmmAlQuraMap;
2835 
2836 // data from
2837 // Official Umm-al-Qura calendar of SA:
2838 // home, http://www.ummulqura.org.sa/default.aspx
2839 // converter, http://www.ummulqura.org.sa/Index.aspx
2840 static const GregoUmmAlQuraMap guMappings[] = {
2841 //  gregorian,    ummAlQura
2842 //  year mo da,   year mo da
2843 //  (using 1-based months here)
2844   { 1882,11,12,   1300, 1, 1 },
2845   { 1892, 7,25,   1310, 1, 1 },
2846   { 1896, 6,12,   1314, 1, 1 },
2847   { 1898, 5,22,   1316, 1, 1 },
2848   { 1900, 4,30,   1318, 1, 1 },
2849   { 1901, 4,20,   1319, 1, 1 },
2850   { 1902, 4,10,   1320, 1, 1 },
2851   { 1903, 3,30,   1321, 1, 1 },
2852   { 1904, 3,19,   1322, 1, 1 },
2853   { 1905, 3, 8,   1323, 1, 1 },
2854   { 1906, 2,25,   1324, 1, 1 },
2855   { 1907, 2,14,   1325, 1, 1 },
2856   { 1908, 2, 4,   1326, 1, 1 },
2857   { 1909, 1,23,   1327, 1, 1 },
2858   { 1910, 1,13,   1328, 1, 1 },
2859   { 1911, 1, 2,   1329, 1, 1 },
2860   { 1911,12,22,   1330, 1, 1 },
2861   { 1912,12,10,   1331, 1, 1 },
2862   { 1913,11,30,   1332, 1, 1 },
2863   { 1914,11,19,   1333, 1, 1 },
2864   { 1915,11, 9,   1334, 1, 1 },
2865   { 1916,10,28,   1335, 1, 1 },
2866   { 1917,10,18,   1336, 1, 1 },
2867   { 1918,10, 7,   1337, 1, 1 },
2868   { 1919, 9,26,   1338, 1, 1 },
2869   { 1920, 9,14,   1339, 1, 1 },
2870   { 1921, 9, 4,   1340, 1, 1 },
2871   { 1922, 8,24,   1341, 1, 1 },
2872   { 1923, 8,14,   1342, 1, 1 },
2873   { 1924, 8, 2,   1343, 1, 1 },
2874   { 1925, 7,22,   1344, 1, 1 },
2875   { 1926, 7,11,   1345, 1, 1 },
2876   { 1927, 6,30,   1346, 1, 1 },
2877   { 1928, 6,19,   1347, 1, 1 },
2878   { 1929, 6, 9,   1348, 1, 1 },
2879   { 1930, 5,29,   1349, 1, 1 },
2880   { 1931, 5,19,   1350, 1, 1 },
2881   { 1932, 5, 7,   1351, 1, 1 },
2882   { 1933, 4,26,   1352, 1, 1 },
2883   { 1934, 4,15,   1353, 1, 1 },
2884   { 1935, 4, 5,   1354, 1, 1 },
2885   { 1936, 3,24,   1355, 1, 1 },
2886   { 1937, 3,14,   1356, 1, 1 },
2887   { 1938, 3, 4,   1357, 1, 1 },
2888   { 1939, 2,21,   1358, 1, 1 },
2889   { 1940, 2,10,   1359, 1, 1 },
2890   { 1941, 1,29,   1360, 1, 1 },
2891   { 1942, 1,18,   1361, 1, 1 },
2892   { 1943, 1, 8,   1362, 1, 1 },
2893   { 1943,12,28,   1363, 1, 1 },
2894   { 1944,12,17,   1364, 1, 1 },
2895   { 1945,12, 6,   1365, 1, 1 },
2896   { 1946,11,25,   1366, 1, 1 },
2897   { 1947,11,14,   1367, 1, 1 },
2898   { 1948,11, 3,   1368, 1, 1 },
2899   { 1949,10,23,   1369, 1, 1 },
2900   { 1950,10,13,   1370, 1, 1 },
2901   { 1951,10, 3,   1371, 1, 1 },
2902   { 1952, 9,21,   1372, 1, 1 },
2903   { 1953, 9,10,   1373, 1, 1 },
2904   { 1954, 8,30,   1374, 1, 1 },
2905   { 1955, 8,19,   1375, 1, 1 },
2906   { 1956, 8, 8,   1376, 1, 1 },
2907   { 1957, 7,29,   1377, 1, 1 },
2908   { 1958, 7,18,   1378, 1, 1 },
2909   { 1959, 7, 8,   1379, 1, 1 },
2910   { 1960, 6,26,   1380, 1, 1 },
2911   { 1961, 6,15,   1381, 1, 1 },
2912   { 1962, 6, 4,   1382, 1, 1 },
2913   { 1963, 5,24,   1383, 1, 1 },
2914   { 1964, 5,13,   1384, 1, 1 },
2915   { 1965, 5, 3,   1385, 1, 1 },
2916   { 1966, 4,22,   1386, 1, 1 },
2917   { 1967, 4,11,   1387, 1, 1 },
2918   { 1968, 3,30,   1388, 1, 1 },
2919   { 1969, 3,19,   1389, 1, 1 },
2920   { 1970, 3, 9,   1390, 1, 1 },
2921   { 1971, 2,27,   1391, 1, 1 },
2922   { 1972, 2,16,   1392, 1, 1 },
2923   { 1973, 2, 5,   1393, 1, 1 },
2924   { 1974, 1,25,   1394, 1, 1 },
2925   { 1975, 1,14,   1395, 1, 1 },
2926   { 1976, 1, 3,   1396, 1, 1 },
2927   { 1976,12,22,   1397, 1, 1 },
2928   { 1977,12,12,   1398, 1, 1 },
2929   { 1978,12, 1,   1399, 1, 1 },
2930   { 1979,11,21,   1400, 1, 1 },
2931   { 1980,11, 9,   1401, 1, 1 },
2932   { 1981,10,29,   1402, 1, 1 },
2933   { 1982,10,18,   1403, 1, 1 },
2934   { 1983,10, 8,   1404, 1, 1 },
2935   { 1984, 9,26,   1405, 1, 1 },
2936   { 1985, 9,16,   1406, 1, 1 },
2937   { 1986, 9, 6,   1407, 1, 1 },
2938   { 1987, 8,26,   1408, 1, 1 },
2939   { 1988, 8,14,   1409, 1, 1 },
2940   { 1989, 8, 3,   1410, 1, 1 },
2941   { 1990, 7,23,   1411, 1, 1 },
2942   { 1991, 7,13,   1412, 1, 1 },
2943   { 1992, 7, 2,   1413, 1, 1 },
2944   { 1993, 6,21,   1414, 1, 1 },
2945   { 1994, 6,11,   1415, 1, 1 },
2946   { 1995, 5,31,   1416, 1, 1 },
2947   { 1996, 5,19,   1417, 1, 1 },
2948   { 1997, 5, 8,   1418, 1, 1 },
2949   { 1998, 4,28,   1419, 1, 1 },
2950   { 1999, 4,17,   1420, 1, 1 },
2951   { 1999, 5,16,   1420, 2, 1 },
2952   { 1999, 6,15,   1420, 3, 1 },
2953   { 1999, 7,14,   1420, 4, 1 },
2954   { 1999, 8,12,   1420, 5, 1 },
2955   { 1999, 9,11,   1420, 6, 1 },
2956   { 1999,10,10,   1420, 7, 1 },
2957   { 1999,11, 9,   1420, 8, 1 },
2958   { 1999,12, 9,   1420, 9, 1 },
2959   { 2000, 1, 8,   1420,10, 1 },
2960   { 2000, 2, 7,   1420,11, 1 },
2961   { 2000, 3, 7,   1420,12, 1 },
2962   { 2000, 4, 6,   1421, 1, 1 },
2963   { 2000, 5, 5,   1421, 2, 1 },
2964   { 2000, 6, 3,   1421, 3, 1 },
2965   { 2000, 7, 3,   1421, 4, 1 },
2966   { 2000, 8, 1,   1421, 5, 1 },
2967   { 2000, 8,30,   1421, 6, 1 },
2968   { 2000, 9,28,   1421, 7, 1 },
2969   { 2000,10,28,   1421, 8, 1 },
2970   { 2000,11,27,   1421, 9, 1 },
2971   { 2000,12,27,   1421,10, 1 },
2972   { 2001, 1,26,   1421,11, 1 },
2973   { 2001, 2,24,   1421,12, 1 },
2974   { 2001, 3,26,   1422, 1, 1 },
2975   { 2001, 4,25,   1422, 2, 1 },
2976   { 2001, 5,24,   1422, 3, 1 },
2977   { 2001, 6,22,   1422, 4, 1 },
2978   { 2001, 7,22,   1422, 5, 1 },
2979   { 2001, 8,20,   1422, 6, 1 },
2980   { 2001, 9,18,   1422, 7, 1 },
2981   { 2001,10,17,   1422, 8, 1 },
2982   { 2001,11,16,   1422, 9, 1 },
2983   { 2001,12,16,   1422,10, 1 },
2984   { 2002, 1,15,   1422,11, 1 },
2985   { 2002, 2,13,   1422,12, 1 },
2986   { 2002, 3,15,   1423, 1, 1 },
2987   { 2002, 4,14,   1423, 2, 1 },
2988   { 2002, 5,13,   1423, 3, 1 },
2989   { 2002, 6,12,   1423, 4, 1 },
2990   { 2002, 7,11,   1423, 5, 1 },
2991   { 2002, 8,10,   1423, 6, 1 },
2992   { 2002, 9, 8,   1423, 7, 1 },
2993   { 2002,10, 7,   1423, 8, 1 },
2994   { 2002,11, 6,   1423, 9, 1 },
2995   { 2002,12, 5,   1423,10, 1 },
2996   { 2003, 1, 4,   1423,11, 1 },
2997   { 2003, 2, 2,   1423,12, 1 },
2998   { 2003, 3, 4,   1424, 1, 1 },
2999   { 2003, 4, 3,   1424, 2, 1 },
3000   { 2003, 5, 2,   1424, 3, 1 },
3001   { 2003, 6, 1,   1424, 4, 1 },
3002   { 2003, 7, 1,   1424, 5, 1 },
3003   { 2003, 7,30,   1424, 6, 1 },
3004   { 2003, 8,29,   1424, 7, 1 },
3005   { 2003, 9,27,   1424, 8, 1 },
3006   { 2003,10,26,   1424, 9, 1 },
3007   { 2003,11,25,   1424,10, 1 },
3008   { 2003,12,24,   1424,11, 1 },
3009   { 2004, 1,23,   1424,12, 1 },
3010   { 2004, 2,21,   1425, 1, 1 },
3011   { 2004, 3,22,   1425, 2, 1 },
3012   { 2004, 4,20,   1425, 3, 1 },
3013   { 2004, 5,20,   1425, 4, 1 },
3014   { 2004, 6,19,   1425, 5, 1 },
3015   { 2004, 7,18,   1425, 6, 1 },
3016   { 2004, 8,17,   1425, 7, 1 },
3017   { 2004, 9,15,   1425, 8, 1 },
3018   { 2004,10,15,   1425, 9, 1 },
3019   { 2004,11,14,   1425,10, 1 },
3020   { 2004,12,13,   1425,11, 1 },
3021   { 2005, 1,12,   1425,12, 1 },
3022   { 2005, 2,10,   1426, 1, 1 },
3023   { 2005, 3,11,   1426, 2, 1 },
3024   { 2005, 4,10,   1426, 3, 1 },
3025   { 2005, 5, 9,   1426, 4, 1 },
3026   { 2005, 6, 8,   1426, 5, 1 },
3027   { 2005, 7, 7,   1426, 6, 1 },
3028   { 2005, 8, 6,   1426, 7, 1 },
3029   { 2005, 9, 5,   1426, 8, 1 },
3030   { 2005,10, 4,   1426, 9, 1 },
3031   { 2005,11, 3,   1426,10, 1 },
3032   { 2005,12, 3,   1426,11, 1 },
3033   { 2006, 1, 1,   1426,12, 1 },
3034   { 2006, 1,31,   1427, 1, 1 },
3035   { 2006, 3, 1,   1427, 2, 1 },
3036   { 2006, 3,30,   1427, 3, 1 },
3037   { 2006, 4,29,   1427, 4, 1 },
3038   { 2006, 5,28,   1427, 5, 1 },
3039   { 2006, 6,27,   1427, 6, 1 },
3040   { 2006, 7,26,   1427, 7, 1 },
3041   { 2006, 8,25,   1427, 8, 1 },
3042   { 2006, 9,24,   1427, 9, 1 },
3043   { 2006,10,23,   1427,10, 1 },
3044   { 2006,11,22,   1427,11, 1 },
3045   { 2006,12,22,   1427,12, 1 },
3046   { 2007, 1,20,   1428, 1, 1 },
3047   { 2007, 2,19,   1428, 2, 1 },
3048   { 2007, 3,20,   1428, 3, 1 },
3049   { 2007, 4,18,   1428, 4, 1 },
3050   { 2007, 5,18,   1428, 5, 1 },
3051   { 2007, 6,16,   1428, 6, 1 },
3052   { 2007, 7,15,   1428, 7, 1 },
3053   { 2007, 8,14,   1428, 8, 1 },
3054   { 2007, 9,13,   1428, 9, 1 },
3055   { 2007,10,13,   1428,10, 1 },
3056   { 2007,11,11,   1428,11, 1 },
3057   { 2007,12,11,   1428,12, 1 },
3058   { 2008, 1,10,   1429, 1, 1 },
3059   { 2008, 2, 8,   1429, 2, 1 },
3060   { 2008, 3, 9,   1429, 3, 1 },
3061   { 2008, 4, 7,   1429, 4, 1 },
3062   { 2008, 5, 6,   1429, 5, 1 },
3063   { 2008, 6, 5,   1429, 6, 1 },
3064   { 2008, 7, 4,   1429, 7, 1 },
3065   { 2008, 8, 2,   1429, 8, 1 },
3066   { 2008, 9, 1,   1429, 9, 1 },
3067   { 2008,10, 1,   1429,10, 1 },
3068   { 2008,10,30,   1429,11, 1 },
3069   { 2008,11,29,   1429,12, 1 },
3070   { 2008,12,29,   1430, 1, 1 },
3071   { 2009, 1,27,   1430, 2, 1 },
3072   { 2009, 2,26,   1430, 3, 1 },
3073   { 2009, 3,28,   1430, 4, 1 },
3074   { 2009, 4,26,   1430, 5, 1 },
3075   { 2009, 5,25,   1430, 6, 1 },
3076   { 2009, 6,24,   1430, 7, 1 },
3077   { 2009, 7,23,   1430, 8, 1 },
3078   { 2009, 8,22,   1430, 9, 1 },
3079   { 2009, 9,20,   1430,10, 1 },
3080   { 2009,10,20,   1430,11, 1 },
3081   { 2009,11,18,   1430,12, 1 },
3082   { 2009,12,18,   1431, 1, 1 },
3083   { 2010, 1,16,   1431, 2, 1 },
3084   { 2010, 2,15,   1431, 3, 1 },
3085   { 2010, 3,17,   1431, 4, 1 },
3086   { 2010, 4,15,   1431, 5, 1 },
3087   { 2010, 5,15,   1431, 6, 1 },
3088   { 2010, 6,13,   1431, 7, 1 },
3089   { 2010, 7,13,   1431, 8, 1 },
3090   { 2010, 8,11,   1431, 9, 1 },
3091   { 2010, 9,10,   1431,10, 1 },
3092   { 2010,10, 9,   1431,11, 1 },
3093   { 2010,11, 7,   1431,12, 1 },
3094   { 2010,12, 7,   1432, 1, 1 },
3095   { 2011, 1, 5,   1432, 2, 1 },
3096   { 2011, 2, 4,   1432, 3, 1 },
3097   { 2011, 3, 6,   1432, 4, 1 },
3098   { 2011, 4, 5,   1432, 5, 1 },
3099   { 2011, 5, 4,   1432, 6, 1 },
3100   { 2011, 6, 3,   1432, 7, 1 },
3101   { 2011, 7, 2,   1432, 8, 1 },
3102   { 2011, 8, 1,   1432, 9, 1 },
3103   { 2011, 8,30,   1432,10, 1 },
3104   { 2011, 9,29,   1432,11, 1 },
3105   { 2011,10,28,   1432,12, 1 },
3106   { 2011,11,26,   1433, 1, 1 },
3107   { 2011,12,26,   1433, 2, 1 },
3108   { 2012, 1,24,   1433, 3, 1 },
3109   { 2012, 2,23,   1433, 4, 1 },
3110   { 2012, 3,24,   1433, 5, 1 },
3111   { 2012, 4,22,   1433, 6, 1 },
3112   { 2012, 5,22,   1433, 7, 1 },
3113   { 2012, 6,21,   1433, 8, 1 },
3114   { 2012, 7,20,   1433, 9, 1 },
3115   { 2012, 8,19,   1433,10, 1 },
3116   { 2012, 9,17,   1433,11, 1 },
3117   { 2012,10,17,   1433,12, 1 },
3118   { 2012,11,15,   1434, 1, 1 },
3119   { 2012,12,14,   1434, 2, 1 },
3120   { 2013, 1,13,   1434, 3, 1 },
3121   { 2013, 2,11,   1434, 4, 1 },
3122   { 2013, 3,13,   1434, 5, 1 },
3123   { 2013, 4,11,   1434, 6, 1 },
3124   { 2013, 5,11,   1434, 7, 1 },
3125   { 2013, 6,10,   1434, 8, 1 },
3126   { 2013, 7, 9,   1434, 9, 1 },
3127   { 2013, 8, 8,   1434,10, 1 },
3128   { 2013, 9, 7,   1434,11, 1 },
3129   { 2013,10, 6,   1434,12, 1 },
3130   { 2013,11, 4,   1435, 1, 1 },
3131   { 2013,12, 4,   1435, 2, 1 },
3132   { 2014, 1, 2,   1435, 3, 1 },
3133   { 2014, 2, 1,   1435, 4, 1 },
3134   { 2014, 3, 2,   1435, 5, 1 },
3135   { 2014, 4, 1,   1435, 6, 1 },
3136   { 2014, 4,30,   1435, 7, 1 },
3137   { 2014, 5,30,   1435, 8, 1 },
3138   { 2014, 6,28,   1435, 9, 1 },
3139   { 2014, 7,28,   1435,10, 1 },
3140   { 2014, 8,27,   1435,11, 1 },
3141   { 2014, 9,25,   1435,12, 1 },
3142   { 2014,10,25,   1436, 1, 1 },
3143   { 2014,11,23,   1436, 2, 1 },
3144   { 2014,12,23,   1436, 3, 1 },
3145   { 2015, 1,21,   1436, 4, 1 },
3146   { 2015, 2,20,   1436, 5, 1 },
3147   { 2015, 3,21,   1436, 6, 1 },
3148   { 2015, 4,20,   1436, 7, 1 },
3149   { 2015, 5,19,   1436, 8, 1 },
3150   { 2015, 6,18,   1436, 9, 1 },
3151   { 2015, 7,17,   1436,10, 1 },
3152   { 2015, 8,16,   1436,11, 1 },
3153   { 2015, 9,14,   1436,12, 1 },
3154   { 2015,10,14,   1437, 1, 1 },
3155   { 2015,11,13,   1437, 2, 1 },
3156   { 2015,12,12,   1437, 3, 1 },
3157   { 2016, 1,11,   1437, 4, 1 },
3158   { 2016, 2,10,   1437, 5, 1 },
3159   { 2016, 3,10,   1437, 6, 1 },
3160   { 2016, 4, 8,   1437, 7, 1 },
3161   { 2016, 5, 8,   1437, 8, 1 },
3162   { 2016, 6, 6,   1437, 9, 1 },
3163   { 2016, 7, 6,   1437,10, 1 },
3164   { 2016, 8, 4,   1437,11, 1 },
3165   { 2016, 9, 2,   1437,12, 1 },
3166   { 2016,10, 2,   1438, 1, 1 },
3167   { 2016,11, 1,   1438, 2, 1 },
3168   { 2016,11,30,   1438, 3, 1 },
3169   { 2016,12,30,   1438, 4, 1 },
3170   { 2017, 1,29,   1438, 5, 1 },
3171   { 2017, 2,28,   1438, 6, 1 },
3172   { 2017, 3,29,   1438, 7, 1 },
3173   { 2017, 4,27,   1438, 8, 1 },
3174   { 2017, 5,27,   1438, 9, 1 },
3175   { 2017, 6,25,   1438,10, 1 },
3176   { 2017, 7,24,   1438,11, 1 },
3177   { 2017, 8,23,   1438,12, 1 },
3178   { 2017, 9,21,   1439, 1, 1 },
3179   { 2017,10,21,   1439, 2, 1 },
3180   { 2017,11,19,   1439, 3, 1 },
3181   { 2017,12,19,   1439, 4, 1 },
3182   { 2018, 1,18,   1439, 5, 1 },
3183   { 2018, 2,17,   1439, 6, 1 },
3184   { 2018, 3,18,   1439, 7, 1 },
3185   { 2018, 4,17,   1439, 8, 1 },
3186   { 2018, 5,16,   1439, 9, 1 },
3187   { 2018, 6,15,   1439,10, 1 },
3188   { 2018, 7,14,   1439,11, 1 },
3189   { 2018, 8,12,   1439,12, 1 },
3190   { 2018, 9,11,   1440, 1, 1 },
3191   { 2019, 8,31,   1441, 1, 1 },
3192   { 2020, 8,20,   1442, 1, 1 },
3193   { 2021, 8, 9,   1443, 1, 1 },
3194   { 2022, 7,30,   1444, 1, 1 },
3195   { 2023, 7,19,   1445, 1, 1 },
3196   { 2024, 7, 7,   1446, 1, 1 },
3197   { 2025, 6,26,   1447, 1, 1 },
3198   { 2026, 6,16,   1448, 1, 1 },
3199   { 2027, 6, 6,   1449, 1, 1 },
3200   { 2028, 5,25,   1450, 1, 1 },
3201   { 2029, 5,14,   1451, 1, 1 },
3202   { 2030, 5, 4,   1452, 1, 1 },
3203   { 2031, 4,23,   1453, 1, 1 },
3204   { 2032, 4,11,   1454, 1, 1 },
3205   { 2033, 4, 1,   1455, 1, 1 },
3206   { 2034, 3,22,   1456, 1, 1 },
3207   { 2035, 3,11,   1457, 1, 1 },
3208   { 2036, 2,29,   1458, 1, 1 },
3209   { 2037, 2,17,   1459, 1, 1 },
3210   { 2038, 2, 6,   1460, 1, 1 },
3211   { 2039, 1,26,   1461, 1, 1 },
3212   { 2040, 1,15,   1462, 1, 1 },
3213   { 2041, 1, 4,   1463, 1, 1 },
3214   { 2041,12,25,   1464, 1, 1 },
3215   { 2042,12,14,   1465, 1, 1 },
3216   { 2043,12, 3,   1466, 1, 1 },
3217   { 2044,11,21,   1467, 1, 1 },
3218   { 2045,11,11,   1468, 1, 1 },
3219   { 2046,10,31,   1469, 1, 1 },
3220   { 2047,10,21,   1470, 1, 1 },
3221   { 2048,10, 9,   1471, 1, 1 },
3222   { 2049, 9,29,   1472, 1, 1 },
3223   { 2050, 9,18,   1473, 1, 1 },
3224   { 2051, 9, 7,   1474, 1, 1 },
3225   { 2052, 8,26,   1475, 1, 1 },
3226   { 2053, 8,15,   1476, 1, 1 },
3227   { 2054, 8, 5,   1477, 1, 1 },
3228   { 2055, 7,26,   1478, 1, 1 },
3229   { 2056, 7,14,   1479, 1, 1 },
3230   { 2057, 7, 3,   1480, 1, 1 },
3231   { 2058, 6,22,   1481, 1, 1 },
3232   { 2059, 6,11,   1482, 1, 1 },
3233   { 2061, 5,21,   1484, 1, 1 },
3234   { 2063, 4,30,   1486, 1, 1 },
3235   { 2065, 4, 7,   1488, 1, 1 },
3236   { 2067, 3,17,   1490, 1, 1 },
3237   { 2069, 2,23,   1492, 1, 1 },
3238   { 2071, 2, 2,   1494, 1, 1 },
3239   { 2073, 1,10,   1496, 1, 1 },
3240   { 2074,12,20,   1498, 1, 1 },
3241   { 2076,11,28,   1500, 1, 1 },
3242   {    0, 0, 0,      0, 0, 0 }, // terminator
3243 };
3244 
3245 static const UChar zoneSA[] = {0x41,0x73,0x69,0x61,0x2F,0x52,0x69,0x79,0x61,0x64,0x68,0}; // "Asia/Riyadh"
3246 
TestIslamicUmAlQura()3247 void CalendarTest::TestIslamicUmAlQura() {
3248 
3249     UErrorCode status = U_ZERO_ERROR;
3250     Locale umalquraLoc("ar_SA@calendar=islamic-umalqura");
3251     Locale gregoLoc("ar_SA@calendar=gregorian");
3252     TimeZone* tzSA = TimeZone::createTimeZone(UnicodeString(TRUE, zoneSA, -1));
3253     Calendar* tstCal = Calendar::createInstance(*((const TimeZone *)tzSA), umalquraLoc, status);
3254     Calendar* gregCal = Calendar::createInstance(*((const TimeZone *)tzSA), gregoLoc, status);
3255 
3256     IslamicCalendar* iCal = (IslamicCalendar*)tstCal;
3257     if(strcmp(iCal->getType(), "islamic-umalqura") != 0) {
3258         errln("wrong type of calendar created - %s", iCal->getType());
3259     }
3260 
3261     int32_t firstYear = 1318;
3262     int32_t lastYear = 1368;    // just enough to be pretty sure
3263     //int32_t lastYear = 1480;    // the whole shootin' match
3264 
3265     tstCal->clear();
3266     tstCal->setLenient(FALSE);
3267 
3268     int32_t day=0, month=0, year=0, initDay = 27, initMonth = IslamicCalendar::RAJAB, initYear = 1434;
3269 
3270     for( int32_t startYear = firstYear; startYear <= lastYear; startYear++) {
3271         setAndTestWholeYear(tstCal, startYear, status);
3272         status = U_ZERO_ERROR;
3273     }
3274 
3275     initMonth = IslamicCalendar::RABI_2;
3276     initDay = 5;
3277     int32_t loopCnt = 25;
3278     tstCal->clear();
3279     setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
3280     TEST_CHECK_STATUS;
3281 
3282     for(int x=1; x<=loopCnt; x++) {
3283         day = tstCal->get(UCAL_DAY_OF_MONTH,status);
3284         month = tstCal->get(UCAL_MONTH,status);
3285         year = tstCal->get(UCAL_YEAR,status);
3286         TEST_CHECK_STATUS;
3287         tstCal->roll(UCAL_DAY_OF_MONTH, (UBool)TRUE, status);
3288         TEST_CHECK_STATUS;
3289     }
3290 
3291     if(day != (initDay + loopCnt - 1) || month != IslamicCalendar::RABI_2 || year != 1434)
3292       errln("invalid values for RABI_2 date after roll of %d", loopCnt);
3293 
3294     status = U_ZERO_ERROR;
3295     tstCal->clear();
3296     initMonth = 2;
3297     initDay = 30;
3298     setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
3299     if(U_SUCCESS(status)) {
3300         errln("error NOT detected status %i",status);
3301         errln("      init values:\tmonth %i\tday %i\tyear %i", initMonth, initDay, initYear);
3302         int32_t day = tstCal->get(UCAL_DAY_OF_MONTH, status);
3303         int32_t month = tstCal->get(UCAL_MONTH, status);
3304         int32_t year = tstCal->get(UCAL_YEAR, status);
3305         errln("values post set():\tmonth %i\tday %i\tyear %i",month, day, year);
3306     }
3307 
3308     status = U_ZERO_ERROR;
3309     tstCal->clear();
3310     initMonth = 3;
3311     initDay = 30;
3312     setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
3313     TEST_CHECK_STATUS;
3314 
3315     SimpleDateFormat* formatter = new SimpleDateFormat("yyyy-MM-dd", Locale::getUS(), status);
3316     UDate date = formatter->parse("1975-05-06", status);
3317     Calendar* is_cal = Calendar::createInstance(umalquraLoc, status);
3318     is_cal->setTime(date, status);
3319     int32_t is_day = is_cal->get(UCAL_DAY_OF_MONTH,status);
3320     int32_t is_month = is_cal->get(UCAL_MONTH,status);
3321     int32_t is_year = is_cal->get(UCAL_YEAR,status);
3322     TEST_CHECK_STATUS;
3323     if(is_day != 24 || is_month != IslamicCalendar::RABI_2 || is_year != 1395)
3324         errln("unexpected conversion date month %i not %i or day %i not 24 or year %i not 1395", is_month, IslamicCalendar::RABI_2, is_day, is_year);
3325 
3326     UDate date2 = is_cal->getTime(status);
3327     TEST_CHECK_STATUS;
3328     if(date2 != date) {
3329         errln("before(%f) and after(%f) dates don't match up!",date, date2);
3330     }
3331 
3332     // check against data
3333     const GregoUmmAlQuraMap* guMapPtr;
3334     gregCal->clear();
3335     tstCal->clear();
3336     for (guMapPtr = guMappings; guMapPtr->gYear != 0; guMapPtr++) {
3337         status = U_ZERO_ERROR;
3338         gregCal->set(guMapPtr->gYear, guMapPtr->gMon - 1, guMapPtr->gDay, 12, 0);
3339         date = gregCal->getTime(status);
3340         tstCal->setTime(date, status);
3341         int32_t uYear = tstCal->get(UCAL_YEAR, status);
3342         int32_t uMon = tstCal->get(UCAL_MONTH, status) + 1;
3343         int32_t uDay = tstCal->get(UCAL_DATE, status);
3344         if(U_FAILURE(status)) {
3345             errln("For gregorian %4d-%02d-%02d, get status %s",
3346                     guMapPtr->gYear, guMapPtr->gMon, guMapPtr->gDay, u_errorName(status) );
3347         } else if (uYear != guMapPtr->uYear || uMon != guMapPtr->uMon || uDay != guMapPtr->uDay) {
3348             errln("For gregorian %4d-%02d-%02d, expect umalqura %4d-%02d-%02d, get %4d-%02d-%02d",
3349                     guMapPtr->gYear, guMapPtr->gMon, guMapPtr->gDay,
3350                     guMapPtr->uYear, guMapPtr->uMon, guMapPtr->uDay, uYear, uMon, uDay );
3351         }
3352     }
3353 
3354     delete is_cal;
3355     delete formatter;
3356     delete gregCal;
3357     delete tstCal;
3358     delete tzSA;
3359 }
3360 
TestIslamicTabularDates()3361 void CalendarTest::TestIslamicTabularDates() {
3362     UErrorCode status = U_ZERO_ERROR;
3363     Locale islamicLoc("ar_SA@calendar=islamic-civil");
3364     Locale tblaLoc("ar_SA@calendar=islamic-tbla");
3365     SimpleDateFormat* formatter = new SimpleDateFormat("yyyy-MM-dd", Locale::getUS(), status);
3366     UDate date = formatter->parse("1975-05-06", status);
3367 
3368     Calendar* tstCal = Calendar::createInstance(islamicLoc, status);
3369     tstCal->setTime(date, status);
3370     int32_t is_day = tstCal->get(UCAL_DAY_OF_MONTH,status);
3371     int32_t is_month = tstCal->get(UCAL_MONTH,status);
3372     int32_t is_year = tstCal->get(UCAL_YEAR,status);
3373     TEST_CHECK_STATUS;
3374     delete tstCal;
3375 
3376     tstCal = Calendar::createInstance(tblaLoc, status);
3377     tstCal->setTime(date, status);
3378     int32_t tbla_day = tstCal->get(UCAL_DAY_OF_MONTH,status);
3379     int32_t tbla_month = tstCal->get(UCAL_MONTH,status);
3380     int32_t tbla_year = tstCal->get(UCAL_YEAR,status);
3381     TEST_CHECK_STATUS;
3382 
3383     if(tbla_month != is_month || tbla_year != is_year)
3384         errln("unexpected difference between islamic and tbla month %d : %d and/or year %d : %d",tbla_month,is_month,tbla_year,is_year);
3385 
3386     if(tbla_day - is_day != 1)
3387         errln("unexpected day difference between islamic and tbla: %d : %d ",tbla_day,is_day);
3388     delete tstCal;
3389     delete formatter;
3390 }
3391 
TestHebrewMonthValidation()3392 void CalendarTest::TestHebrewMonthValidation() {
3393     UErrorCode status = U_ZERO_ERROR;
3394     LocalPointer<Calendar>  cal(Calendar::createInstance(Locale::createFromName("he_IL@calendar=hebrew"), status));
3395     if (failure(status, "Calendar::createInstance, locale:he_IL@calendar=hebrew", TRUE)) return;
3396     Calendar *pCal = cal.getAlias();
3397 
3398     UDate d;
3399     pCal->setLenient(FALSE);
3400 
3401     // 5776 is a leap year and has month Adar I
3402     pCal->set(5776, HebrewCalendar::ADAR_1, 1);
3403     d = pCal->getTime(status);
3404     if (U_FAILURE(status)) {
3405         errln("Fail: 5776 Adar I 1 is a valid date.");
3406     }
3407     status = U_ZERO_ERROR;
3408 
3409     // 5777 is NOT a lear year and does not have month Adar I
3410     pCal->set(5777, HebrewCalendar::ADAR_1, 1);
3411     d = pCal->getTime(status);
3412     (void)d;
3413     if (status == U_ILLEGAL_ARGUMENT_ERROR) {
3414         logln("Info: U_ILLEGAL_ARGUMENT_ERROR, because 5777 Adar I 1 is not a valid date.");
3415     } else {
3416         errln("Fail: U_ILLEGAL_ARGUMENT_ERROR should be set for input date 5777 Adar I 1.");
3417     }
3418 }
3419 
TestWeekData()3420 void CalendarTest::TestWeekData() {
3421     // Each line contains two locales using the same set of week rule data.
3422     const char* LOCALE_PAIRS[] = {
3423         "en",       "en_US",
3424         "de",       "de_DE",
3425         "de_DE",    "en_DE",
3426         "en_GB",    "und_GB",
3427         "ar_EG",    "en_EG",
3428         "ar_SA",    "fr_SA",
3429         0
3430     };
3431 
3432     UErrorCode status;
3433 
3434     for (int32_t i = 0; LOCALE_PAIRS[i] != 0; i += 2) {
3435         status = U_ZERO_ERROR;
3436         LocalPointer<Calendar>  cal1(Calendar::createInstance(LOCALE_PAIRS[i], status));
3437         LocalPointer<Calendar>  cal2(Calendar::createInstance(LOCALE_PAIRS[i + 1], status));
3438         TEST_CHECK_STATUS_LOCALE(LOCALE_PAIRS[i]);
3439 
3440         // First day of week
3441         UCalendarDaysOfWeek dow1 = cal1->getFirstDayOfWeek(status);
3442         UCalendarDaysOfWeek dow2 = cal2->getFirstDayOfWeek(status);
3443         TEST_CHECK_STATUS;
3444         TEST_ASSERT(dow1 == dow2);
3445 
3446         // Minimum days in first week
3447         uint8_t minDays1 = cal1->getMinimalDaysInFirstWeek();
3448         uint8_t minDays2 = cal2->getMinimalDaysInFirstWeek();
3449         TEST_ASSERT(minDays1 == minDays2);
3450 
3451         // Weekdays and Weekends
3452         for (int32_t d = UCAL_SUNDAY; d <= UCAL_SATURDAY; d++) {
3453             status = U_ZERO_ERROR;
3454             UCalendarWeekdayType wdt1 = cal1->getDayOfWeekType((UCalendarDaysOfWeek)d, status);
3455             UCalendarWeekdayType wdt2 = cal2->getDayOfWeekType((UCalendarDaysOfWeek)d, status);
3456             TEST_CHECK_STATUS;
3457             TEST_ASSERT(wdt1 == wdt2);
3458         }
3459     }
3460 }
3461 
3462 typedef struct {
3463     const char* zone;
3464     const CalFields base;
3465     int32_t deltaDays;
3466     UCalendarWallTimeOption skippedWTOpt;
3467     const CalFields expected;
3468 } TestAddAcrossZoneTransitionData;
3469 
3470 static const TestAddAcrossZoneTransitionData AAZTDATA[] =
3471 {
3472     // Time zone                Base wall time                      day(s)  Skipped time options
3473     //                          Expected wall time
3474 
3475     // Add 1 day, from the date before DST transition
3476     {"America/Los_Angeles",     CalFields(2014,3,8,1,59,59,999),    1,      UCAL_WALLTIME_FIRST,
3477                                 CalFields(2014,3,9,1,59,59,999)},
3478 
3479     {"America/Los_Angeles",     CalFields(2014,3,8,1,59,59,999),    1,      UCAL_WALLTIME_LAST,
3480                                 CalFields(2014,3,9,1,59,59,999)},
3481 
3482     {"America/Los_Angeles",     CalFields(2014,3,8,1,59,59,999),    1,      UCAL_WALLTIME_NEXT_VALID,
3483                                 CalFields(2014,3,9,1,59,59,999)},
3484 
3485 
3486     {"America/Los_Angeles",     CalFields(2014,3,8,2,0,0,0),        1,      UCAL_WALLTIME_FIRST,
3487                                 CalFields(2014,3,9,1,0,0,0)},
3488 
3489     {"America/Los_Angeles",     CalFields(2014,3,8,2,0,0,0),        1,      UCAL_WALLTIME_LAST,
3490                                 CalFields(2014,3,9,3,0,0,0)},
3491 
3492     {"America/Los_Angeles",     CalFields(2014,3,8,2,0,0,0),        1,      UCAL_WALLTIME_NEXT_VALID,
3493                                 CalFields(2014,3,9,3,0,0,0)},
3494 
3495 
3496     {"America/Los_Angeles",     CalFields(2014,3,8,2,30,0,0),       1,      UCAL_WALLTIME_FIRST,
3497                                 CalFields(2014,3,9,1,30,0,0)},
3498 
3499     {"America/Los_Angeles",     CalFields(2014,3,8,2,30,0,0),       1,      UCAL_WALLTIME_LAST,
3500                                 CalFields(2014,3,9,3,30,0,0)},
3501 
3502     {"America/Los_Angeles",     CalFields(2014,3,8,2,30,0,0),       1,      UCAL_WALLTIME_NEXT_VALID,
3503                                 CalFields(2014,3,9,3,0,0,0)},
3504 
3505 
3506     {"America/Los_Angeles",     CalFields(2014,3,8,3,0,0,0),        1,      UCAL_WALLTIME_FIRST,
3507                                 CalFields(2014,3,9,3,0,0,0)},
3508 
3509     {"America/Los_Angeles",     CalFields(2014,3,8,3,0,0,0),        1,      UCAL_WALLTIME_LAST,
3510                                 CalFields(2014,3,9,3,0,0,0)},
3511 
3512     {"America/Los_Angeles",     CalFields(2014,3,8,3,0,0,0),        1,      UCAL_WALLTIME_NEXT_VALID,
3513                                 CalFields(2014,3,9,3,0,0,0)},
3514 
3515     // Subtract 1 day, from one day after DST transition
3516     {"America/Los_Angeles",     CalFields(2014,3,10,1,59,59,999),   -1,     UCAL_WALLTIME_FIRST,
3517                                 CalFields(2014,3,9,1,59,59,999)},
3518 
3519     {"America/Los_Angeles",     CalFields(2014,3,10,1,59,59,999),   -1,     UCAL_WALLTIME_LAST,
3520                                 CalFields(2014,3,9,1,59,59,999)},
3521 
3522     {"America/Los_Angeles",     CalFields(2014,3,10,1,59,59,999),   -1,     UCAL_WALLTIME_NEXT_VALID,
3523                                 CalFields(2014,3,9,1,59,59,999)},
3524 
3525 
3526     {"America/Los_Angeles",     CalFields(2014,3,10,2,0,0,0),       -1,     UCAL_WALLTIME_FIRST,
3527                                 CalFields(2014,3,9,1,0,0,0)},
3528 
3529     {"America/Los_Angeles",     CalFields(2014,3,10,2,0,0,0),       -1,     UCAL_WALLTIME_LAST,
3530                                 CalFields(2014,3,9,3,0,0,0)},
3531 
3532     {"America/Los_Angeles",     CalFields(2014,3,10,2,0,0,0),       -1,     UCAL_WALLTIME_NEXT_VALID,
3533                                 CalFields(2014,3,9,3,0,0,0)},
3534 
3535 
3536     {"America/Los_Angeles",     CalFields(2014,3,10,2,30,0,0),      -1,     UCAL_WALLTIME_FIRST,
3537                                 CalFields(2014,3,9,1,30,0,0)},
3538 
3539     {"America/Los_Angeles",     CalFields(2014,3,10,2,30,0,0),      -1,     UCAL_WALLTIME_LAST,
3540                                 CalFields(2014,3,9,3,30,0,0)},
3541 
3542     {"America/Los_Angeles",     CalFields(2014,3,10,2,30,0,0),      -1,     UCAL_WALLTIME_NEXT_VALID,
3543                                 CalFields(2014,3,9,3,0,0,0)},
3544 
3545 
3546     {"America/Los_Angeles",     CalFields(2014,3,10,3,0,0,0),       -1,     UCAL_WALLTIME_FIRST,
3547                                 CalFields(2014,3,9,3,0,0,0)},
3548 
3549     {"America/Los_Angeles",     CalFields(2014,3,10,3,0,0,0),       -1,     UCAL_WALLTIME_LAST,
3550                                 CalFields(2014,3,9,3,0,0,0)},
3551 
3552     {"America/Los_Angeles",     CalFields(2014,3,10,3,0,0,0),       -1,     UCAL_WALLTIME_NEXT_VALID,
3553                                 CalFields(2014,3,9,3,0,0,0)},
3554 
3555 
3556     // Test case for ticket#10544
3557     {"America/Santiago",        CalFields(2013,4,27,0,0,0,0),       134,    UCAL_WALLTIME_FIRST,
3558                                 CalFields(2013,9,7,23,0,0,0)},
3559 
3560     {"America/Santiago",        CalFields(2013,4,27,0,0,0,0),       134,    UCAL_WALLTIME_LAST,
3561                                 CalFields(2013,9,8,1,0,0,0)},
3562 
3563     {"America/Santiago",        CalFields(2013,4,27,0,0,0,0),       134,    UCAL_WALLTIME_NEXT_VALID,
3564                                 CalFields(2013,9,8,1,0,0,0)},
3565 
3566 
3567     {"America/Santiago",        CalFields(2013,4,27,0,30,0,0),      134,    UCAL_WALLTIME_FIRST,
3568                                 CalFields(2013,9,7,23,30,0,0)},
3569 
3570     {"America/Santiago",        CalFields(2013,4,27,0,30,0,0),      134,    UCAL_WALLTIME_LAST,
3571                                 CalFields(2013,9,8,1,30,0,0)},
3572 
3573     {"America/Santiago",        CalFields(2013,4,27,0,30,0,0),      134,    UCAL_WALLTIME_NEXT_VALID,
3574                                 CalFields(2013,9,8,1,0,0,0)},
3575 
3576 
3577     // Extreme transition - Pacific/Apia completely skips 2011-12-30
3578     {"Pacific/Apia",            CalFields(2011,12,29,0,0,0,0),      1,      UCAL_WALLTIME_FIRST,
3579                                 CalFields(2011,12,31,0,0,0,0)},
3580 
3581     {"Pacific/Apia",            CalFields(2011,12,29,0,0,0,0),      1,      UCAL_WALLTIME_LAST,
3582                                 CalFields(2011,12,31,0,0,0,0)},
3583 
3584     {"Pacific/Apia",            CalFields(2011,12,29,0,0,0,0),      1,      UCAL_WALLTIME_NEXT_VALID,
3585                                 CalFields(2011,12,31,0,0,0,0)},
3586 
3587 
3588     {"Pacific/Apia",            CalFields(2011,12,31,12,0,0,0),     -1,     UCAL_WALLTIME_FIRST,
3589                                 CalFields(2011,12,29,12,0,0,0)},
3590 
3591     {"Pacific/Apia",            CalFields(2011,12,31,12,0,0,0),     -1,     UCAL_WALLTIME_LAST,
3592                                 CalFields(2011,12,29,12,0,0,0)},
3593 
3594     {"Pacific/Apia",            CalFields(2011,12,31,12,0,0,0),     -1,     UCAL_WALLTIME_NEXT_VALID,
3595                                 CalFields(2011,12,29,12,0,0,0)},
3596 
3597 
3598     // 30 minutes DST - Australia/Lord_Howe
3599     {"Australia/Lord_Howe",     CalFields(2013,10,5,2,15,0,0),      1,      UCAL_WALLTIME_FIRST,
3600                                 CalFields(2013,10,6,1,45,0,0)},
3601 
3602     {"Australia/Lord_Howe",     CalFields(2013,10,5,2,15,0,0),      1,      UCAL_WALLTIME_LAST,
3603                                 CalFields(2013,10,6,2,45,0,0)},
3604 
3605     {"Australia/Lord_Howe",     CalFields(2013,10,5,2,15,0,0),      1,      UCAL_WALLTIME_NEXT_VALID,
3606                                 CalFields(2013,10,6,2,30,0,0)},
3607 
3608     {NULL, CalFields(0,0,0,0,0,0,0), 0, UCAL_WALLTIME_LAST, CalFields(0,0,0,0,0,0,0)}
3609 };
3610 
TestAddAcrossZoneTransition()3611 void CalendarTest::TestAddAcrossZoneTransition() {
3612     UErrorCode status = U_ZERO_ERROR;
3613     GregorianCalendar cal(status);
3614     TEST_CHECK_STATUS;
3615 
3616     for (int32_t i = 0; AAZTDATA[i].zone; i++) {
3617         status = U_ZERO_ERROR;
3618         TimeZone *tz = TimeZone::createTimeZone(AAZTDATA[i].zone);
3619         cal.adoptTimeZone(tz);
3620         cal.setSkippedWallTimeOption(AAZTDATA[i].skippedWTOpt);
3621         AAZTDATA[i].base.setTo(cal);
3622         cal.add(UCAL_DATE, AAZTDATA[i].deltaDays, status);
3623         TEST_CHECK_STATUS;
3624 
3625         if (!AAZTDATA[i].expected.isEquivalentTo(cal, status)) {
3626             CalFields res(cal, status);
3627             TEST_CHECK_STATUS;
3628             char buf[32];
3629             const char *optDisp = AAZTDATA[i].skippedWTOpt == UCAL_WALLTIME_FIRST ? "FIRST" :
3630                     AAZTDATA[i].skippedWTOpt == UCAL_WALLTIME_LAST ? "LAST" : "NEXT_VALID";
3631             dataerrln(UnicodeString("Error: base:") + AAZTDATA[i].base.toString(buf, sizeof(buf)) + ", tz:" + AAZTDATA[i].zone
3632                         + ", delta:" + AAZTDATA[i].deltaDays + " day(s), opt:" + optDisp
3633                         + ", result:" + res.toString(buf, sizeof(buf))
3634                         + " - expected:" + AAZTDATA[i].expected.toString(buf, sizeof(buf)));
3635         }
3636     }
3637 }
3638 
3639 // Data in a separate file (Gregorian to Chinese lunar map)
3640 #define INCLUDED_FROM_CALTEST_CPP
3641 #include "caltestdata.h"
3642 
TestChineseCalendarMapping()3643 void CalendarTest::TestChineseCalendarMapping() {
3644     UErrorCode status = U_ZERO_ERROR;
3645     LocalPointer<TimeZone> zone(TimeZone::createTimeZone(UnicodeString("China")));
3646     Locale locEnCalGregory = Locale::createFromName("en@calendar=gregorian");
3647     Locale locEnCalChinese = Locale::createFromName("en@calendar=chinese");
3648     LocalPointer<Calendar>  calGregory(Calendar::createInstance(zone->clone(), locEnCalGregory, status));
3649     LocalPointer<Calendar>  calChinese(Calendar::createInstance(zone.orphan(), locEnCalChinese, status));
3650     if ( U_FAILURE(status) ) {
3651         errln("Fail: Calendar::createInstance fails for en with calendar=gregorian or calendar=chinese: %s", u_errorName(status));
3652     } else {
3653         const GregoToLunar * mapPtr = gregoToLunar; // in "caltestdata.h" included above
3654         calGregory->clear();
3655         calChinese->clear();
3656         for (; mapPtr->gyr != 0; mapPtr++) {
3657             status = U_ZERO_ERROR;
3658             calGregory->set(mapPtr->gyr, mapPtr->gmo - 1, mapPtr->gda, 8, 0);
3659             UDate date = calGregory->getTime(status);
3660             calChinese->setTime(date, status);
3661             if ( U_FAILURE(status) ) {
3662                 errln("Fail: for Gregorian %4d-%02d-%02d, calGregory->getTime or calChinese->setTime reports: %s",
3663                         mapPtr->gyr, mapPtr->gmo, mapPtr->gda, u_errorName(status));
3664                 continue;
3665             }
3666             int32_t era = calChinese->get(UCAL_ERA, status);
3667             int32_t yr  = calChinese->get(UCAL_YEAR, status);
3668             int32_t mo  = calChinese->get(UCAL_MONTH, status) + 1;
3669             int32_t lp  = calChinese->get(UCAL_IS_LEAP_MONTH, status);
3670             int32_t da  = calChinese->get(UCAL_DATE, status);
3671             if ( U_FAILURE(status) ) {
3672                 errln("Fail: for Gregorian %4d-%02d-%02d, calChinese->get (for era, yr, mo, leapmo, da) reports: %s",
3673                         mapPtr->gyr, mapPtr->gmo, mapPtr->gda, u_errorName(status));
3674                 continue;
3675             }
3676             if (yr != mapPtr->cyr || mo != mapPtr->cmo || lp != mapPtr->clp || da != mapPtr->cda) {
3677                 errln("Fail: for Gregorian %4d-%02d-%02d, expected Chinese %2d-%02d(%d)-%02d, got %2d-%02d(%d)-%02d",
3678                         mapPtr->gyr, mapPtr->gmo, mapPtr->gda, mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, yr, mo, lp, da);
3679                 continue;
3680             }
3681             // If Grego->Chinese worked, try reverse mapping
3682             calChinese->set(UCAL_ERA, era);
3683             calChinese->set(UCAL_YEAR, mapPtr->cyr);
3684             calChinese->set(UCAL_MONTH, mapPtr->cmo - 1);
3685             calChinese->set(UCAL_IS_LEAP_MONTH, mapPtr->clp);
3686             calChinese->set(UCAL_DATE, mapPtr->cda);
3687             calChinese->set(UCAL_HOUR_OF_DAY, 8);
3688             date = calChinese->getTime(status);
3689             calGregory->setTime(date, status);
3690             if ( U_FAILURE(status) ) {
3691                 errln("Fail: for Chinese %2d-%02d(%d)-%02d, calChinese->getTime or calGregory->setTime reports: %s",
3692                         mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, u_errorName(status));
3693                 continue;
3694             }
3695             yr  = calGregory->get(UCAL_YEAR, status);
3696             mo  = calGregory->get(UCAL_MONTH, status) + 1;
3697             da  = calGregory->get(UCAL_DATE, status);
3698             if ( U_FAILURE(status) ) {
3699                 errln("Fail: for Chinese %2d-%02d(%d)-%02d, calGregory->get (for yr, mo, da) reports: %s",
3700                         mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, u_errorName(status));
3701                 continue;
3702             }
3703             if (yr != mapPtr->gyr || mo != mapPtr->gmo || da != mapPtr->gda) {
3704                 errln("Fail: for Chinese %2d-%02d(%d)-%02d, Gregorian %4d-%02d-%02d, got %4d-%02d-%02d",
3705                         mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, mapPtr->gyr, mapPtr->gmo, mapPtr->gda, yr, mo, da);
3706                 continue;
3707             }
3708         }
3709     }
3710 }
3711 
3712 #endif /* #if !UCONFIG_NO_FORMATTING */
3713 
3714 //eof
3715