1 /* 2 ** 3 ** Copyright 2009, The Android Open Source Project 4 ** 5 ** Licensed under the Apache License, Version 2.0 (the "License"); 6 ** you may not use this file except in compliance with the License. 7 ** You may obtain a copy of the License at 8 ** 9 ** http://www.apache.org/licenses/LICENSE-2.0 10 ** 11 ** Unless required by applicable law or agreed to in writing, software 12 ** distributed under the License is distributed on an "AS IS" BASIS, 13 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14 ** See the License for the specific language governing permissions and 15 ** limitations under the License. 16 */ 17 18 /* Based on http://code.google.com/p/google-rfc-2445/source/browse/trunk/test/com/google/ical/iter/RRuleIteratorImplTest.java */ 19 20 package com.android.calendarcommon2; 21 22 import com.android.calendarcommon2.RecurrenceSet; 23 24 import android.os.Debug; 25 import android.test.suitebuilder.annotation.MediumTest; 26 import android.test.suitebuilder.annotation.Suppress; 27 import android.text.format.Time; 28 import junit.framework.TestCase; 29 30 /** 31 * You can run those tests with: 32 * 33 * adb shell am instrument 34 * -e debug false 35 * -w 36 * -e 37 * class com.android.providers.calendar.RRuleTest 38 * com.android.providers.calendar.tests/android.test.InstrumentationTestRunner 39 */ 40 41 public class RRuleTest extends TestCase { 42 private static final String TAG = "RRuleTest"; 43 private static final boolean METHOD_TRACE = false; 44 45 private static String[] getFormattedDates(long[] dates, Time time, boolean truncate) { 46 String[] out = new String[dates.length]; 47 int i = 0; 48 for (long date : dates) { 49 time.set(date); 50 if (truncate) { 51 out[i] = time.format2445().substring(0, 8); // Just YYMMDD 52 } else { 53 out[i] = time.format2445().substring(0, 15); // YYMMDDThhmmss 54 } 55 ++i; 56 } 57 return out; 58 } 59 60 static final String PST = "America/Los_Angeles"; 61 static final String UTC = "UTC"; 62 // Use this date as end of recurrence unlessotherwise specified. 63 static final String DEFAULT_END = "20091212"; 64 65 private void runRecurrenceIteratorTest(String rruleText, String dtStart, int limit, 66 String golden) throws Exception { 67 runRecurrenceIteratorTest(rruleText, dtStart, limit, golden, null, null, UTC); 68 } 69 70 private void runRecurrenceIteratorTest(String rrule, String dtstartStr, int limit, 71 String golden, String advanceTo, String tz) throws Exception { 72 runRecurrenceIteratorTest(rrule, dtstartStr, limit, golden, advanceTo, null, tz); 73 } 74 75 /** 76 * Tests a recurrence rule 77 * @param rrule The rule to expand 78 * @param dtstartStr The dtstart to use 79 * @param limit Maximum number of entries to expand. if there are more, "..." is appended to 80 * the result. Note that Android's recurrence expansion doesn't support expanding n results, 81 * so this is faked by expanding until the endAt date, and then taking limit results. 82 * @param golden The desired results 83 * @param advanceTo The starting date for expansion. dtstartStr is used if null is passed in. 84 * @param endAt The ending date. DEFAULT_END is used if null is passed in. 85 * @param tz The time zone. UTC is used if null is passed in. 86 * @throws Exception if anything goes wrong. 87 */ 88 private void runRecurrenceIteratorTest(String rrule, String dtstartStr, int limit, 89 String golden, String advanceTo, String endAt, String tz) throws Exception { 90 91 String rdate = ""; 92 String exrule = ""; 93 String exdate = ""; 94 rrule = rrule.replace("RRULE:", ""); 95 // RecurrenceSet does not support folding of lines, so fold here 96 rrule = rrule.replace("\n ", ""); 97 98 Time dtstart = new Time(tz); 99 Time rangeStart = new Time(tz); 100 Time rangeEnd = new Time(tz); 101 Time outCal = new Time(tz); 102 103 dtstart.parse(dtstartStr); 104 if (advanceTo == null) { 105 advanceTo = dtstartStr; 106 } 107 if (endAt == null) { 108 endAt = DEFAULT_END; 109 } 110 111 rangeStart.parse(advanceTo); 112 rangeEnd.parse(endAt); 113 114 115 RecurrenceProcessor rp = new RecurrenceProcessor(); 116 RecurrenceSet recur = new RecurrenceSet(rrule, rdate, exrule, exdate); 117 118 long[] out = rp.expand(dtstart, recur, rangeStart.toMillis(false /* use isDst */), 119 rangeEnd.toMillis(false /* use isDst */)); 120 121 if (METHOD_TRACE) { 122 Debug.stopMethodTracing(); 123 } 124 125 boolean truncate = dtstartStr.length() <= 8; // Just date, not date-time 126 String[] actual = getFormattedDates(out, outCal, truncate); 127 128 StringBuilder sb = new StringBuilder(); 129 int k = 0; 130 while (k < actual.length && --limit >= 0) { 131 if (k != 0) { 132 sb.append(','); 133 } 134 sb.append(actual[k]); 135 k++; 136 } 137 if (limit < 0) { 138 sb.append(",..."); 139 } 140 assertEquals(golden, sb.toString()); 141 } 142 143 // Infinite loop, bug 1662110 144 @MediumTest 145 public void testFrequencyLimits() throws Exception { 146 // Rather than checking that we get an exception, 147 // we now want to finish, but in a reasonable time 148 final long tenSeconds = 10000; 149 long start = System.currentTimeMillis(); 150 runRecurrenceIteratorTest( 151 "RRULE:FREQ=SECONDLY;BYSECOND=0,1,2,3,4,5,6,7,8,9,10,11,12,13,14," + 152 "15,16,17,18,19,20,21,22,23,24,25,26,27,28,29," + 153 "30,31,32,33,34,35,36,37,38,39,40,41,42,43,44," + 154 "45,46,47,48,49,50,51,52,53,54,55,56,57,58,59", "20000101", 1, "20000101"); 155 if (System.currentTimeMillis() - start > tenSeconds) { 156 fail("Don't do that"); 157 } 158 } 159 160 @MediumTest 161 public void testSimpleDaily() throws Exception { 162 runRecurrenceIteratorTest("RRULE:FREQ=DAILY", "20060120", 5, "20060120,20060121,20060122,20060123,20060124,..."); 163 } 164 165 @MediumTest 166 public void testSimpleWeekly() throws Exception { 167 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY", "20060120", 5, "20060120,20060127,20060203,20060210,20060217,..."); 168 } 169 170 @MediumTest 171 public void testSimpleMonthly() throws Exception { 172 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY", "20060120", 5, "20060120,20060220,20060320,20060420,20060520,..."); 173 } 174 175 @MediumTest 176 public void testSimpleYearly() throws Exception { 177 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY", "20060120", 5, "20060120,20070120,20080120,20090120,20100120,...", null, "20120101", UTC); 178 } 179 180 // from section 4.3.10 181 @MediumTest 182 public void testMultipleByParts() throws Exception { 183 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=2;BYMONTH=1;BYDAY=SU", "19970105", 8, "19970105,19970112,19970119,19970126," + "19990103,19990110,19990117,19990124,..."); 184 } 185 186 @MediumTest 187 public void testCountWithInterval() throws Exception { 188 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;COUNT=10;INTERVAL=2", "19970105", 11, "19970105,19970107,19970109,19970111,19970113," + "19970115,19970117,19970119,19970121,19970123"); 189 } 190 191 // from section 4.6.5 192 // Fails: wrong dates 193 @MediumTest 194 @Suppress 195 public void testNegativeOffsetsA() throws Exception { 196 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=-1SU;BYMONTH=10", "19970105", 5, "19971026,19981025,19991031,20001029,20011028,..."); 197 } 198 199 // Fails: wrong dates 200 @MediumTest 201 @Suppress 202 public void testNegativeOffsetsB() throws Exception { 203 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4", "19970105", 5, "19970406,19980405,19990404,20000402,20010401,..."); 204 } 205 206 // Fails: wrong dates 207 @MediumTest 208 @Suppress 209 public void testNegativeOffsetsC() throws Exception { 210 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=1SU;BYMONTH=4;UNTIL=19980404T150000Z", "19970105", 5, "19970406"); 211 } 212 213 // feom section 4.8.5.4 214 @MediumTest 215 public void testDailyFor10Occ() throws Exception { 216 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;COUNT=10", "19970902T090000", 11, "19970902T090000,19970903T090000,19970904T090000,19970905T090000," + "19970906T090000,19970907T090000,19970908T090000,19970909T090000," + "19970910T090000,19970911T090000"); 217 218 } 219 220 @MediumTest 221 public void testDailyUntilDec4() throws Exception { 222 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;UNTIL=19971204", "19971128", 11, "19971128,19971129,19971130,19971201,19971202,19971203,19971204"); 223 } 224 225 // Fails: infinite loop 226 @MediumTest 227 @Suppress 228 public void testEveryOtherDayForever() throws Exception { 229 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;INTERVAL=2", "19971128", 5, "19971128,19971130,19971202,19971204,19971206,..."); 230 } 231 232 @MediumTest 233 public void testEvery10Days5Occ() throws Exception { 234 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;INTERVAL=10;COUNT=5", "19970902", 5, "19970902,19970912,19970922,19971002,19971012"); 235 } 236 237 @MediumTest 238 public void testWeeklyFor10Occ() throws Exception { 239 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;COUNT=10", "19970902", 10, "19970902,19970909,19970916,19970923,19970930," + "19971007,19971014,19971021,19971028,19971104"); 240 } 241 242 @MediumTest 243 public void testWeeklyUntilDec24() throws Exception { 244 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;UNTIL=19971224", "19970902", 25, "19970902,19970909,19970916,19970923,19970930," + "19971007,19971014,19971021,19971028,19971104," + "19971111,19971118,19971125,19971202,19971209," + "19971216,19971223"); 245 } 246 247 @MediumTest 248 public void testEveryOtherWeekForever() throws Exception { 249 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;WKST=SU", "19970902", 11, "19970902,19970916,19970930,19971014,19971028," + "19971111,19971125,19971209,19971223,19980106," + "19980120,..."); 250 } 251 252 @MediumTest 253 public void testWeeklyOnTuesdayAndThursdayFor5Weeks() throws Exception { 254 // if UNTIL date does not match start date, then until date treated as 255 // occurring on midnight. 256 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;UNTIL=19971007;WKST=SU;BYDAY=TU,TH", "19970902T090000", 11, "19970902T090000,19970904T090000,19970909T090000,19970911T090000," + "19970916T090000,19970918T090000,19970923T090000,19970925T090000," + "19970930T090000,19971002T090000"); 257 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;UNTIL=19971007T000000Z;WKST=SU;BYDAY=TU,TH", "19970902T090000", 11, "19970902T090000,19970904T090000,19970909T090000,19970911T090000," + "19970916T090000,19970918T090000,19970923T090000,19970925T090000," + "19970930T090000,19971002T090000"); 258 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;COUNT=10;WKST=SU;BYDAY=TU,TH", "19970902", 11, "19970902,19970904,19970909,19970911,19970916," + "19970918,19970923,19970925,19970930,19971002"); 259 } 260 261 @MediumTest 262 public void testEveryOtherWeekOnMWFUntilDec24() throws Exception { 263 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;UNTIL=19971224T000000Z;WKST=SU;\n" + " BYDAY=MO,WE,FR", "19970903T090000", 25, "19970903T090000,19970905T090000,19970915T090000,19970917T090000," + "19970919T090000,19970929T090000,19971001T090000,19971003T090000," + "19971013T090000,19971015T090000,19971017T090000,19971027T090000," + "19971029T090000,19971031T090000,19971110T090000,19971112T090000," + "19971114T090000,19971124T090000,19971126T090000,19971128T090000," + "19971208T090000,19971210T090000,19971212T090000,19971222T090000"); 264 } 265 266 @MediumTest 267 public void testEveryOtherWeekOnMWFUntilDec24a() throws Exception { 268 // if the UNTIL date is timed, when the start is not, the time should be 269 // ignored, so we get one more instance 270 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;UNTIL=19971224T000000Z;WKST=SU;\n" + " BYDAY=MO,WE,FR", "19970903", 25, "19970903,19970905,19970915,19970917," + "19970919,19970929,19971001,19971003," + "19971013,19971015,19971017,19971027," + "19971029,19971031,19971110,19971112," + "19971114,19971124,19971126,19971128," + "19971208,19971210,19971212,19971222," + "19971224"); 271 } 272 273 // Fails with wrong times 274 @MediumTest 275 @Suppress 276 public void testEveryOtherWeekOnMWFUntilDec24b() throws Exception { 277 // test with an alternate timezone 278 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;UNTIL=19971224T090000Z;WKST=SU;\n" + " BYDAY=MO,WE,FR", "19970903T090000", 25, "19970903T160000,19970905T160000,19970915T160000,19970917T160000," + "19970919T160000,19970929T160000,19971001T160000,19971003T160000," + "19971013T160000,19971015T160000,19971017T160000,19971027T170000," + "19971029T170000,19971031T170000,19971110T170000,19971112T170000," + "19971114T170000,19971124T170000,19971126T170000,19971128T170000," + "19971208T170000,19971210T170000,19971212T170000,19971222T170000", null, PST); 279 } 280 281 @MediumTest 282 public void testEveryOtherWeekOnTuThFor8Occ() throws Exception { 283 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;COUNT=8;WKST=SU;BYDAY=TU,TH", "19970902", 8, "19970902,19970904,19970916,19970918,19970930," + "19971002,19971014,19971016"); 284 } 285 286 @MediumTest 287 public void testMonthlyOnThe1stFridayFor10Occ() throws Exception { 288 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;COUNT=10;BYDAY=1FR", "19970905", 10, "19970905,19971003,19971107,19971205,19980102," + "19980206,19980306,19980403,19980501,19980605"); 289 } 290 291 @MediumTest 292 public void testMonthlyOnThe1stFridayUntilDec24() throws Exception { 293 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;UNTIL=19971224T000000Z;BYDAY=1FR", "19970905", 4, "19970905,19971003,19971107,19971205"); 294 } 295 296 @MediumTest 297 public void testEveryOtherMonthOnThe1stAndLastSundayFor10Occ() throws Exception { 298 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;INTERVAL=2;COUNT=10;BYDAY=1SU,-1SU", "19970907", 10, "19970907,19970928,19971102,19971130,19980104," + "19980125,19980301,19980329,19980503,19980531"); 299 } 300 301 @MediumTest 302 public void testMonthlyOnTheSecondToLastMondayOfTheMonthFor6Months() throws Exception { 303 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;COUNT=6;BYDAY=-2MO", "19970922", 6, "19970922,19971020,19971117,19971222,19980119," + "19980216"); 304 } 305 306 @MediumTest 307 public void testMonthlyOnTheThirdToLastDay() throws Exception { 308 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYMONTHDAY=-3", "19970928", 6, "19970928,19971029,19971128,19971229,19980129,19980226,..."); 309 } 310 311 @MediumTest 312 public void testMonthlyOnThe2ndAnd15thFor10Occ() throws Exception { 313 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;COUNT=10;BYMONTHDAY=2,15", "19970902", 10, "19970902,19970915,19971002,19971015,19971102," + "19971115,19971202,19971215,19980102,19980115"); 314 } 315 316 @MediumTest 317 public void testMonthlyOnTheFirstAndLastFor10Occ() throws Exception { 318 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;COUNT=10;BYMONTHDAY=1,-1", "19970930", 10, "19970930,19971001,19971031,19971101,19971130," + "19971201,19971231,19980101,19980131,19980201"); 319 } 320 321 @MediumTest 322 public void testEvery18MonthsOnThe10thThru15thFor10Occ() throws Exception { 323 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;INTERVAL=18;COUNT=10;BYMONTHDAY=10,11,12,13,14,\n" + " 15", "19970910", 10, "19970910,19970911,19970912,19970913,19970914," + "19970915,19990310,19990311,19990312,19990313"); 324 } 325 326 @MediumTest 327 public void testEveryTuesdayEveryOtherMonth() throws Exception { 328 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;INTERVAL=2;BYDAY=TU", "19970902", 18, "19970902,19970909,19970916,19970923,19970930," + "19971104,19971111,19971118,19971125,19980106," + "19980113,19980120,19980127,19980303,19980310," + "19980317,19980324,19980331,..."); 329 } 330 331 @MediumTest 332 public void testYearlyInJuneAndJulyFor10Occurrences() throws Exception { 333 // Note: Since none of the BYDAY, BYMONTHDAY or BYYEARDAY components 334 // are specified, the day is gotten from DTSTART 335 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;COUNT=10;BYMONTH=6,7", "19970610", 10, "19970610,19970710,19980610,19980710,19990610," + "19990710,20000610,20000710,20010610,20010710"); 336 } 337 338 @MediumTest 339 public void testEveryOtherYearOnJanuaryFebruaryAndMarchFor10Occurrences() throws Exception { 340 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=2;COUNT=10;BYMONTH=1,2,3", "19970310", 10, "19970310,19990110,19990210,19990310,20010110," + "20010210,20010310,20030110,20030210,20030310"); 341 } 342 343 //Fails: wrong dates 344 @MediumTest 345 @Suppress 346 public void testEvery3rdYearOnThe1st100thAnd200thDayFor10Occurrences() throws Exception { 347 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=3;COUNT=10;BYYEARDAY=1,100,200", "19970101", 10, "19970101,19970410,19970719,20000101,20000409," + "20000718,20030101,20030410,20030719,20060101"); 348 } 349 350 // Fails: infinite loop 351 @MediumTest 352 @Suppress 353 public void testEvery20thMondayOfTheYearForever() throws Exception { 354 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=20MO", "19970519", 3, "19970519,19980518,19990517,..."); 355 } 356 357 // Fails: generates wrong dates 358 @MediumTest 359 @Suppress 360 public void testMondayOfWeekNumber20WhereTheDefaultStartOfTheWeekIsMonday() throws Exception { 361 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYWEEKNO=20;BYDAY=MO", "19970512", 3, "19970512,19980511,19990517,..."); 362 } 363 364 @MediumTest 365 public void testEveryThursdayInMarchForever() throws Exception { 366 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=TH", "19970313", 11, "19970313,19970320,19970327,19980305,19980312," + "19980319,19980326,19990304,19990311,19990318," + "19990325,..."); 367 } 368 369 //Fails: wrong dates 370 @MediumTest 371 @Suppress 372 public void testEveryThursdayButOnlyDuringJuneJulyAndAugustForever() throws Exception { 373 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=TH;BYMONTH=6,7,8", "19970605", 39, "19970605,19970612,19970619,19970626,19970703," + "19970710,19970717,19970724,19970731,19970807," + "19970814,19970821,19970828,19980604,19980611," + "19980618,19980625,19980702,19980709,19980716," + "19980723,19980730,19980806,19980813,19980820," + "19980827,19990603,19990610,19990617,19990624," + "19990701,19990708,19990715,19990722,19990729," + "19990805,19990812,19990819,19990826,..."); 374 } 375 376 //Fails: infinite loop 377 @MediumTest 378 @Suppress 379 public void testEveryFridayThe13thForever() throws Exception { 380 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYDAY=FR;BYMONTHDAY=13", "19970902", 5, "19980213,19980313,19981113,19990813,20001013," + "..."); 381 } 382 383 @MediumTest 384 public void testTheFirstSaturdayThatFollowsTheFirstSundayOfTheMonthForever() throws Exception { 385 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYDAY=SA;BYMONTHDAY=7,8,9,10,11,12,13", "19970913", 10, "19970913,19971011,19971108,19971213,19980110," + "19980207,19980307,19980411,19980509,19980613," + "..."); 386 } 387 388 @MediumTest 389 public void testEvery4YearsThe1stTuesAfterAMonInNovForever() throws Exception { 390 // US Presidential Election Day 391 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=4;BYMONTH=11;BYDAY=TU;BYMONTHDAY=2,3,4,\n" + " 5,6,7,8", "19961105", 3, "19961105,20001107,20041102,..."); 392 } 393 394 // Fails: wrong dates 395 @MediumTest 396 @Suppress 397 public void testThe3rdInstanceIntoTheMonthOfOneOfTuesWedThursForNext3Months() throws Exception { 398 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;COUNT=3;BYDAY=TU,WE,TH;BYSETPOS=3", "19970904", 3, "19970904,19971007,19971106"); 399 } 400 401 // Fails: wrong dates 402 @MediumTest 403 @Suppress 404 public void testThe2ndToLastWeekdayOfTheMonth() throws Exception { 405 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=-2", "19970929", 7, "19970929,19971030,19971127,19971230,19980129," + "19980226,19980330,..."); 406 } 407 408 // Fails: infinite loop 409 @MediumTest 410 @Suppress 411 public void testEvery3HoursFrom900AmTo500PmOnASpecificDay() throws Exception { 412 runRecurrenceIteratorTest("RRULE:FREQ=HOURLY;INTERVAL=3;UNTIL=19970903T010000Z", "19970902", 7, "00000902,19970909,19970900,19970912,19970900," + "19970915,19970900"); 413 } 414 415 // Fails: infinite loop 416 @MediumTest 417 @Suppress 418 public void testEvery15MinutesFor6Occurrences() throws Exception { 419 runRecurrenceIteratorTest("RRULE:FREQ=MINUTELY;INTERVAL=15;COUNT=6", "19970902", 13, "00000902,19970909,19970900,19970909,19970915," + "19970909,19970930,19970909,19970945,19970910," + "19970900,19970910,19970915"); 420 } 421 422 @MediumTest 423 @Suppress 424 public void testEveryHourAndAHalfFor4Occurrences() throws Exception { 425 runRecurrenceIteratorTest("RRULE:FREQ=MINUTELY;INTERVAL=90;COUNT=4", "19970902", 9, "00000902,19970909,19970900,19970910,19970930," + "19970912,19970900,19970913,19970930"); 426 } 427 428 // Fails: wrong dates 429 @MediumTest 430 @Suppress 431 public void testAnExampleWhereTheDaysGeneratedMakesADifferenceBecauseOfWkst() throws Exception { 432 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;COUNT=4;BYDAY=TU,SU;WKST=MO", "19970805", 4, "19970805,19970810,19970819,19970824"); 433 } 434 435 // Fails: wrong dates 436 @MediumTest 437 @Suppress 438 public void testAnExampleWhereTheDaysGeneratedMakesADifferenceBecauseOfWkst2() throws Exception { 439 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;COUNT=4;BYDAY=TU,SU;WKST=SU", "19970805", 8, "19970805,19970817,19970819,19970831"); 440 } 441 442 // Fails: wrong dates 443 @MediumTest 444 @Suppress 445 public void testWithByDayAndByMonthDayFilter() throws Exception { 446 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;COUNT=4;BYDAY=TU,SU;" + "BYMONTHDAY=13,14,15,16,17,18,19,20", "19970805", 8, "19970817,19970819,19970914,19970916"); 447 } 448 449 // Failed: wrong dates 450 @MediumTest 451 @Suppress 452 public void testAnnuallyInAugustOnTuesAndSunBetween13thAnd20th() throws Exception { 453 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;COUNT=4;BYDAY=TU,SU;" + "BYMONTHDAY=13,14,15,16,17,18,19,20;BYMONTH=8", "19970605", 8, "19970817,19970819,19980816,19980818"); 454 } 455 456 // Failed: wrong dates 457 @MediumTest 458 @Suppress 459 public void testLastDayOfTheYearIsASundayOrTuesday() throws Exception { 460 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;COUNT=4;BYDAY=TU,SU;BYYEARDAY=-1", "19940605", 8, "19951231,19961231,20001231,20021231"); 461 } 462 463 // Fails: wrong dates 464 @MediumTest 465 @Suppress 466 public void testLastWeekdayOfMonth() throws Exception { 467 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYSETPOS=-1;BYDAY=-1MO,-1TU,-1WE,-1TH,-1FR", "19940605", 8, "19940630,19940729,19940831,19940930," + "19941031,19941130,19941230,19950131,..."); 468 } 469 470 // Fails: generates wrong dates 471 @MediumTest 472 @Suppress 473 public void testMonthsThatStartOrEndOnFriday() throws Exception { 474 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYMONTHDAY=1,-1;BYDAY=FR;COUNT=6", "19940605", 8, "19940701,19940930,19950331,19950630,19950901,19951201"); 475 } 476 477 // Fails: can't go that far into future 478 @MediumTest 479 @Suppress 480 public void testCenturiesThatAreNotLeapYears() throws Exception { 481 // I can't think of a good reason anyone would want to specify both a 482 // month day and a year day, so here's a really contrived example 483 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=100;BYYEARDAY=60;BYMONTHDAY=1", "19000101", 4, "19000301,21000301,22000301,23000301,...", null, "25000101", UTC); 484 } 485 486 // Fails: generates instances when it shouldn't 487 @MediumTest 488 @Suppress 489 public void testNoInstancesGenerated() throws Exception { 490 runRecurrenceIteratorTest("RRULE:FREQ=DAILY;UNTIL=19990101", "20000101", 4, ""); 491 } 492 493 // Fails: generates instances when it shouldn't 494 @MediumTest 495 @Suppress 496 public void testNoInstancesGenerated2() throws Exception { 497 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=2;BYMONTHDAY=30", "20000101", 4, ""); 498 } 499 500 // Fails: generates instances when it shouldn't 501 @MediumTest 502 @Suppress 503 public void testNoInstancesGenerated3() throws Exception { 504 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=4;BYYEARDAY=366", "20000101", 4, ""); 505 } 506 507 //Fails: wrong dates 508 @MediumTest 509 @Suppress 510 public void testLastWeekdayOfMarch() throws Exception { 511 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYMONTH=3;BYDAY=SA,SU;BYSETPOS=-1", "20000101", 4, "20000326,20010331,20020331,20030330,..."); 512 } 513 514 // Fails: wrong dates 515 @MediumTest 516 @Suppress 517 public void testFirstWeekdayOfMarch() throws Exception { 518 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYMONTH=3;BYDAY=SA,SU;BYSETPOS=1", "20000101", 4, "20000304,20010303,20020302,20030301,..."); 519 } 520 521 // January 1999 522 // Mo Tu We Th Fr Sa Su 523 // 1 2 3 // < 4 days, so not a week 524 // 4 5 6 7 8 9 10 525 526 // January 2000 527 // Mo Tu We Th Fr Sa Su 528 // 1 2 // < 4 days, so not a week 529 // 3 4 5 6 7 8 9 530 531 // January 2001 532 // Mo Tu We Th Fr Sa Su 533 // 1 2 3 4 5 6 7 534 // 8 9 10 11 12 13 14 535 536 // January 2002 537 // Mo Tu We Th Fr Sa Su 538 // 1 2 3 4 5 6 539 // 7 8 9 10 11 12 13 540 541 /** 542 * Find the first weekday of the first week of the year. 543 * The first week of the year may be partial, and the first week is considered 544 * to be the first one with at least four days. 545 */ 546 // Fails: wrong dates 547 @MediumTest 548 @Suppress 549 public void testFirstWeekdayOfFirstWeekOfYear() throws Exception { 550 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYWEEKNO=1;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=1", "19990101", 4, "19990104,20000103,20010101,20020101,..."); 551 } 552 553 // Fails: wrong dates 554 @MediumTest 555 @Suppress 556 public void testFirstSundayOfTheYear1() throws Exception { 557 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYWEEKNO=1;BYDAY=SU", "19990101", 4, "19990110,20000109,20010107,20020106,..."); 558 } 559 560 // Fails: wrong dates 561 @MediumTest 562 @Suppress 563 public void testFirstSundayOfTheYear2() throws Exception { 564 // TODO(msamuel): is this right? 565 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=1SU", "19990101", 4, "19990103,20000102,20010107,20020106,..."); 566 } 567 568 // Fails: wrong dates 569 @MediumTest 570 @Suppress 571 public void testFirstSundayOfTheYear3() throws Exception { 572 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=SU;BYYEARDAY=1,2,3,4,5,6,7,8,9,10,11,12,13" + ";BYSETPOS=1", "19990101", 4, "19990103,20000102,20010107,20020106,..."); 573 } 574 575 // Fails: wrong dates 576 @MediumTest 577 @Suppress 578 public void testFirstWeekdayOfYear() throws Exception { 579 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=1", "19990101", 4, "19990101,20000103,20010101,20020101,..."); 580 } 581 582 // Fails: wrong dates 583 @MediumTest 584 @Suppress 585 public void testLastWeekdayOfFirstWeekOfYear() throws Exception { 586 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYWEEKNO=1;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=-1", "19990101", 4, "19990108,20000107,20010105,20020104,..."); 587 } 588 589 // January 1999 590 // Mo Tu We Th Fr Sa Su 591 // 1 2 3 592 // 4 5 6 7 8 9 10 593 // 11 12 13 14 15 16 17 594 // 18 19 20 21 22 23 24 595 // 25 26 27 28 29 30 31 596 597 // Fails: wrong dates 598 @MediumTest 599 @Suppress 600 public void testSecondWeekday1() throws Exception { 601 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=2", "19990101", 4, "19990105,19990112,19990119,19990126,..."); 602 } 603 604 // January 1997 605 // Mo Tu We Th Fr Sa Su 606 // 1 2 3 4 5 607 // 6 7 8 9 10 11 12 608 // 13 14 15 16 17 18 19 609 // 20 21 22 23 24 25 26 610 // 27 28 29 30 31 611 612 // Fails: wrong dates 613 @MediumTest 614 @Suppress 615 public void testSecondWeekday2() throws Exception { 616 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=2", "19970101", 4, "19970102,19970107,19970114,19970121,..."); 617 } 618 619 // Fails: wrong dates 620 @MediumTest 621 @Suppress 622 public void testByYearDayAndByDayFilterInteraction() throws Exception { 623 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYYEARDAY=15;BYDAY=3MO", "19990101", 4, "20010115,20070115,20180115,20240115,..."); 624 } 625 626 // Fails: wrong dates 627 @MediumTest 628 @Suppress 629 public void testByDayWithNegWeekNoAsFilter() throws Exception { 630 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;BYMONTHDAY=26;BYDAY=-1FR", "19990101", 4, "19990226,19990326,19991126,20000526,..."); 631 } 632 633 // Fails: wrong dates 634 @MediumTest 635 @Suppress 636 public void testLastWeekOfTheYear() throws Exception { 637 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYWEEKNO=-1", "19990101", 6, "19991227,19991228,19991229,19991230,19991231,20001225,..."); 638 } 639 640 // Fails: not enough dates generated 641 @MediumTest 642 @Suppress 643 public void testUserSubmittedTest1() throws Exception { 644 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;INTERVAL=2;WKST=WE;BYDAY=SU,TU,TH,SA" + ";UNTIL=20000215T113000Z", "20000127T033000", 20, "20000127T033000,20000129T033000,20000130T033000,20000201T033000," + "20000210T033000,20000212T033000,20000213T033000,20000215T033000"); 645 } 646 647 @MediumTest 648 public void testAdvanceTo1() throws Exception { 649 // a bunch of tests grabbed from above with an advance-to date tacked on 650 651 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=TH", "19970313", 11, 652 /*"19970313,19970320,19970327,"*/"19980305,19980312," + "19980319,19980326,19990304,19990311,19990318," + "19990325,20000302,20000309,20000316,...", "19970601", UTC); 653 } 654 655 // Fails: infinite loop 656 @MediumTest 657 @Suppress 658 public void testAdvanceTo2() throws Exception { 659 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYDAY=20MO", "19970519", 3, 660 /*"19970519,"*/"19980518,19990517,20000515,...", "19980515", UTC); 661 } 662 663 // Fails: wrong dates 664 @MediumTest 665 @Suppress 666 public void testAdvanceTo3() throws Exception { 667 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=3;UNTIL=20090101;BYYEARDAY=1,100,200", "19970101", 10, 668 /*"19970101,19970410,19970719,20000101,"*/"20000409," + "20000718,20030101,20030410,20030719,20060101,20060410,20060719," + "20090101", "20000228", UTC); 669 } 670 671 //Fails: wrong dates 672 @MediumTest 673 @Suppress 674 public void testAdvanceTo4() throws Exception { 675 // make sure that count preserved 676 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=3;COUNT=10;BYYEARDAY=1,100,200", "19970101", 10, 677 /*"19970101,19970410,19970719,20000101,"*/"20000409," + "20000718,20030101,20030410,20030719,20060101", "20000228", UTC); 678 } 679 680 // Fails: too many dates 681 @MediumTest 682 @Suppress 683 public void testAdvanceTo5() throws Exception { 684 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=2;COUNT=10;BYMONTH=1,2,3", "19970310", 10, 685 /*"19970310,"*/"19990110,19990210,19990310,20010110," + "20010210,20010310,20030110,20030210,20030310", "19980401", UTC); 686 } 687 688 @MediumTest 689 public void testAdvanceTo6() throws Exception { 690 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;UNTIL=19971224", "19970902", 25, 691 /*"19970902,19970909,19970916,19970923,"*/"19970930," + "19971007,19971014,19971021,19971028,19971104," + "19971111,19971118,19971125,19971202,19971209," + "19971216,19971223", "19970930", UTC); 692 } 693 694 @MediumTest 695 public void testAdvanceTo7() throws Exception { 696 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;INTERVAL=18;BYMONTHDAY=10,11,12,13,14,\n" + " 15", "19970910", 5, 697 /*"19970910,19970911,19970912,19970913,19970914," + 698 "19970915,"*/"19990310,19990311,19990312,19990313,19990314,...", "19990101", UTC); 699 } 700 701 @MediumTest 702 public void testAdvanceTo8() throws Exception { 703 // advancing into the past 704 runRecurrenceIteratorTest("RRULE:FREQ=MONTHLY;INTERVAL=18;BYMONTHDAY=10,11,12,13,14,\n" + " 15", "19970910", 11, "19970910,19970911,19970912,19970913,19970914," + "19970915,19990310,19990311,19990312,19990313,19990314,...", "19970901", UTC); 705 } 706 707 // Fails: wrong dates 708 @MediumTest 709 @Suppress 710 public void testAdvanceTo9() throws Exception { 711 // skips first instance 712 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=100;BYMONTH=2;BYMONTHDAY=29", "19000101", 5, 713 // would return 2000 714 "24000229,28000229,32000229,36000229,40000229,...", "20040101", UTC); 715 } 716 717 // Infinite loop in native code (bug 1686327) 718 @MediumTest 719 @Suppress 720 public void testAdvanceTo10() throws Exception { 721 // filter hits until date before first instnace 722 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;INTERVAL=100;BYMONTH=2;BYMONTHDAY=29;UNTIL=21000101", "19000101", 5, "", "20040101", UTC); 723 } 724 725 // Fails: generates wrong dates 726 @MediumTest 727 @Suppress 728 public void testAdvanceTo11() throws Exception { 729 // advancing something that returns no instances 730 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=2;BYMONTHDAY=30", "20000101", 10, "", "19970901", UTC); 731 } 732 733 // Fails: generates wrong dates 734 @MediumTest 735 @Suppress 736 public void testAdvanceTo12() throws Exception { 737 // advancing something that returns no instances and has a BYSETPOS rule 738 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=2;BYMONTHDAY=30,31;BYSETPOS=1", "20000101", 10, "", "19970901", UTC); 739 } 740 741 // Fails: generates wrong dates 742 @MediumTest 743 @Suppress 744 public void testAdvanceTo13() throws Exception { 745 // advancing way past year generator timeout 746 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=2;BYMONTHDAY=28", "20000101", 10, "", "25000101", UTC); 747 } 748 749 /** 750 * a testcase that yielded dupes due to bysetPos evilness 751 */ 752 @MediumTest 753 @Suppress 754 public void testCaseThatYieldedDupes() throws Exception { 755 runRecurrenceIteratorTest("RRULE:FREQ=WEEKLY;WKST=SU;INTERVAL=1;BYMONTH=9,1,12,8" + ";BYMONTHDAY=-9,-29,24;BYSETPOS=-1,-4,10,-6,-1,-10,-10,-9,-8", "20060528", 200, "20060924,20061203,20061224,20070902,20071223,20080803,20080824," + "20090823,20100103,20100124,20110123,20120902,20121223,20130922," + "20140803,20140824,20150823,20160103,20160124,20170924,20171203," + "20171224,20180902,20181223,20190922,20200823,20210103,20210124," + "20220123,20230924,20231203,20231224,20240922,20250803,20250824," + "20260823,20270103,20270124,20280123,20280924,20281203,20281224," + "20290902,20291223,20300922,20310803,20310824,20330123,20340924," + "20341203,20341224,20350902,20351223,20360803,20360824,20370823," + "20380103,20380124,20390123,20400902,20401223,20410922,20420803," + "20420824,20430823,20440103,20440124,20450924,20451203,20451224," + "20460902,20461223,20470922,20480823,20490103,20490124,20500123," + "20510924,20511203,20511224,20520922,20530803,20530824,20540823," + "20550103,20550124,20560123,20560924,20561203,20561224,20570902," + "20571223,20580922,20590803,20590824,20610123,20620924,20621203," + "20621224,20630902,20631223,20640803,20640824,20650823,20660103," + "20660124,20670123,20680902,20681223,20690922,20700803,20700824," + "20710823,20720103,20720124,20730924,20731203,20731224,20740902," + "20741223,20750922,20760823,20770103,20770124,20780123,20790924," + "20791203,20791224,20800922,20810803,20810824,20820823,20830103," + "20830124,20840123,20840924,20841203,20841224,20850902,20851223," + "20860922,20870803,20870824,20890123,20900924,20901203,20901224," + "20910902,20911223,20920803,20920824,20930823,20940103,20940124," + "20950123,20960902,20961223,20970922,20980803,20980824,20990823," + "21000103,21000124,21010123,21020924,21021203,21021224,21030902," + "21031223,21040803,21040824,21050823,21060103,21060124,21070123," + "21080902,21081223,21090922,21100803,21100824,21110823,21120103," + "21120124,21130924,21131203,21131224,21140902,21141223,21150922," + "21160823,21170103,21170124,21180123,21190924,21191203,21191224," + "21200922,21210803,21210824,21220823,..."); 756 } 757 758 @MediumTest 759 public void testEveryThursdayinMarchEachYear() throws Exception { 760 runRecurrenceIteratorTest("RRULE:FREQ=YEARLY;BYMONTH=3;BYDAY=TH", "20100304", 9, "20100304,20100311,20100318,20100325,20110303,20110310,20110317,20110324,20110331", null, "20111231", UTC); 761 } 762 } 763