1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.provider.cts.calendar; 18 19 import android.content.BroadcastReceiver; 20 import android.content.ContentResolver; 21 import android.content.ContentUris; 22 import android.content.ContentValues; 23 import android.content.Context; 24 import android.content.Entity; 25 import android.content.EntityIterator; 26 import android.content.Intent; 27 import android.content.IntentFilter; 28 import android.database.Cursor; 29 import android.net.Uri; 30 import android.provider.CalendarContract; 31 import android.provider.CalendarContract.Attendees; 32 import android.provider.CalendarContract.CalendarEntity; 33 import android.provider.CalendarContract.Calendars; 34 import android.provider.CalendarContract.Colors; 35 import android.provider.CalendarContract.Events; 36 import android.provider.CalendarContract.EventsEntity; 37 import android.provider.CalendarContract.ExtendedProperties; 38 import android.provider.CalendarContract.Instances; 39 import android.provider.CalendarContract.Reminders; 40 import android.provider.CalendarContract.SyncState; 41 import android.test.InstrumentationTestCase; 42 import android.text.TextUtils; 43 import android.text.format.DateUtils; 44 import android.text.format.Time; 45 import android.util.Log; 46 47 import androidx.test.filters.MediumTest; 48 49 import com.android.compatibility.common.util.PollingCheck; 50 51 import java.util.ArrayList; 52 import java.util.HashSet; 53 import java.util.List; 54 import java.util.Set; 55 56 public class CalendarTest extends InstrumentationTestCase { 57 58 private static final String TAG = "CalCTS"; 59 private static final String CTS_TEST_TYPE = "LOCAL"; 60 61 // an arbitrary int used by some tests 62 private static final int SOME_ARBITRARY_INT = 143234; 63 64 // 15 sec timeout for reminder broadcast (but shouldn't usually take this long). 65 private static final int POLLING_TIMEOUT = 15000; 66 67 // @formatter:off 68 private static final String[] TIME_ZONES = new String[] { 69 "UTC", 70 "America/Los_Angeles", 71 "Asia/Beirut", 72 "Pacific/Auckland", }; 73 // @formatter:on 74 75 private static final String SQL_WHERE_ID = Events._ID + "=?"; 76 private static final String SQL_WHERE_CALENDAR_ID = Events.CALENDAR_ID + "=?"; 77 78 private ContentResolver mContentResolver; 79 80 /** If set, log verbose instance info when running recurrence tests. */ 81 private static final boolean DEBUG_RECURRENCE = false; 82 83 private static class CalendarHelper { 84 85 // @formatter:off 86 public static final String[] CALENDARS_SYNC_PROJECTION = new String[] { 87 Calendars._ID, 88 Calendars.ACCOUNT_NAME, 89 Calendars.ACCOUNT_TYPE, 90 Calendars._SYNC_ID, 91 Calendars.CAL_SYNC7, 92 Calendars.CAL_SYNC8, 93 Calendars.DIRTY, 94 Calendars.NAME, 95 Calendars.CALENDAR_DISPLAY_NAME, 96 Calendars.CALENDAR_COLOR, 97 Calendars.CALENDAR_COLOR_KEY, 98 Calendars.CALENDAR_ACCESS_LEVEL, 99 Calendars.VISIBLE, 100 Calendars.SYNC_EVENTS, 101 Calendars.CALENDAR_LOCATION, 102 Calendars.CALENDAR_TIME_ZONE, 103 Calendars.OWNER_ACCOUNT, 104 Calendars.CAN_ORGANIZER_RESPOND, 105 Calendars.CAN_MODIFY_TIME_ZONE, 106 Calendars.MAX_REMINDERS, 107 Calendars.ALLOWED_REMINDERS, 108 Calendars.ALLOWED_AVAILABILITY, 109 Calendars.ALLOWED_ATTENDEE_TYPES, 110 Calendars.DELETED, 111 Calendars.CAL_SYNC1, 112 Calendars.CAL_SYNC2, 113 Calendars.CAL_SYNC3, 114 Calendars.CAL_SYNC4, 115 Calendars.CAL_SYNC5, 116 Calendars.CAL_SYNC6, 117 }; 118 // @formatter:on 119 CalendarHelper()120 private CalendarHelper() {} // do not instantiate this class 121 122 /** 123 * Generates the e-mail address for the Calendar owner. Use this for 124 * Calendars.OWNER_ACCOUNT, Events.OWNER_ACCOUNT, and for Attendees.ATTENDEE_EMAIL 125 * when you want a "self" attendee entry. 126 */ generateCalendarOwnerEmail(String account)127 static String generateCalendarOwnerEmail(String account) { 128 return "OWNER_" + account + "@example.com"; 129 } 130 131 /** 132 * Creates a new set of values for creating a single calendar with every 133 * field. 134 * 135 * @param account The account name to create this calendar with 136 * @param seed A number used to generate the values 137 * @return A complete set of values for the calendar 138 */ getNewCalendarValues( String account, int seed)139 public static ContentValues getNewCalendarValues( 140 String account, int seed) { 141 String seedString = Long.toString(seed); 142 ContentValues values = new ContentValues(); 143 values.put(Calendars.ACCOUNT_TYPE, CTS_TEST_TYPE); 144 145 values.put(Calendars.ACCOUNT_NAME, account); 146 values.put(Calendars._SYNC_ID, "SYNC_ID:" + seedString); 147 values.put(Calendars.CAL_SYNC7, "SYNC_V:" + seedString); 148 values.put(Calendars.CAL_SYNC8, "SYNC_TIME:" + seedString); 149 values.put(Calendars.DIRTY, 0); 150 values.put(Calendars.OWNER_ACCOUNT, generateCalendarOwnerEmail(account)); 151 152 values.put(Calendars.NAME, seedString); 153 values.put(Calendars.CALENDAR_DISPLAY_NAME, "DISPLAY_" + seedString); 154 155 values.put(Calendars.CALENDAR_ACCESS_LEVEL, (seed % 8) * 100); 156 157 values.put(Calendars.CALENDAR_COLOR, 0xff000000 + seed); 158 values.put(Calendars.VISIBLE, seed % 2); 159 values.put(Calendars.SYNC_EVENTS, 1); // must be 1 for recurrence expansion 160 values.put(Calendars.CALENDAR_LOCATION, "LOCATION:" + seedString); 161 values.put(Calendars.CALENDAR_TIME_ZONE, TIME_ZONES[seed % TIME_ZONES.length]); 162 values.put(Calendars.CAN_ORGANIZER_RESPOND, seed % 2); 163 values.put(Calendars.CAN_MODIFY_TIME_ZONE, seed % 2); 164 values.put(Calendars.MAX_REMINDERS, 3); 165 values.put(Calendars.ALLOWED_REMINDERS, "0,1,2"); // does not include SMS (3) 166 values.put(Calendars.ALLOWED_ATTENDEE_TYPES, "0,1,2,3"); 167 values.put(Calendars.ALLOWED_AVAILABILITY, "0,1,2,3"); 168 values.put(Calendars.CAL_SYNC1, "SYNC1:" + seedString); 169 values.put(Calendars.CAL_SYNC2, "SYNC2:" + seedString); 170 values.put(Calendars.CAL_SYNC3, "SYNC3:" + seedString); 171 values.put(Calendars.CAL_SYNC4, "SYNC4:" + seedString); 172 values.put(Calendars.CAL_SYNC5, "SYNC5:" + seedString); 173 values.put(Calendars.CAL_SYNC6, "SYNC6:" + seedString); 174 175 return values; 176 } 177 178 /** 179 * Creates a set of values with just the updates and modifies the 180 * original values to the expected values 181 */ getUpdateCalendarValuesWithOriginal( ContentValues original, int seed)182 public static ContentValues getUpdateCalendarValuesWithOriginal( 183 ContentValues original, int seed) { 184 ContentValues values = new ContentValues(); 185 String seedString = Long.toString(seed); 186 187 values.put(Calendars.CALENDAR_DISPLAY_NAME, "DISPLAY_" + seedString); 188 values.put(Calendars.CALENDAR_COLOR, 0xff000000 + seed); 189 values.put(Calendars.VISIBLE, seed % 2); 190 values.put(Calendars.SYNC_EVENTS, seed % 2); 191 192 original.putAll(values); 193 original.put(Calendars.DIRTY, 1); 194 195 return values; 196 } 197 deleteCalendarById(ContentResolver resolver, long id)198 public static int deleteCalendarById(ContentResolver resolver, long id) { 199 return resolver.delete(Calendars.CONTENT_URI, Calendars._ID + "=?", 200 new String[] { Long.toString(id) }); 201 } 202 deleteCalendarByAccount(ContentResolver resolver, String account)203 public static int deleteCalendarByAccount(ContentResolver resolver, String account) { 204 return resolver.delete(Calendars.CONTENT_URI, Calendars.ACCOUNT_NAME + "=?", 205 new String[] { account }); 206 } 207 getCalendarsByAccount(ContentResolver resolver, String account)208 public static Cursor getCalendarsByAccount(ContentResolver resolver, String account) { 209 String selection = Calendars.ACCOUNT_TYPE + "=?"; 210 String[] selectionArgs; 211 if (account != null) { 212 selection += " AND " + Calendars.ACCOUNT_NAME + "=?"; 213 selectionArgs = new String[2]; 214 selectionArgs[1] = account; 215 } else { 216 selectionArgs = new String[1]; 217 } 218 selectionArgs[0] = CTS_TEST_TYPE; 219 220 return resolver.query(Calendars.CONTENT_URI, CALENDARS_SYNC_PROJECTION, selection, 221 selectionArgs, null); 222 } 223 } 224 225 /** 226 * Helper class for manipulating entries in the _sync_state table. 227 */ 228 private static class SyncStateHelper { 229 public static final String[] SYNCSTATE_PROJECTION = new String[] { 230 SyncState._ID, 231 SyncState.ACCOUNT_NAME, 232 SyncState.ACCOUNT_TYPE, 233 SyncState.DATA 234 }; 235 236 private static final byte[] SAMPLE_SYNC_DATA = { 237 (byte) 'H', (byte) 'e', (byte) 'l', (byte) 'l', (byte) 'o' 238 }; 239 SyncStateHelper()240 private SyncStateHelper() {} // do not instantiate 241 242 /** 243 * Creates a new set of values for creating a new _sync_state entry. 244 */ getNewSyncStateValues(String account)245 public static ContentValues getNewSyncStateValues(String account) { 246 ContentValues values = new ContentValues(); 247 values.put(SyncState.DATA, SAMPLE_SYNC_DATA); 248 values.put(SyncState.ACCOUNT_NAME, account); 249 values.put(SyncState.ACCOUNT_TYPE, CTS_TEST_TYPE); 250 return values; 251 } 252 253 /** 254 * Retrieves the _sync_state entry with the specified ID. 255 */ getSyncStateById(ContentResolver resolver, long id)256 public static Cursor getSyncStateById(ContentResolver resolver, long id) { 257 Uri uri = ContentUris.withAppendedId(SyncState.CONTENT_URI, id); 258 return resolver.query(uri, SYNCSTATE_PROJECTION, null, null, null); 259 } 260 261 /** 262 * Retrieves the _sync_state entry for the specified account. 263 */ getSyncStateByAccount(ContentResolver resolver, String account)264 public static Cursor getSyncStateByAccount(ContentResolver resolver, String account) { 265 assertNotNull(account); 266 String selection = SyncState.ACCOUNT_TYPE + "=? AND " + SyncState.ACCOUNT_NAME + "=?"; 267 String[] selectionArgs = new String[] { CTS_TEST_TYPE, account }; 268 269 return resolver.query(SyncState.CONTENT_URI, SYNCSTATE_PROJECTION, selection, 270 selectionArgs, null); 271 } 272 273 /** 274 * Deletes the _sync_state entry with the specified ID. Always done as app. 275 */ deleteSyncStateById(ContentResolver resolver, long id)276 public static int deleteSyncStateById(ContentResolver resolver, long id) { 277 Uri uri = ContentUris.withAppendedId(SyncState.CONTENT_URI, id); 278 return resolver.delete(uri, null, null); 279 } 280 281 /** 282 * Deletes the _sync_state entry associated with the specified account. Can be done 283 * as app or sync adapter. 284 */ deleteSyncStateByAccount(ContentResolver resolver, String account, boolean asSyncAdapter)285 public static int deleteSyncStateByAccount(ContentResolver resolver, String account, 286 boolean asSyncAdapter) { 287 Uri uri = SyncState.CONTENT_URI; 288 if (asSyncAdapter) { 289 uri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 290 } 291 return resolver.delete(uri, SyncState.ACCOUNT_NAME + "=?", 292 new String[] { account }); 293 } 294 } 295 296 // @formatter:off 297 private static class EventHelper { 298 public static final String[] EVENTS_PROJECTION = new String[] { 299 Events._ID, 300 Events.ACCOUNT_NAME, 301 Events.ACCOUNT_TYPE, 302 Events.OWNER_ACCOUNT, 303 // Events.ORGANIZER_CAN_RESPOND, from Calendars 304 // Events.CAN_CHANGE_TZ, from Calendars 305 // Events.MAX_REMINDERS, from Calendars 306 Events.CALENDAR_ID, 307 // Events.CALENDAR_DISPLAY_NAME, from Calendars 308 // Events.CALENDAR_COLOR, from Calendars 309 // Events.CALENDAR_ACL, from Calendars 310 // Events.CALENDAR_VISIBLE, from Calendars 311 Events.SYNC_DATA3, 312 Events.SYNC_DATA6, 313 Events.TITLE, 314 Events.EVENT_LOCATION, 315 Events.DESCRIPTION, 316 Events.STATUS, 317 Events.SELF_ATTENDEE_STATUS, 318 Events.DTSTART, 319 Events.DTEND, 320 Events.EVENT_TIMEZONE, 321 Events.EVENT_END_TIMEZONE, 322 Events.EVENT_COLOR, 323 Events.EVENT_COLOR_KEY, 324 Events.DURATION, 325 Events.ALL_DAY, 326 Events.ACCESS_LEVEL, 327 Events.AVAILABILITY, 328 Events.HAS_ALARM, 329 Events.HAS_EXTENDED_PROPERTIES, 330 Events.RRULE, 331 Events.RDATE, 332 Events.EXRULE, 333 Events.EXDATE, 334 Events.ORIGINAL_ID, 335 Events.ORIGINAL_SYNC_ID, 336 Events.ORIGINAL_INSTANCE_TIME, 337 Events.ORIGINAL_ALL_DAY, 338 Events.LAST_DATE, 339 Events.HAS_ATTENDEE_DATA, 340 Events.GUESTS_CAN_MODIFY, 341 Events.GUESTS_CAN_INVITE_OTHERS, 342 Events.GUESTS_CAN_SEE_GUESTS, 343 Events.ORGANIZER, 344 Events.DELETED, 345 Events._SYNC_ID, 346 Events.SYNC_DATA4, 347 Events.SYNC_DATA5, 348 Events.DIRTY, 349 Events.SYNC_DATA8, 350 Events.SYNC_DATA2, 351 Events.SYNC_DATA1, 352 Events.SYNC_DATA2, 353 Events.SYNC_DATA3, 354 Events.SYNC_DATA4, 355 Events.MUTATORS, 356 }; 357 // @formatter:on 358 EventHelper()359 private EventHelper() {} // do not instantiate this class 360 361 /** 362 * Constructs a set of name/value pairs that can be used to create a Calendar event. 363 * Various fields are generated from the seed value. 364 */ getNewEventValues( String account, int seed, long calendarId, boolean asSyncAdapter)365 public static ContentValues getNewEventValues( 366 String account, int seed, long calendarId, boolean asSyncAdapter) { 367 String seedString = Long.toString(seed); 368 ContentValues values = new ContentValues(); 369 values.put(Events.ORGANIZER, "ORGANIZER:" + seedString); 370 371 values.put(Events.TITLE, "TITLE:" + seedString); 372 values.put(Events.EVENT_LOCATION, "LOCATION_" + seedString); 373 374 values.put(Events.CALENDAR_ID, calendarId); 375 376 values.put(Events.DESCRIPTION, "DESCRIPTION:" + seedString); 377 values.put(Events.STATUS, seed % 2); // avoid STATUS_CANCELED for general testing 378 379 values.put(Events.DTSTART, seed); 380 values.put(Events.DTEND, seed + DateUtils.HOUR_IN_MILLIS); 381 values.put(Events.EVENT_TIMEZONE, TIME_ZONES[seed % TIME_ZONES.length]); 382 values.put(Events.EVENT_COLOR, seed); 383 // values.put(Events.EVENT_TIMEZONE2, TIME_ZONES[(seed +1) % 384 // TIME_ZONES.length]); 385 if ((seed % 2) == 0) { 386 // Either set to zero, or leave unset to get default zero. 387 // Must be 0 or dtstart/dtend will get adjusted. 388 values.put(Events.ALL_DAY, 0); 389 } 390 values.put(Events.ACCESS_LEVEL, seed % 4); 391 values.put(Events.AVAILABILITY, seed % 2); 392 values.put(Events.HAS_EXTENDED_PROPERTIES, seed % 2); 393 values.put(Events.HAS_ATTENDEE_DATA, seed % 2); 394 values.put(Events.GUESTS_CAN_MODIFY, seed % 2); 395 values.put(Events.GUESTS_CAN_INVITE_OTHERS, seed % 2); 396 values.put(Events.GUESTS_CAN_SEE_GUESTS, seed % 2); 397 398 // Default is STATUS_TENTATIVE (0). We either set it to that explicitly, or leave 399 // it set to the default. 400 if (seed != Events.STATUS_TENTATIVE) { 401 values.put(Events.SELF_ATTENDEE_STATUS, Events.STATUS_TENTATIVE); 402 } 403 404 if (asSyncAdapter) { 405 values.put(Events._SYNC_ID, "SYNC_ID:" + seedString); 406 values.put(Events.SYNC_DATA4, "SYNC_V:" + seedString); 407 values.put(Events.SYNC_DATA5, "SYNC_TIME:" + seedString); 408 values.put(Events.SYNC_DATA3, "HTML:" + seedString); 409 values.put(Events.SYNC_DATA6, "COMMENTS:" + seedString); 410 values.put(Events.DIRTY, 0); 411 values.put(Events.SYNC_DATA8, "0"); 412 } else { 413 // only the sync adapter can set the DIRTY flag 414 //values.put(Events.DIRTY, 1); 415 } 416 // values.put(Events.SYNC1, "SYNC1:" + seedString); 417 // values.put(Events.SYNC2, "SYNC2:" + seedString); 418 // values.put(Events.SYNC3, "SYNC3:" + seedString); 419 // values.put(Events.SYNC4, "SYNC4:" + seedString); 420 // values.put(Events.SYNC5, "SYNC5:" + seedString); 421 // Events.RRULE, 422 // Events.RDATE, 423 // Events.EXRULE, 424 // Events.EXDATE, 425 // // Events.ORIGINAL_ID 426 // Events.ORIGINAL_EVENT, // rename ORIGINAL_SYNC_ID 427 // Events.ORIGINAL_INSTANCE_TIME, 428 // Events.ORIGINAL_ALL_DAY, 429 430 return values; 431 } 432 433 /** 434 * Constructs a set of name/value pairs that can be used to create a recurring 435 * Calendar event. 436 * 437 * A duration of "P1D" is treated as an all-day event. 438 * 439 * @param startWhen Starting date/time in RFC 3339 format 440 * @param duration Event duration, in RFC 2445 duration format 441 * @param rrule Recurrence rule 442 * @return name/value pairs to use when creating event 443 */ getNewRecurringEventValues(String account, int seed, long calendarId, boolean asSyncAdapter, String startWhen, String duration, String rrule)444 public static ContentValues getNewRecurringEventValues(String account, int seed, 445 long calendarId, boolean asSyncAdapter, String startWhen, String duration, 446 String rrule) { 447 448 // Set up some general stuff. 449 ContentValues values = getNewEventValues(account, seed, calendarId, asSyncAdapter); 450 451 // Replace the DTSTART field. 452 String timeZone = values.getAsString(Events.EVENT_TIMEZONE); 453 Time time = new Time(timeZone); 454 time.parse3339(startWhen); 455 values.put(Events.DTSTART, time.toMillis(false)); 456 457 // Add in the recurrence-specific fields, and drop DTEND. 458 values.put(Events.RRULE, rrule); 459 values.put(Events.DURATION, duration); 460 values.remove(Events.DTEND); 461 462 return values; 463 } 464 465 /** 466 * Constructs the basic name/value pairs required for an exception to a recurring event. 467 * 468 * @param instanceStartMillis The start time of the instance 469 * @return name/value pairs to use when creating event 470 */ getNewExceptionValues(long instanceStartMillis)471 public static ContentValues getNewExceptionValues(long instanceStartMillis) { 472 ContentValues values = new ContentValues(); 473 values.put(Events.ORIGINAL_INSTANCE_TIME, instanceStartMillis); 474 475 return values; 476 } 477 getUpdateEventValuesWithOriginal(ContentValues original, int seed, boolean asSyncAdapter)478 public static ContentValues getUpdateEventValuesWithOriginal(ContentValues original, 479 int seed, boolean asSyncAdapter) { 480 String seedString = Long.toString(seed); 481 ContentValues values = new ContentValues(); 482 483 values.put(Events.TITLE, "TITLE:" + seedString); 484 values.put(Events.EVENT_LOCATION, "LOCATION_" + seedString); 485 values.put(Events.DESCRIPTION, "DESCRIPTION:" + seedString); 486 values.put(Events.STATUS, seed % 3); 487 488 values.put(Events.DTSTART, seed); 489 values.put(Events.DTEND, seed + DateUtils.HOUR_IN_MILLIS); 490 values.put(Events.EVENT_TIMEZONE, TIME_ZONES[seed % TIME_ZONES.length]); 491 // values.put(Events.EVENT_TIMEZONE2, TIME_ZONES[(seed +1) % 492 // TIME_ZONES.length]); 493 values.put(Events.ACCESS_LEVEL, seed % 4); 494 values.put(Events.AVAILABILITY, seed % 2); 495 values.put(Events.HAS_EXTENDED_PROPERTIES, seed % 2); 496 values.put(Events.HAS_ATTENDEE_DATA, seed % 2); 497 values.put(Events.GUESTS_CAN_MODIFY, seed % 2); 498 values.put(Events.GUESTS_CAN_INVITE_OTHERS, seed % 2); 499 values.put(Events.GUESTS_CAN_SEE_GUESTS, seed % 2); 500 if (asSyncAdapter) { 501 values.put(Events._SYNC_ID, "SYNC_ID:" + seedString); 502 values.put(Events.SYNC_DATA4, "SYNC_V:" + seedString); 503 values.put(Events.SYNC_DATA5, "SYNC_TIME:" + seedString); 504 values.put(Events.DIRTY, 0); 505 } 506 original.putAll(values); 507 return values; 508 } 509 addDefaultReadOnlyValues(ContentValues values, String account, boolean asSyncAdapter)510 public static void addDefaultReadOnlyValues(ContentValues values, String account, 511 boolean asSyncAdapter) { 512 values.put(Events.SELF_ATTENDEE_STATUS, Events.STATUS_TENTATIVE); 513 values.put(Events.DELETED, 0); 514 values.put(Events.DIRTY, asSyncAdapter ? 0 : 1); 515 values.put(Events.OWNER_ACCOUNT, CalendarHelper.generateCalendarOwnerEmail(account)); 516 values.put(Events.ACCOUNT_TYPE, CTS_TEST_TYPE); 517 values.put(Events.ACCOUNT_NAME, account); 518 } 519 520 /** 521 * Generates a RFC2445-format duration string. 522 */ generateDurationString(long durationMillis, boolean isAllDay)523 private static String generateDurationString(long durationMillis, boolean isAllDay) { 524 long durationSeconds = durationMillis / 1000; 525 526 // The server may react differently to an all-day event specified as "P1D" than 527 // it will to "PT86400S"; see b/1594638. 528 if (isAllDay && (durationSeconds % 86400) == 0) { 529 return "P" + durationSeconds / 86400 + "D"; 530 } else { 531 return "PT" + durationSeconds + "S"; 532 } 533 } 534 535 /** 536 * Deletes the event, and updates the values. 537 * @param resolver The resolver to issue the query against. 538 * @param uri The deletion URI. 539 * @param values Set of values to update (sets DELETED and DIRTY). 540 * @return The number of rows modified. 541 */ deleteEvent(ContentResolver resolver, Uri uri, ContentValues values)542 public static int deleteEvent(ContentResolver resolver, Uri uri, ContentValues values) { 543 values.put(Events.DELETED, 1); 544 values.put(Events.DIRTY, 1); 545 return resolver.delete(uri, null, null); 546 } 547 deleteEventAsSyncAdapter(ContentResolver resolver, Uri uri, String account)548 public static int deleteEventAsSyncAdapter(ContentResolver resolver, Uri uri, 549 String account) { 550 Uri syncUri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 551 return resolver.delete(syncUri, null, null); 552 } 553 getEventsByAccount(ContentResolver resolver, String account)554 public static Cursor getEventsByAccount(ContentResolver resolver, String account) { 555 String selection = Calendars.ACCOUNT_TYPE + "=?"; 556 String[] selectionArgs; 557 if (account != null) { 558 selection += " AND " + Calendars.ACCOUNT_NAME + "=?"; 559 selectionArgs = new String[2]; 560 selectionArgs[1] = account; 561 } else { 562 selectionArgs = new String[1]; 563 } 564 selectionArgs[0] = CTS_TEST_TYPE; 565 return resolver.query(Events.CONTENT_URI, EVENTS_PROJECTION, selection, selectionArgs, 566 null); 567 } 568 getEventByUri(ContentResolver resolver, Uri uri)569 public static Cursor getEventByUri(ContentResolver resolver, Uri uri) { 570 return resolver.query(uri, EVENTS_PROJECTION, null, null, null); 571 } 572 573 /** 574 * Looks up the specified Event in the database and returns the "selfAttendeeStatus" 575 * value. 576 */ lookupSelfAttendeeStatus(ContentResolver resolver, long eventId)577 public static int lookupSelfAttendeeStatus(ContentResolver resolver, long eventId) { 578 return getIntFromDatabase(resolver, Events.CONTENT_URI, eventId, 579 Events.SELF_ATTENDEE_STATUS); 580 } 581 582 /** 583 * Looks up the specified Event in the database and returns the "hasAlarm" 584 * value. 585 */ lookupHasAlarm(ContentResolver resolver, long eventId)586 public static int lookupHasAlarm(ContentResolver resolver, long eventId) { 587 return getIntFromDatabase(resolver, Events.CONTENT_URI, eventId, 588 Events.HAS_ALARM); 589 } 590 } 591 592 /** 593 * Helper class for manipulating entries in the Attendees table. 594 */ 595 private static class AttendeeHelper { 596 public static final String[] ATTENDEES_PROJECTION = new String[] { 597 Attendees._ID, 598 Attendees.EVENT_ID, 599 Attendees.ATTENDEE_NAME, 600 Attendees.ATTENDEE_EMAIL, 601 Attendees.ATTENDEE_STATUS, 602 Attendees.ATTENDEE_RELATIONSHIP, 603 Attendees.ATTENDEE_TYPE 604 }; 605 // indexes into projection 606 public static final int ATTENDEES_ID_INDEX = 0; 607 public static final int ATTENDEES_EVENT_ID_INDEX = 1; 608 609 // do not instantiate AttendeeHelper()610 private AttendeeHelper() {} 611 612 /** 613 * Adds a new attendee to the specified event. 614 * 615 * @return the _id of the new attendee, or -1 on failure 616 */ addAttendee(ContentResolver resolver, long eventId, String name, String email, int status, int relationship, int type)617 public static long addAttendee(ContentResolver resolver, long eventId, String name, 618 String email, int status, int relationship, int type) { 619 Uri uri = Attendees.CONTENT_URI; 620 621 ContentValues attendee = new ContentValues(); 622 attendee.put(Attendees.EVENT_ID, eventId); 623 attendee.put(Attendees.ATTENDEE_NAME, name); 624 attendee.put(Attendees.ATTENDEE_EMAIL, email); 625 attendee.put(Attendees.ATTENDEE_STATUS, status); 626 attendee.put(Attendees.ATTENDEE_RELATIONSHIP, relationship); 627 attendee.put(Attendees.ATTENDEE_TYPE, type); 628 Uri result = resolver.insert(uri, attendee); 629 return ContentUris.parseId(result); 630 } 631 632 /** 633 * Finds all Attendees rows for the specified event and email address. The returned 634 * cursor will use {@link AttendeeHelper#ATTENDEES_PROJECTION}. 635 */ findAttendeesByEmail(ContentResolver resolver, long eventId, String email)636 public static Cursor findAttendeesByEmail(ContentResolver resolver, long eventId, 637 String email) { 638 return resolver.query(Attendees.CONTENT_URI, ATTENDEES_PROJECTION, 639 Attendees.EVENT_ID + "=? AND " + Attendees.ATTENDEE_EMAIL + "=?", 640 new String[] { String.valueOf(eventId), email }, null); 641 } 642 } 643 644 /** 645 * Helper class for manipulating entries in the Colors table. 646 */ 647 private static class ColorHelper { 648 public static final String WHERE_COLOR_ACCOUNT = Colors.ACCOUNT_NAME + "=? AND " 649 + Colors.ACCOUNT_TYPE + "=?"; 650 public static final String WHERE_COLOR_ACCOUNT_AND_INDEX = WHERE_COLOR_ACCOUNT + " AND " 651 + Colors.COLOR_KEY + "=?"; 652 653 public static final String[] COLORS_PROJECTION = new String[] { 654 Colors._ID, // 0 655 Colors.ACCOUNT_NAME, // 1 656 Colors.ACCOUNT_TYPE, // 2 657 Colors.DATA, // 3 658 Colors.COLOR_TYPE, // 4 659 Colors.COLOR_KEY, // 5 660 Colors.COLOR, // 6 661 }; 662 // indexes into projection 663 public static final int COLORS_ID_INDEX = 0; 664 public static final int COLORS_INDEX_INDEX = 5; 665 public static final int COLORS_COLOR_INDEX = 6; 666 667 public static final int[] DEFAULT_TYPES = new int[] { 668 Colors.TYPE_CALENDAR, Colors.TYPE_CALENDAR, Colors.TYPE_CALENDAR, 669 Colors.TYPE_CALENDAR, Colors.TYPE_EVENT, Colors.TYPE_EVENT, Colors.TYPE_EVENT, 670 Colors.TYPE_EVENT, 671 }; 672 public static final int[] DEFAULT_COLORS = new int[] { 673 0xFFFF0000, 0xFF00FF00, 0xFF0000FF, 0xFFAA00AA, 0xFF00AAAA, 0xFF333333, 0xFFAAAA00, 674 0xFFAAAAAA, 675 }; 676 public static final String[] DEFAULT_INDICES = new String[] { 677 "000", "001", "010", "011", "100", "101", "110", "111", 678 }; 679 680 public static final int C_COLOR_0 = 0; 681 public static final int C_COLOR_1 = 1; 682 public static final int C_COLOR_2 = 2; 683 public static final int C_COLOR_3 = 3; 684 public static final int E_COLOR_0 = 4; 685 public static final int E_COLOR_1 = 5; 686 public static final int E_COLOR_2 = 6; 687 public static final int E_COLOR_3 = 7; 688 689 // do not instantiate ColorHelper()690 private ColorHelper() { 691 } 692 693 /** 694 * Adds a new color to the colors table. 695 * 696 * @return the _id of the new color, or -1 on failure 697 */ addColor(ContentResolver resolver, String accountName, String accountType, String data, String index, int type, int color)698 public static long addColor(ContentResolver resolver, String accountName, 699 String accountType, String data, String index, int type, int color) { 700 Uri uri = asSyncAdapter(Colors.CONTENT_URI, accountName, accountType); 701 702 ContentValues colorValues = new ContentValues(); 703 colorValues.put(Colors.DATA, data); 704 colorValues.put(Colors.COLOR_KEY, index); 705 colorValues.put(Colors.COLOR_TYPE, type); 706 colorValues.put(Colors.COLOR, color); 707 Uri result = resolver.insert(uri, colorValues); 708 return ContentUris.parseId(result); 709 } 710 711 /** 712 * Finds the color specified by an account name/type and a color index. 713 * The returned cursor will use {@link ColorHelper#COLORS_PROJECTION}. 714 */ findColorByIndex(ContentResolver resolver, String accountName, String accountType, String index)715 public static Cursor findColorByIndex(ContentResolver resolver, String accountName, 716 String accountType, String index) { 717 return resolver.query(Colors.CONTENT_URI, COLORS_PROJECTION, 718 WHERE_COLOR_ACCOUNT_AND_INDEX, 719 new String[] {accountName, accountType, index}, null); 720 } 721 findColorsByAccount(ContentResolver resolver, String accountName, String accountType)722 public static Cursor findColorsByAccount(ContentResolver resolver, String accountName, 723 String accountType) { 724 return resolver.query(Colors.CONTENT_URI, COLORS_PROJECTION, WHERE_COLOR_ACCOUNT, 725 new String[] { accountName, accountType }, null); 726 } 727 728 /** 729 * Adds a default set of test colors to the Colors table under the given 730 * account. 731 * 732 * @return true if the default colors were added successfully 733 */ addDefaultColorsToAccount(ContentResolver resolver, String accountName, String accountType)734 public static boolean addDefaultColorsToAccount(ContentResolver resolver, 735 String accountName, String accountType) { 736 for (int i = 0; i < DEFAULT_INDICES.length; i++) { 737 long id = addColor(resolver, accountName, accountType, null, DEFAULT_INDICES[i], 738 DEFAULT_TYPES[i], DEFAULT_COLORS[i]); 739 if (id == -1) { 740 return false; 741 } 742 } 743 return true; 744 } 745 deleteColorsByAccount(ContentResolver resolver, String accountName, String accountType)746 public static void deleteColorsByAccount(ContentResolver resolver, String accountName, 747 String accountType) { 748 Uri uri = asSyncAdapter(Colors.CONTENT_URI, accountName, accountType); 749 resolver.delete(uri, WHERE_COLOR_ACCOUNT, new String[] { accountName, accountType }); 750 } 751 } 752 753 754 /** 755 * Helper class for manipulating entries in the Reminders table. 756 */ 757 private static class ReminderHelper { 758 public static final String[] REMINDERS_PROJECTION = new String[] { 759 Reminders._ID, 760 Reminders.EVENT_ID, 761 Reminders.MINUTES, 762 Reminders.METHOD 763 }; 764 // indexes into projection 765 public static final int REMINDERS_ID_INDEX = 0; 766 public static final int REMINDERS_EVENT_ID_INDEX = 1; 767 public static final int REMINDERS_MINUTES_INDEX = 2; 768 public static final int REMINDERS_METHOD_INDEX = 3; 769 770 // do not instantiate ReminderHelper()771 private ReminderHelper() {} 772 773 /** 774 * Adds a new reminder to the specified event. 775 * 776 * @return the _id of the new reminder, or -1 on failure 777 */ addReminder(ContentResolver resolver, long eventId, int minutes, int method)778 public static long addReminder(ContentResolver resolver, long eventId, int minutes, 779 int method) { 780 Uri uri = Reminders.CONTENT_URI; 781 782 ContentValues reminder = new ContentValues(); 783 reminder.put(Reminders.EVENT_ID, eventId); 784 reminder.put(Reminders.MINUTES, minutes); 785 reminder.put(Reminders.METHOD, method); 786 Uri result = resolver.insert(uri, reminder); 787 return ContentUris.parseId(result); 788 } 789 790 /** 791 * Finds all Reminders rows for the specified event. The returned cursor will use 792 * {@link ReminderHelper#REMINDERS_PROJECTION}. 793 */ findRemindersByEventId(ContentResolver resolver, long eventId)794 public static Cursor findRemindersByEventId(ContentResolver resolver, long eventId) { 795 return resolver.query(Reminders.CONTENT_URI, REMINDERS_PROJECTION, 796 Reminders.EVENT_ID + "=?", new String[] { String.valueOf(eventId) }, null); 797 } 798 799 /** 800 * Looks up the specified Reminders row and returns the "method" value. 801 */ lookupMethod(ContentResolver resolver, long remId)802 public static int lookupMethod(ContentResolver resolver, long remId) { 803 return getIntFromDatabase(resolver, Reminders.CONTENT_URI, remId, 804 Reminders.METHOD); 805 } 806 } 807 808 /** 809 * Helper class for manipulating entries in the ExtendedProperties table. 810 */ 811 private static class ExtendedPropertiesHelper { 812 public static final String[] EXTENDED_PROPERTIES_PROJECTION = new String[] { 813 ExtendedProperties._ID, 814 ExtendedProperties.EVENT_ID, 815 ExtendedProperties.NAME, 816 ExtendedProperties.VALUE 817 }; 818 // indexes into projection 819 public static final int EXTENDED_PROPERTIES_ID_INDEX = 0; 820 public static final int EXTENDED_PROPERTIES_EVENT_ID_INDEX = 1; 821 public static final int EXTENDED_PROPERTIES_NAME_INDEX = 2; 822 public static final int EXTENDED_PROPERTIES_VALUE_INDEX = 3; 823 824 // do not instantiate ExtendedPropertiesHelper()825 private ExtendedPropertiesHelper() {} 826 827 /** 828 * Adds a new ExtendedProperty for the specified event. Runs as sync adapter. 829 * 830 * @return the _id of the new ExtendedProperty, or -1 on failure 831 */ addExtendedProperty(ContentResolver resolver, String account, long eventId, String name, String value)832 public static long addExtendedProperty(ContentResolver resolver, String account, 833 long eventId, String name, String value) { 834 Uri uri = asSyncAdapter(ExtendedProperties.CONTENT_URI, account, CTS_TEST_TYPE); 835 836 ContentValues ep = new ContentValues(); 837 ep.put(ExtendedProperties.EVENT_ID, eventId); 838 ep.put(ExtendedProperties.NAME, name); 839 ep.put(ExtendedProperties.VALUE, value); 840 Uri result = resolver.insert(uri, ep); 841 return ContentUris.parseId(result); 842 } 843 844 /** 845 * Finds all ExtendedProperties rows for the specified event. The returned cursor will 846 * use {@link ExtendedPropertiesHelper#EXTENDED_PROPERTIES_PROJECTION}. 847 */ findExtendedPropertiesByEventId(ContentResolver resolver, long eventId)848 public static Cursor findExtendedPropertiesByEventId(ContentResolver resolver, 849 long eventId) { 850 return resolver.query(ExtendedProperties.CONTENT_URI, EXTENDED_PROPERTIES_PROJECTION, 851 ExtendedProperties.EVENT_ID + "=?", 852 new String[] { String.valueOf(eventId) }, null); 853 } 854 855 /** 856 * Finds an ExtendedProperties entry with a matching name for the specified event, and 857 * returns the value. Throws an exception if we don't find exactly one row. 858 */ lookupValueByName(ContentResolver resolver, long eventId, String name)859 public static String lookupValueByName(ContentResolver resolver, long eventId, 860 String name) { 861 Cursor cursor = resolver.query(ExtendedProperties.CONTENT_URI, 862 EXTENDED_PROPERTIES_PROJECTION, 863 ExtendedProperties.EVENT_ID + "=? AND " + ExtendedProperties.NAME + "=?", 864 new String[] { String.valueOf(eventId), name }, null); 865 866 try { 867 if (cursor.getCount() != 1) { 868 throw new RuntimeException("Got " + cursor.getCount() + " results, expected 1"); 869 } 870 871 cursor.moveToFirst(); 872 return cursor.getString(EXTENDED_PROPERTIES_VALUE_INDEX); 873 } finally { 874 if (cursor != null) { 875 cursor.close(); 876 } 877 } 878 } 879 } 880 881 /** 882 * Creates an updated URI that includes query parameters that identify the source as a 883 * sync adapter. 884 */ asSyncAdapter(Uri uri, String account, String accountType)885 static Uri asSyncAdapter(Uri uri, String account, String accountType) { 886 return uri.buildUpon() 887 .appendQueryParameter(android.provider.CalendarContract.CALLER_IS_SYNCADAPTER, 888 "true") 889 .appendQueryParameter(Calendars.ACCOUNT_NAME, account) 890 .appendQueryParameter(Calendars.ACCOUNT_TYPE, accountType).build(); 891 } 892 893 /** 894 * Returns the value of the specified row and column in the Events table, as an integer. 895 * Throws an exception if the specified row or column doesn't exist or doesn't contain 896 * an integer (e.g. null entry). 897 */ getIntFromDatabase(ContentResolver resolver, Uri uri, long rowId, String columnName)898 private static int getIntFromDatabase(ContentResolver resolver, Uri uri, long rowId, 899 String columnName) { 900 String[] projection = { columnName }; 901 String selection = SQL_WHERE_ID; 902 String[] selectionArgs = { String.valueOf(rowId) }; 903 904 Cursor c = resolver.query(uri, projection, selection, selectionArgs, null); 905 try { 906 assertEquals(1, c.getCount()); 907 c.moveToFirst(); 908 return c.getInt(0); 909 } finally { 910 c.close(); 911 } 912 } 913 914 @Override setUp()915 protected void setUp() throws Exception { 916 super.setUp(); 917 mContentResolver = getInstrumentation().getTargetContext().getContentResolver(); 918 } 919 920 @MediumTest testCalendarCreationAndDeletion()921 public void testCalendarCreationAndDeletion() { 922 String account = "cc1_account"; 923 int seed = 0; 924 925 // Clean up just in case 926 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 927 long id = createAndVerifyCalendar(account, seed++, null); 928 929 removeAndVerifyCalendar(account, id); 930 } 931 932 /** 933 * Tests whether the default projections work. We don't need to have any data in 934 * the calendar, since it's testing the database schema. 935 */ 936 @MediumTest testDefaultProjections()937 public void testDefaultProjections() { 938 String account = "dproj_account"; 939 int seed = 0; 940 941 // Clean up just in case 942 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 943 long id = createAndVerifyCalendar(account, seed++, null); 944 945 Cursor c; 946 Uri uri; 947 // Calendars 948 c = mContentResolver.query(Calendars.CONTENT_URI, null, null, null, null); 949 c.close(); 950 // Events 951 c = mContentResolver.query(Events.CONTENT_URI, null, null, null, null); 952 c.close(); 953 // Instances 954 uri = Uri.withAppendedPath(Instances.CONTENT_URI, "0/1"); 955 c = mContentResolver.query(uri, null, null, null, null); 956 c.close(); 957 // Attendees 958 c = mContentResolver.query(Attendees.CONTENT_URI, null, null, null, null); 959 c.close(); 960 // Reminders (only REMINDERS_ID currently uses default projection) 961 uri = ContentUris.withAppendedId(Reminders.CONTENT_URI, 0); 962 c = mContentResolver.query(uri, null, null, null, null); 963 c.close(); 964 // CalendarAlerts 965 c = mContentResolver.query(CalendarContract.CalendarAlerts.CONTENT_URI, 966 null, null, null, null); 967 c.close(); 968 // CalendarCache 969 c = mContentResolver.query(CalendarContract.CalendarCache.URI, 970 null, null, null, null); 971 c.close(); 972 // CalendarEntity 973 c = mContentResolver.query(CalendarContract.CalendarEntity.CONTENT_URI, 974 null, null, null, null); 975 c.close(); 976 // EventEntities 977 c = mContentResolver.query(CalendarContract.EventsEntity.CONTENT_URI, 978 null, null, null, null); 979 c.close(); 980 // EventDays 981 uri = Uri.withAppendedPath(CalendarContract.EventDays.CONTENT_URI, "1/2"); 982 c = mContentResolver.query(uri, null, null, null, null); 983 c.close(); 984 // ExtendedProperties 985 c = mContentResolver.query(CalendarContract.ExtendedProperties.CONTENT_URI, 986 null, null, null, null); 987 c.close(); 988 989 removeAndVerifyCalendar(account, id); 990 } 991 992 /** 993 * Exercises the EventsEntity class. 994 */ 995 @MediumTest testEventsEntityQuery()996 public void testEventsEntityQuery() { 997 String account = "eeq_account"; 998 int seed = 0; 999 1000 // Clean up just in case. 1001 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1002 1003 // Create calendar. 1004 long calendarId = createAndVerifyCalendar(account, seed++, null); 1005 1006 // Create three events. We need to make sure SELF_ATTENDEE_STATUS isn't set, because 1007 // that causes the provider to generate an Attendees entry, and that'll throw off 1008 // our expected count. 1009 ContentValues eventValues; 1010 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1011 eventValues.remove(Events.SELF_ATTENDEE_STATUS); 1012 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1013 assertTrue(eventId1 >= 0); 1014 1015 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1016 eventValues.remove(Events.SELF_ATTENDEE_STATUS); 1017 long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1018 assertTrue(eventId2 >= 0); 1019 1020 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1021 eventValues.remove(Events.SELF_ATTENDEE_STATUS); 1022 long eventId3 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1023 assertTrue(eventId3 >= 0); 1024 1025 /* 1026 * Add some attendees, reminders, and extended properties. 1027 */ 1028 Uri uri, syncUri; 1029 1030 syncUri = asSyncAdapter(Reminders.CONTENT_URI, account, CTS_TEST_TYPE); 1031 ContentValues remValues = new ContentValues(); 1032 remValues.put(Reminders.EVENT_ID, eventId1); 1033 remValues.put(Reminders.MINUTES, 10); 1034 remValues.put(Reminders.METHOD, Reminders.METHOD_ALERT); 1035 mContentResolver.insert(syncUri, remValues); 1036 remValues.put(Reminders.MINUTES, 20); 1037 mContentResolver.insert(syncUri, remValues); 1038 1039 syncUri = asSyncAdapter(ExtendedProperties.CONTENT_URI, account, CTS_TEST_TYPE); 1040 ContentValues extended = new ContentValues(); 1041 extended.put(ExtendedProperties.NAME, "foo"); 1042 extended.put(ExtendedProperties.VALUE, "bar"); 1043 extended.put(ExtendedProperties.EVENT_ID, eventId2); 1044 mContentResolver.insert(syncUri, extended); 1045 extended.put(ExtendedProperties.EVENT_ID, eventId1); 1046 mContentResolver.insert(syncUri, extended); 1047 extended.put(ExtendedProperties.NAME, "foo2"); 1048 extended.put(ExtendedProperties.VALUE, "bar2"); 1049 mContentResolver.insert(syncUri, extended); 1050 1051 syncUri = asSyncAdapter(Attendees.CONTENT_URI, account, CTS_TEST_TYPE); 1052 ContentValues attendee = new ContentValues(); 1053 attendee.put(Attendees.ATTENDEE_NAME, "Joe"); 1054 attendee.put(Attendees.ATTENDEE_EMAIL, CalendarHelper.generateCalendarOwnerEmail(account)); 1055 attendee.put(Attendees.ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_DECLINED); 1056 attendee.put(Attendees.ATTENDEE_TYPE, Attendees.TYPE_REQUIRED); 1057 attendee.put(Attendees.ATTENDEE_RELATIONSHIP, Attendees.RELATIONSHIP_PERFORMER); 1058 attendee.put(Attendees.EVENT_ID, eventId3); 1059 mContentResolver.insert(syncUri, attendee); 1060 1061 /* 1062 * Iterate over all events on our calendar. Peek at a few values to see if they 1063 * look reasonable. 1064 */ 1065 EntityIterator ei = EventsEntity.newEntityIterator( 1066 mContentResolver.query(EventsEntity.CONTENT_URI, null, Events.CALENDAR_ID + "=?", 1067 new String[] { String.valueOf(calendarId) }, null), 1068 mContentResolver); 1069 int count = 0; 1070 try { 1071 while (ei.hasNext()) { 1072 Entity entity = ei.next(); 1073 ContentValues values = entity.getEntityValues(); 1074 ArrayList<Entity.NamedContentValues> subvalues = entity.getSubValues(); 1075 long eventId = values.getAsLong(Events._ID); 1076 if (eventId == eventId1) { 1077 // 2 x reminder, 2 x extended properties 1078 assertEquals(4, subvalues.size()); 1079 } else if (eventId == eventId2) { 1080 // Extended properties 1081 assertEquals(1, subvalues.size()); 1082 ContentValues subContentValues = subvalues.get(0).values; 1083 String name = subContentValues.getAsString( 1084 CalendarContract.ExtendedProperties.NAME); 1085 String value = subContentValues.getAsString( 1086 CalendarContract.ExtendedProperties.VALUE); 1087 assertEquals("foo", name); 1088 assertEquals("bar", value); 1089 } else if (eventId == eventId3) { 1090 // Attendees 1091 assertEquals(1, subvalues.size()); 1092 } else { 1093 fail("should not be here"); 1094 } 1095 count++; 1096 } 1097 assertEquals(3, count); 1098 } finally { 1099 ei.close(); 1100 } 1101 1102 // Confirm that querying for a single event yields a single event. 1103 ei = EventsEntity.newEntityIterator( 1104 mContentResolver.query(EventsEntity.CONTENT_URI, null, SQL_WHERE_ID, 1105 new String[] { String.valueOf(eventId3) }, null), 1106 mContentResolver); 1107 try { 1108 count = 0; 1109 while (ei.hasNext()) { 1110 Entity entity = ei.next(); 1111 count++; 1112 } 1113 assertEquals(1, count); 1114 } finally { 1115 ei.close(); 1116 } 1117 1118 1119 removeAndVerifyCalendar(account, calendarId); 1120 } 1121 1122 /** 1123 * Exercises the CalendarEntity class. 1124 */ 1125 @MediumTest testCalendarEntityQuery()1126 public void testCalendarEntityQuery() { 1127 String account1 = "ceq1_account"; 1128 String account2 = "ceq2_account"; 1129 String account3 = "ceq3_account"; 1130 int seed = 0; 1131 1132 // Clean up just in case. 1133 CalendarHelper.deleteCalendarByAccount(mContentResolver, account1); 1134 CalendarHelper.deleteCalendarByAccount(mContentResolver, account2); 1135 CalendarHelper.deleteCalendarByAccount(mContentResolver, account3); 1136 1137 // Create calendars. 1138 long calendarId1 = createAndVerifyCalendar(account1, seed++, null); 1139 long calendarId2 = createAndVerifyCalendar(account2, seed++, null); 1140 long calendarId3 = createAndVerifyCalendar(account3, seed++, null); 1141 1142 EntityIterator ei = CalendarEntity.newEntityIterator( 1143 mContentResolver.query(CalendarEntity.CONTENT_URI, null, 1144 Calendars._ID + "=? OR " + Calendars._ID + "=? OR " + Calendars._ID + "=?", 1145 new String[] { String.valueOf(calendarId1), String.valueOf(calendarId2), 1146 String.valueOf(calendarId3) }, 1147 null)); 1148 1149 try { 1150 int count = 0; 1151 while (ei.hasNext()) { 1152 Entity entity = ei.next(); 1153 count++; 1154 } 1155 assertEquals(3, count); 1156 } finally { 1157 ei.close(); 1158 } 1159 1160 removeAndVerifyCalendar(account1, calendarId1); 1161 removeAndVerifyCalendar(account2, calendarId2); 1162 removeAndVerifyCalendar(account3, calendarId3); 1163 } 1164 1165 /** 1166 * Tests creation and manipulation of Attendees. 1167 */ 1168 @MediumTest testAttendees()1169 public void testAttendees() { 1170 String account = "att_account"; 1171 int seed = 0; 1172 1173 // Clean up just in case. 1174 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1175 1176 // Create calendar. 1177 long calendarId = createAndVerifyCalendar(account, seed++, null); 1178 1179 // Create two events, one with a value set for SELF_ATTENDEE_STATUS, one without. 1180 ContentValues eventValues; 1181 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1182 eventValues.put(Events.SELF_ATTENDEE_STATUS, Events.STATUS_TENTATIVE); 1183 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1184 assertTrue(eventId1 >= 0); 1185 1186 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1187 eventValues.remove(Events.SELF_ATTENDEE_STATUS); 1188 long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1189 assertTrue(eventId2 >= 0); 1190 1191 /* 1192 * Add some attendees to the first event. 1193 */ 1194 long attId1 = AttendeeHelper.addAttendee(mContentResolver, eventId1, 1195 "Alice", 1196 "alice@example.com", 1197 Attendees.ATTENDEE_STATUS_TENTATIVE, 1198 Attendees.RELATIONSHIP_ATTENDEE, 1199 Attendees.TYPE_REQUIRED); 1200 long attId2 = AttendeeHelper.addAttendee(mContentResolver, eventId1, 1201 "Betty", 1202 "betty@example.com", 1203 Attendees.ATTENDEE_STATUS_DECLINED, 1204 Attendees.RELATIONSHIP_ATTENDEE, 1205 Attendees.TYPE_NONE); 1206 long attId3 = AttendeeHelper.addAttendee(mContentResolver, eventId1, 1207 "Carol", 1208 "carol@example.com", 1209 Attendees.ATTENDEE_STATUS_DECLINED, 1210 Attendees.RELATIONSHIP_ATTENDEE, 1211 Attendees.TYPE_OPTIONAL); 1212 1213 /* 1214 * Find the event1 "self" attendee entry. 1215 */ 1216 Cursor cursor = AttendeeHelper.findAttendeesByEmail(mContentResolver, eventId1, 1217 CalendarHelper.generateCalendarOwnerEmail(account)); 1218 try { 1219 assertEquals(1, cursor.getCount()); 1220 //DatabaseUtils.dumpCursor(cursor); 1221 1222 cursor.moveToFirst(); 1223 long id = cursor.getLong(AttendeeHelper.ATTENDEES_ID_INDEX); 1224 1225 /* 1226 * Update the status field. The provider should automatically propagate the result. 1227 */ 1228 ContentValues update = new ContentValues(); 1229 Uri uri = ContentUris.withAppendedId(Attendees.CONTENT_URI, id); 1230 1231 update.put(Attendees.ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_ACCEPTED); 1232 int count = mContentResolver.update(uri, update, null, null); 1233 assertEquals(1, count); 1234 1235 int status = EventHelper.lookupSelfAttendeeStatus(mContentResolver, eventId1); 1236 assertEquals(Attendees.ATTENDEE_STATUS_ACCEPTED, status); 1237 1238 } finally { 1239 if (cursor != null) { 1240 cursor.close(); 1241 } 1242 } 1243 1244 /* 1245 * Do a bulk update of all Attendees for this event, changing any Attendee with status 1246 * "declined" to "invited". 1247 */ 1248 ContentValues bulkUpdate = new ContentValues(); 1249 bulkUpdate.put(Attendees.ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_INVITED); 1250 1251 int count = mContentResolver.update(Attendees.CONTENT_URI, bulkUpdate, 1252 Attendees.EVENT_ID + "=? AND " + Attendees.ATTENDEE_STATUS + "=?", 1253 new String[] { 1254 String.valueOf(eventId1), String.valueOf(Attendees.ATTENDEE_STATUS_DECLINED) 1255 }); 1256 assertEquals(2, count); 1257 1258 /* 1259 * Add a new, non-self attendee to the second event. 1260 */ 1261 long attId4 = AttendeeHelper.addAttendee(mContentResolver, eventId2, 1262 "Diana", 1263 "diana@example.com", 1264 Attendees.ATTENDEE_STATUS_ACCEPTED, 1265 Attendees.RELATIONSHIP_ATTENDEE, 1266 Attendees.TYPE_REQUIRED); 1267 1268 /* 1269 * Confirm that the selfAttendeeStatus on the second event has the default value. 1270 */ 1271 int status = EventHelper.lookupSelfAttendeeStatus(mContentResolver, eventId2); 1272 assertEquals(Attendees.ATTENDEE_STATUS_NONE, status); 1273 1274 /* 1275 * Create a new "self" attendee in the second event by updating the email address to 1276 * match that of the calendar owner. 1277 */ 1278 ContentValues newSelf = new ContentValues(); 1279 newSelf.put(Attendees.ATTENDEE_EMAIL, CalendarHelper.generateCalendarOwnerEmail(account)); 1280 count = mContentResolver.update(ContentUris.withAppendedId(Attendees.CONTENT_URI, attId4), 1281 newSelf, null, null); 1282 assertEquals(1, count); 1283 1284 /* 1285 * Confirm that the event's selfAttendeeStatus has been updated. 1286 */ 1287 status = EventHelper.lookupSelfAttendeeStatus(mContentResolver, eventId2); 1288 assertEquals(Attendees.ATTENDEE_STATUS_ACCEPTED, status); 1289 1290 /* 1291 * TODO: (these are unexpected usage patterns) 1292 * - Update an Attendee's status and event_id to move it to a different event, and 1293 * confirm that the selfAttendeeStatus in the destination event is updated (rather 1294 * than that of the source event). 1295 * - Create two Attendees with email addresses that match "self" but have different 1296 * values for "status". Delete one and confirm that selfAttendeeStatus is changed 1297 * to that of the remaining Attendee. (There is no defined behavior for 1298 * selfAttendeeStatus when there are multiple matching Attendees.) 1299 */ 1300 1301 /* 1302 * Test deletion, singly by ID and in bulk. 1303 */ 1304 count = mContentResolver.delete(ContentUris.withAppendedId(Attendees.CONTENT_URI, attId4), 1305 null, null); 1306 assertEquals(1, count); 1307 1308 count = mContentResolver.delete(Attendees.CONTENT_URI, Attendees.EVENT_ID + "=?", 1309 new String[] { String.valueOf(eventId1) }); 1310 assertEquals(4, count); // 3 we created + 1 auto-added by the provider 1311 1312 removeAndVerifyCalendar(account, calendarId); 1313 } 1314 1315 /** 1316 * Tests creation and manipulation of Reminders. 1317 */ 1318 @MediumTest testReminders()1319 public void testReminders() { 1320 String account = "rem_account"; 1321 int seed = 0; 1322 1323 // Clean up just in case. 1324 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1325 1326 // Create calendar. 1327 long calendarId = createAndVerifyCalendar(account, seed++, null); 1328 1329 // Create events. 1330 ContentValues eventValues; 1331 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1332 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1333 assertTrue(eventId1 >= 0); 1334 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1335 long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1336 assertTrue(eventId2 >= 0); 1337 1338 // No reminders, hasAlarm should be zero. 1339 int hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId1); 1340 assertEquals(0, hasAlarm); 1341 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId2); 1342 assertEquals(0, hasAlarm); 1343 1344 /* 1345 * Add some reminders. 1346 */ 1347 long remId1 = ReminderHelper.addReminder(mContentResolver, eventId1, 1348 10, Reminders.METHOD_DEFAULT); 1349 long remId2 = ReminderHelper.addReminder(mContentResolver, eventId1, 1350 15, Reminders.METHOD_ALERT); 1351 long remId3 = ReminderHelper.addReminder(mContentResolver, eventId1, 1352 20, Reminders.METHOD_SMS); // SMS isn't allowed for this calendar 1353 1354 // Should have been set to 1 by provider. 1355 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId1); 1356 assertEquals(1, hasAlarm); 1357 1358 // Add a reminder to event2. 1359 ReminderHelper.addReminder(mContentResolver, eventId2, 1360 20, Reminders.METHOD_DEFAULT); 1361 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId2); 1362 assertEquals(1, hasAlarm); 1363 1364 1365 /* 1366 * Check the entries. 1367 */ 1368 Cursor cursor = ReminderHelper.findRemindersByEventId(mContentResolver, eventId1); 1369 try { 1370 assertEquals(3, cursor.getCount()); 1371 //DatabaseUtils.dumpCursor(cursor); 1372 1373 while (cursor.moveToNext()) { 1374 int minutes = cursor.getInt(ReminderHelper.REMINDERS_MINUTES_INDEX); 1375 int method = cursor.getInt(ReminderHelper.REMINDERS_METHOD_INDEX); 1376 switch (minutes) { 1377 case 10: 1378 assertEquals(Reminders.METHOD_DEFAULT, method); 1379 break; 1380 case 15: 1381 assertEquals(Reminders.METHOD_ALERT, method); 1382 break; 1383 case 20: 1384 assertEquals(Reminders.METHOD_SMS, method); 1385 break; 1386 default: 1387 fail("unexpected minutes " + minutes); 1388 break; 1389 } 1390 } 1391 } finally { 1392 if (cursor != null) { 1393 cursor.close(); 1394 } 1395 } 1396 1397 /* 1398 * Use the bulk update feature to change all METHOD_DEFAULT to METHOD_EMAIL. To make 1399 * this more interesting we first change remId3 to METHOD_DEFAULT. 1400 */ 1401 int count; 1402 ContentValues newValues = new ContentValues(); 1403 newValues.put(Reminders.METHOD, Reminders.METHOD_DEFAULT); 1404 count = mContentResolver.update(ContentUris.withAppendedId(Reminders.CONTENT_URI, remId3), 1405 newValues, null, null); 1406 assertEquals(1, count); 1407 1408 newValues.put(Reminders.METHOD, Reminders.METHOD_EMAIL); 1409 count = mContentResolver.update(Reminders.CONTENT_URI, newValues, 1410 Reminders.EVENT_ID + "=? AND " + Reminders.METHOD + "=?", 1411 new String[] { 1412 String.valueOf(eventId1), String.valueOf(Reminders.METHOD_DEFAULT) 1413 }); 1414 assertEquals(2, count); 1415 1416 // check it 1417 int method = ReminderHelper.lookupMethod(mContentResolver, remId3); 1418 assertEquals(Reminders.METHOD_EMAIL, method); 1419 1420 /* 1421 * Delete some / all reminders and confirm that hasAlarm tracks it. 1422 * 1423 * You can also remove reminders from an event by updating the event_id column, but 1424 * that's defined as producing undefined behavior, so we don't do it here. 1425 */ 1426 count = mContentResolver.delete(Reminders.CONTENT_URI, 1427 Reminders.EVENT_ID + "=? AND " + Reminders.MINUTES + ">=?", 1428 new String[] { String.valueOf(eventId1), "15" }); 1429 assertEquals(2, count); 1430 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId1); 1431 assertEquals(1, hasAlarm); 1432 1433 // Delete all reminders from both events. 1434 count = mContentResolver.delete(Reminders.CONTENT_URI, 1435 Reminders.EVENT_ID + "=? OR " + Reminders.EVENT_ID + "=?", 1436 new String[] { String.valueOf(eventId1), String.valueOf(eventId2) }); 1437 assertEquals(2, count); 1438 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId1); 1439 assertEquals(0, hasAlarm); 1440 hasAlarm = EventHelper.lookupHasAlarm(mContentResolver, eventId2); 1441 assertEquals(0, hasAlarm); 1442 1443 /* 1444 * Add a couple of reminders and then delete one with the by-ID URI. 1445 */ 1446 long remId4 = ReminderHelper.addReminder(mContentResolver, eventId1, 1447 10, Reminders.METHOD_EMAIL); 1448 long remId5 = ReminderHelper.addReminder(mContentResolver, eventId1, 1449 15, Reminders.METHOD_EMAIL); 1450 count = mContentResolver.delete(ContentUris.withAppendedId(Reminders.CONTENT_URI, remId4), 1451 null, null); 1452 assertEquals(1, count); 1453 1454 removeAndVerifyCalendar(account, calendarId); 1455 } 1456 1457 /** 1458 * A listener for the EVENT_REMINDER broadcast that is expected to be fired by the 1459 * provider at the reminder time. 1460 */ 1461 public class MockReminderReceiver extends BroadcastReceiver { 1462 public boolean received = false; 1463 1464 @Override onReceive(Context context, Intent intent)1465 public void onReceive(Context context, Intent intent) { 1466 final String action = intent.getAction(); 1467 if (action.equals(CalendarContract.ACTION_EVENT_REMINDER)) { 1468 received = true; 1469 } 1470 } 1471 } 1472 1473 /** 1474 * Test that reminders result in the expected broadcast at reminder time. 1475 */ testRemindersAlarm()1476 public void testRemindersAlarm() throws Exception { 1477 // Setup: register a mock listener for the broadcast we expect to fire at the 1478 // reminder time. 1479 final MockReminderReceiver reminderReceiver = new MockReminderReceiver(); 1480 IntentFilter filter = new IntentFilter(CalendarContract.ACTION_EVENT_REMINDER); 1481 filter.addDataScheme("content"); 1482 getInstrumentation().getTargetContext().registerReceiver(reminderReceiver, filter); 1483 1484 // Clean up just in case. 1485 String account = "rem_account"; 1486 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1487 1488 // Create calendar. Use '1' as seed as this sets the VISIBLE field to 1. 1489 // The calendar must be visible for its notifications to occur. 1490 long calendarId = createAndVerifyCalendar(account, 1, null); 1491 1492 // Create event for 15 min in the past, with a 10 min reminder, so that it will 1493 // trigger immediately. 1494 ContentValues eventValues; 1495 int seed = 0; 1496 long now = System.currentTimeMillis(); 1497 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1498 eventValues.put(Events.DTSTART, now - DateUtils.MINUTE_IN_MILLIS * 15); 1499 eventValues.put(Events.DTEND, now + DateUtils.HOUR_IN_MILLIS); 1500 long eventId = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1501 assertTrue(eventId >= 0); 1502 ReminderHelper.addReminder(mContentResolver, eventId, 10, Reminders.METHOD_ALERT); 1503 1504 // Confirm that the EVENT_REMINDER broadcast was fired by the provider. 1505 new PollingCheck(POLLING_TIMEOUT) { 1506 @Override 1507 protected boolean check() { 1508 return reminderReceiver.received; 1509 } 1510 }.run(); 1511 assertTrue(reminderReceiver.received); 1512 1513 removeAndVerifyCalendar(account, calendarId); 1514 } 1515 1516 @MediumTest testColorWriteRequirements()1517 public void testColorWriteRequirements() { 1518 String account = "colw_account"; 1519 String account2 = "colw2_account"; 1520 int seed = 0; 1521 Uri uri = asSyncAdapter(Colors.CONTENT_URI, account, CTS_TEST_TYPE); 1522 Uri uri2 = asSyncAdapter(Colors.CONTENT_URI, account2, CTS_TEST_TYPE); 1523 1524 // Clean up just in case 1525 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1526 ColorHelper.deleteColorsByAccount(mContentResolver, account2, CTS_TEST_TYPE); 1527 1528 ContentValues colorValues = new ContentValues(); 1529 // Account name/type must be in the query params, so may be left 1530 // out here 1531 colorValues.put(Colors.DATA, "0"); 1532 colorValues.put(Colors.COLOR_KEY, "1"); 1533 colorValues.put(Colors.COLOR_TYPE, 0); 1534 colorValues.put(Colors.COLOR, 0xff000000); 1535 1536 // Verify only a sync adapter can write to Colors 1537 try { 1538 mContentResolver.insert(Colors.CONTENT_URI, colorValues); 1539 fail("Should not allow non-sync adapter to insert colors"); 1540 } catch (IllegalArgumentException e) { 1541 // WAI 1542 } 1543 1544 // Verify everything except DATA is required 1545 ContentValues testVals = new ContentValues(colorValues); 1546 for (String key : colorValues.keySet()) { 1547 1548 testVals.remove(key); 1549 try { 1550 Uri colUri = mContentResolver.insert(uri, testVals); 1551 if (!TextUtils.equals(key, Colors.DATA)) { 1552 // The DATA field is allowed to be empty. 1553 fail("Should not allow color creation without " + key); 1554 } 1555 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1556 } catch (IllegalArgumentException e) { 1557 if (TextUtils.equals(key, Colors.DATA)) { 1558 // The DATA field is allowed to be empty. 1559 fail("Should allow color creation without " + key); 1560 } 1561 } 1562 testVals.put(key, colorValues.getAsString(key)); 1563 } 1564 1565 // Verify writing a color works 1566 Uri col1 = mContentResolver.insert(uri, colorValues); 1567 1568 // Verify adding the same color fails 1569 try { 1570 mContentResolver.insert(uri, colorValues); 1571 fail("Should not allow adding the same color twice"); 1572 } catch (IllegalArgumentException e) { 1573 // WAI 1574 } 1575 1576 // Verify specifying a different account than the query params doesn't work 1577 colorValues.put(Colors.ACCOUNT_NAME, account2); 1578 try { 1579 mContentResolver.insert(uri, colorValues); 1580 fail("Should use the account from the query params, not the values."); 1581 } catch (IllegalArgumentException e) { 1582 // WAI 1583 } 1584 1585 // Verify adding a color to a different account works 1586 Uri col2 = mContentResolver.insert(uri2, colorValues); 1587 1588 // And a different index on the same account 1589 colorValues.put(Colors.COLOR_KEY, "2"); 1590 Uri col3 = mContentResolver.insert(uri2, colorValues); 1591 1592 // Verify that all three colors are in the table 1593 Cursor c = ColorHelper.findColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1594 assertEquals(1, c.getCount()); 1595 c.close(); 1596 c = ColorHelper.findColorsByAccount(mContentResolver, account2, CTS_TEST_TYPE); 1597 assertEquals(2, c.getCount()); 1598 c.close(); 1599 1600 // Verify deleting them works 1601 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1602 ColorHelper.deleteColorsByAccount(mContentResolver, account2, CTS_TEST_TYPE); 1603 1604 c = ColorHelper.findColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1605 assertEquals(0, c.getCount()); 1606 c.close(); 1607 c = ColorHelper.findColorsByAccount(mContentResolver, account2, CTS_TEST_TYPE); 1608 assertEquals(0, c.getCount()); 1609 c.close(); 1610 } 1611 1612 /** 1613 * Tests Colors interaction with the Calendars table. 1614 */ 1615 @MediumTest testCalendarColors()1616 public void testCalendarColors() { 1617 String account = "cc_account"; 1618 int seed = 0; 1619 1620 // Clean up just in case 1621 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1622 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1623 1624 // Test inserting a calendar with an invalid color index 1625 ContentValues cv = CalendarHelper.getNewCalendarValues(account, seed++); 1626 cv.put(Calendars.CALENDAR_COLOR_KEY, "badIndex"); 1627 Uri calSyncUri = asSyncAdapter(Calendars.CONTENT_URI, account, CTS_TEST_TYPE); 1628 Uri colSyncUri = asSyncAdapter(Colors.CONTENT_URI, account, CTS_TEST_TYPE); 1629 1630 try { 1631 Uri uri = mContentResolver.insert(calSyncUri, cv); 1632 fail("Should not allow insertion of invalid color index into Calendars"); 1633 } catch (IllegalArgumentException e) { 1634 // WAI 1635 } 1636 1637 // Test updating a calendar with an invalid color index 1638 long calendarId = createAndVerifyCalendar(account, seed++, null); 1639 cv.clear(); 1640 cv.put(Calendars.CALENDAR_COLOR_KEY, "badIndex2"); 1641 Uri calendarUri = ContentUris.withAppendedId(Calendars.CONTENT_URI, calendarId); 1642 try { 1643 mContentResolver.update(calendarUri, cv, null, null); 1644 fail("Should not allow update of invalid color index into Calendars"); 1645 } catch (IllegalArgumentException e) { 1646 // WAI 1647 } 1648 1649 assertTrue(ColorHelper.addDefaultColorsToAccount(mContentResolver, account, CTS_TEST_TYPE)); 1650 1651 // Test that inserting a valid color index works 1652 cv = CalendarHelper.getNewCalendarValues(account, seed++); 1653 cv.put(Calendars.CALENDAR_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_0]); 1654 1655 Uri uri = mContentResolver.insert(calSyncUri, cv); 1656 long calendarId2 = ContentUris.parseId(uri); 1657 assertTrue(calendarId2 >= 0); 1658 // And updates the calendar's color to the one in the table 1659 cv.put(Calendars.CALENDAR_COLOR, ColorHelper.DEFAULT_COLORS[ColorHelper.C_COLOR_0]); 1660 verifyCalendar(account, cv, calendarId2, 2); 1661 1662 // Test that updating a valid color index also updates the color in a 1663 // calendar 1664 cv.clear(); 1665 cv.put(Calendars.CALENDAR_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_0]); 1666 mContentResolver.update(calendarUri, cv, null, null); 1667 Cursor c = mContentResolver.query(calendarUri, 1668 new String[] { Calendars.CALENDAR_COLOR_KEY, Calendars.CALENDAR_COLOR }, 1669 null, null, null); 1670 try { 1671 c.moveToFirst(); 1672 String index = c.getString(0); 1673 int color = c.getInt(1); 1674 assertEquals(index, ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_0]); 1675 assertEquals(color, ColorHelper.DEFAULT_COLORS[ColorHelper.C_COLOR_0]); 1676 } finally { 1677 if (c != null) { 1678 c.close(); 1679 } 1680 } 1681 1682 // And clearing it doesn't change the color 1683 cv.put(Calendars.CALENDAR_COLOR_KEY, (String) null); 1684 mContentResolver.update(calendarUri, cv, null, null); 1685 c = mContentResolver.query(calendarUri, 1686 new String[] { Calendars.CALENDAR_COLOR_KEY, Calendars.CALENDAR_COLOR }, 1687 null, null, null); 1688 try { 1689 c.moveToFirst(); 1690 String index = c.getString(0); 1691 int color = c.getInt(1); 1692 assertEquals(index, null); 1693 assertEquals(ColorHelper.DEFAULT_COLORS[ColorHelper.C_COLOR_0], color); 1694 } finally { 1695 if (c != null) { 1696 c.close(); 1697 } 1698 } 1699 1700 // Test that setting a calendar color to an event color fails 1701 cv.put(Calendars.CALENDAR_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_0]); 1702 try { 1703 mContentResolver.update(calendarUri, cv, null, null); 1704 fail("Should not allow a calendar to use an event color"); 1705 } catch (IllegalArgumentException e) { 1706 // WAI 1707 } 1708 1709 // Test that you can't remove a color that is referenced by a calendar 1710 cv.put(Calendars.CALENDAR_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_3]); 1711 mContentResolver.update(calendarUri, cv, null, null); 1712 1713 try { 1714 mContentResolver.delete(colSyncUri, ColorHelper.WHERE_COLOR_ACCOUNT_AND_INDEX, 1715 new String[] { 1716 account, CTS_TEST_TYPE, 1717 ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_3] 1718 }); 1719 fail("Should not allow deleting referenced color"); 1720 } catch (UnsupportedOperationException e) { 1721 // WAI 1722 } 1723 1724 // Clean up 1725 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1726 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1727 } 1728 1729 /** 1730 * Tests Colors interaction with the Events table. 1731 */ 1732 @MediumTest testEventColors()1733 public void testEventColors() { 1734 String account = "ec_account"; 1735 int seed = 0; 1736 1737 // Clean up just in case 1738 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1739 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1740 1741 // Test inserting an event with an invalid color index 1742 long cal_id = createAndVerifyCalendar(account, seed++, null); 1743 1744 Uri colSyncUri = asSyncAdapter(Colors.CONTENT_URI, account, CTS_TEST_TYPE); 1745 1746 ContentValues ev = EventHelper.getNewEventValues(account, seed++, cal_id, false); 1747 ev.put(Events.EVENT_COLOR_KEY, "badIndex"); 1748 1749 try { 1750 Uri uri = mContentResolver.insert(Events.CONTENT_URI, ev); 1751 fail("Should not allow insertion of invalid color index into Events"); 1752 } catch (IllegalArgumentException e) { 1753 // WAI 1754 } 1755 1756 // Test updating an event with an invalid color index fails 1757 long event_id = createAndVerifyEvent(account, seed++, cal_id, false, null); 1758 ev.clear(); 1759 ev.put(Events.EVENT_COLOR_KEY, "badIndex2"); 1760 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, event_id); 1761 try { 1762 mContentResolver.update(eventUri, ev, null, null); 1763 fail("Should not allow update of invalid color index into Events"); 1764 } catch (IllegalArgumentException e) { 1765 // WAI 1766 } 1767 1768 assertTrue(ColorHelper.addDefaultColorsToAccount(mContentResolver, account, CTS_TEST_TYPE)); 1769 1770 // Test that inserting a valid color index works 1771 ev = EventHelper.getNewEventValues(account, seed++, cal_id, false); 1772 final String defaultColorIndex = ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_0]; 1773 ev.put(Events.EVENT_COLOR_KEY, defaultColorIndex); 1774 1775 Uri uri = mContentResolver.insert(Events.CONTENT_URI, ev); 1776 long eventId2 = ContentUris.parseId(uri); 1777 assertTrue(eventId2 >= 0); 1778 // And updates the event's color to the one in the table 1779 final int expectedColor = ColorHelper.DEFAULT_COLORS[ColorHelper.E_COLOR_0]; 1780 ev.put(Events.EVENT_COLOR, expectedColor); 1781 verifyEvent(ev, eventId2); 1782 1783 // Test that event iterator has COLOR columns 1784 final EntityIterator iterator = EventsEntity.newEntityIterator(mContentResolver.query( 1785 ContentUris.withAppendedId(EventsEntity.CONTENT_URI, eventId2), 1786 null, null, null, null), mContentResolver); 1787 assertTrue("Empty Iterator", iterator.hasNext()); 1788 final Entity entity = iterator.next(); 1789 final ContentValues values = entity.getEntityValues(); 1790 assertTrue("Missing EVENT_COLOR", values.containsKey(EventsEntity.EVENT_COLOR)); 1791 assertEquals("Wrong EVENT_COLOR", 1792 expectedColor, 1793 (int) values.getAsInteger(EventsEntity.EVENT_COLOR)); 1794 assertTrue("Missing EVENT_COLOR_KEY", values.containsKey(EventsEntity.EVENT_COLOR_KEY)); 1795 assertEquals("Wrong EVENT_COLOR_KEY", 1796 defaultColorIndex, 1797 values.getAsString(EventsEntity.EVENT_COLOR_KEY)); 1798 iterator.close(); 1799 1800 // Test that updating a valid color index also updates the color in an 1801 // event 1802 ev.clear(); 1803 ev.put(Events.EVENT_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_1]); 1804 mContentResolver.update(eventUri, ev, null, null); 1805 Cursor c = mContentResolver.query(eventUri, new String[] { 1806 Events.EVENT_COLOR_KEY, Events.EVENT_COLOR 1807 }, null, null, null); 1808 try { 1809 c.moveToFirst(); 1810 String index = c.getString(0); 1811 int color = c.getInt(1); 1812 assertEquals(index, ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_1]); 1813 assertEquals(color, ColorHelper.DEFAULT_COLORS[ColorHelper.E_COLOR_1]); 1814 } finally { 1815 if (c != null) { 1816 c.close(); 1817 } 1818 } 1819 1820 // And clearing it doesn't change the color 1821 ev.put(Events.EVENT_COLOR_KEY, (String) null); 1822 mContentResolver.update(eventUri, ev, null, null); 1823 c = mContentResolver.query(eventUri, new String[] { 1824 Events.EVENT_COLOR_KEY, Events.EVENT_COLOR 1825 }, null, null, null); 1826 try { 1827 c.moveToFirst(); 1828 String index = c.getString(0); 1829 int color = c.getInt(1); 1830 assertEquals(index, null); 1831 assertEquals(ColorHelper.DEFAULT_COLORS[ColorHelper.E_COLOR_1], color); 1832 } finally { 1833 if (c != null) { 1834 c.close(); 1835 } 1836 } 1837 1838 // Test that setting an event color to a calendar color fails 1839 ev.put(Events.EVENT_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.C_COLOR_2]); 1840 try { 1841 mContentResolver.update(eventUri, ev, null, null); 1842 fail("Should not allow an event to use a calendar color"); 1843 } catch (IllegalArgumentException e) { 1844 // WAI 1845 } 1846 1847 // Test that you can't remove a color that is referenced by an event 1848 ev.put(Events.EVENT_COLOR_KEY, ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_1]); 1849 mContentResolver.update(eventUri, ev, null, null); 1850 try { 1851 mContentResolver.delete(colSyncUri, ColorHelper.WHERE_COLOR_ACCOUNT_AND_INDEX, 1852 new String[] { 1853 account, CTS_TEST_TYPE, 1854 ColorHelper.DEFAULT_INDICES[ColorHelper.E_COLOR_1] 1855 }); 1856 fail("Should not allow deleting referenced color"); 1857 } catch (UnsupportedOperationException e) { 1858 // WAI 1859 } 1860 1861 // TODO test colors with exceptions 1862 1863 // Clean up 1864 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1865 ColorHelper.deleteColorsByAccount(mContentResolver, account, CTS_TEST_TYPE); 1866 } 1867 1868 /** 1869 * Tests creation and manipulation of ExtendedProperties. 1870 */ 1871 @MediumTest testExtendedProperties()1872 public void testExtendedProperties() { 1873 String account = "ep_account"; 1874 int seed = 0; 1875 1876 // Clean up just in case. 1877 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1878 1879 // Create calendar. 1880 long calendarId = createAndVerifyCalendar(account, seed++, null); 1881 1882 // Create events. 1883 ContentValues eventValues; 1884 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 1885 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 1886 assertTrue(eventId1 >= 0); 1887 1888 /* 1889 * Add some extended properties. 1890 */ 1891 long epId1 = ExtendedPropertiesHelper.addExtendedProperty(mContentResolver, account, 1892 eventId1, "first", "Jeffrey"); 1893 long epId2 = ExtendedPropertiesHelper.addExtendedProperty(mContentResolver, account, 1894 eventId1, "last", "Lebowski"); 1895 long epId3 = ExtendedPropertiesHelper.addExtendedProperty(mContentResolver, account, 1896 eventId1, "title", "Dude"); 1897 1898 /* 1899 * Spot-check a couple of entries. 1900 */ 1901 Cursor cursor = ExtendedPropertiesHelper.findExtendedPropertiesByEventId(mContentResolver, 1902 eventId1); 1903 try { 1904 assertEquals(3, cursor.getCount()); 1905 //DatabaseUtils.dumpCursor(cursor); 1906 1907 while (cursor.moveToNext()) { 1908 String name = 1909 cursor.getString(ExtendedPropertiesHelper.EXTENDED_PROPERTIES_NAME_INDEX); 1910 String value = 1911 cursor.getString(ExtendedPropertiesHelper.EXTENDED_PROPERTIES_VALUE_INDEX); 1912 1913 if (name.equals("last")) { 1914 assertEquals("Lebowski", value); 1915 } 1916 } 1917 1918 String title = ExtendedPropertiesHelper.lookupValueByName(mContentResolver, eventId1, 1919 "title"); 1920 assertEquals("Dude", title); 1921 } finally { 1922 if (cursor != null) { 1923 cursor.close(); 1924 } 1925 } 1926 1927 // Update the title. Must be done as a sync adapter. 1928 ContentValues newValues = new ContentValues(); 1929 newValues.put(ExtendedProperties.VALUE, "Big"); 1930 Uri uri = ContentUris.withAppendedId(ExtendedProperties.CONTENT_URI, epId3); 1931 uri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 1932 int count = mContentResolver.update(uri, newValues, null, null); 1933 assertEquals(1, count); 1934 1935 // check it 1936 String title = ExtendedPropertiesHelper.lookupValueByName(mContentResolver, eventId1, 1937 "title"); 1938 assertEquals("Big", title); 1939 1940 removeAndVerifyCalendar(account, calendarId); 1941 } 1942 1943 private class CalendarEventHelper { 1944 1945 private long mCalendarId; 1946 private String mAccount; 1947 private int mSeed; 1948 CalendarEventHelper(String account, int seed)1949 public CalendarEventHelper(String account, int seed) { 1950 mAccount = account; 1951 mSeed = seed; 1952 ContentValues values = CalendarHelper.getNewCalendarValues(account, seed); 1953 mCalendarId = createAndVerifyCalendar(account, seed++, values); 1954 } 1955 addEvent(String timeString, int timeZoneIndex, long duration)1956 public ContentValues addEvent(String timeString, int timeZoneIndex, long duration) { 1957 long event1Start = timeInMillis(timeString, timeZoneIndex); 1958 ContentValues eventValues; 1959 eventValues = EventHelper.getNewEventValues(mAccount, mSeed++, mCalendarId, true); 1960 eventValues.put(Events.DESCRIPTION, timeString); 1961 eventValues.put(Events.DTSTART, event1Start); 1962 eventValues.put(Events.DTEND, event1Start + duration); 1963 eventValues.put(Events.EVENT_TIMEZONE, TIME_ZONES[timeZoneIndex]); 1964 long eventId = createAndVerifyEvent(mAccount, mSeed, mCalendarId, true, eventValues); 1965 assertTrue(eventId >= 0); 1966 return eventValues; 1967 } 1968 getCalendarId()1969 public long getCalendarId() { 1970 return mCalendarId; 1971 } 1972 } 1973 1974 /** 1975 * Test query to retrieve instances within a certain time interval. 1976 */ testWhenByDayQuery()1977 public void testWhenByDayQuery() { 1978 String account = "cser_account"; 1979 int seed = 0; 1980 1981 // Clean up just in case 1982 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 1983 1984 // Create a calendar 1985 CalendarEventHelper helper = new CalendarEventHelper(account, seed); 1986 1987 // Add events to the calendar--the first two in the queried range 1988 List<ContentValues> eventsWithinRange = new ArrayList<ContentValues>(); 1989 1990 ContentValues values = helper.addEvent("2009-10-01T08:00:00", 0, DateUtils.HOUR_IN_MILLIS); 1991 eventsWithinRange.add(values); 1992 1993 values = helper.addEvent("2010-10-01T08:00:00", 0, DateUtils.HOUR_IN_MILLIS); 1994 eventsWithinRange.add(values); 1995 1996 helper.addEvent("2011-10-01T08:00:00", 0, DateUtils.HOUR_IN_MILLIS); 1997 1998 // Prepare the start time and end time of the range to query 1999 String startTime = "2009-01-01T00:00:00"; 2000 String endTime = "2011-01-01T00:00:00"; 2001 int julianStart = getJulianDay(startTime, 0); 2002 int julianEnd = getJulianDay(endTime, 0); 2003 Uri uri = Uri.withAppendedPath( 2004 CalendarContract.Instances.CONTENT_BY_DAY_URI, julianStart + "/" + julianEnd); 2005 2006 // Query the range, sorting by event start time 2007 Cursor c = mContentResolver.query(uri, null, Instances.CALENDAR_ID + "=" 2008 + helper.getCalendarId(), null, Events.DTSTART); 2009 2010 // Assert that two events are returned 2011 assertEquals(c.getCount(), 2); 2012 2013 Set<String> keySet = new HashSet(); 2014 keySet.add(Events.DESCRIPTION); 2015 keySet.add(Events.DTSTART); 2016 keySet.add(Events.DTEND); 2017 keySet.add(Events.EVENT_TIMEZONE); 2018 2019 // Verify that the contents of those two events match the cursor results 2020 verifyContentValuesAgainstCursor(eventsWithinRange, keySet, c); 2021 } 2022 verifyContentValuesAgainstCursor(List<ContentValues> cvs, Set<String> keys, Cursor cursor)2023 private void verifyContentValuesAgainstCursor(List<ContentValues> cvs, 2024 Set<String> keys, Cursor cursor) { 2025 assertEquals(cursor.getCount(), cvs.size()); 2026 2027 cursor.moveToFirst(); 2028 2029 int i=0; 2030 do { 2031 ContentValues cv = cvs.get(i); 2032 for (String key : keys) { 2033 assertEquals(cv.get(key).toString(), 2034 cursor.getString(cursor.getColumnIndex(key))); 2035 } 2036 i++; 2037 } while (cursor.moveToNext()); 2038 2039 cursor.close(); 2040 } 2041 timeInMillis(String timeString, int timeZoneIndex)2042 private long timeInMillis(String timeString, int timeZoneIndex) { 2043 Time startTime = new Time(TIME_ZONES[timeZoneIndex]); 2044 startTime.parse3339(timeString); 2045 return startTime.toMillis(false); 2046 } 2047 getJulianDay(String timeString, int timeZoneIndex)2048 private int getJulianDay(String timeString, int timeZoneIndex) { 2049 Time time = new Time(TIME_ZONES[timeZoneIndex]); 2050 time.parse3339(timeString); 2051 return Time.getJulianDay(time.toMillis(false), time.gmtoff); 2052 } 2053 2054 /** 2055 * Test instance queries with search parameters. 2056 */ 2057 @MediumTest testInstanceSearch()2058 public void testInstanceSearch() { 2059 String account = "cser_account"; 2060 int seed = 0; 2061 2062 // Clean up just in case 2063 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2064 2065 // Create a calendar 2066 ContentValues values = CalendarHelper.getNewCalendarValues(account, seed); 2067 long calendarId = createAndVerifyCalendar(account, seed++, values); 2068 2069 String testStart = "2009-10-01T08:00:00"; 2070 String timeZone = TIME_ZONES[0]; 2071 Time startTime = new Time(timeZone); 2072 startTime.parse3339(testStart); 2073 long startMillis = startTime.toMillis(false); 2074 2075 // Create some events, with different descriptions. (Could also create a single 2076 // recurring event and some instance exceptions.) 2077 ContentValues eventValues; 2078 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2079 eventValues.put(Events.DESCRIPTION, "testevent event-one fiddle"); 2080 eventValues.put(Events.DTSTART, startMillis); 2081 eventValues.put(Events.DTEND, startMillis + DateUtils.HOUR_IN_MILLIS); 2082 eventValues.put(Events.EVENT_TIMEZONE, timeZone); 2083 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2084 assertTrue(eventId1 >= 0); 2085 2086 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2087 eventValues.put(Events.DESCRIPTION, "testevent event-two fuzzle"); 2088 eventValues.put(Events.DTSTART, startMillis + DateUtils.HOUR_IN_MILLIS); 2089 eventValues.put(Events.DTEND, startMillis + DateUtils.HOUR_IN_MILLIS * 2); 2090 eventValues.put(Events.EVENT_TIMEZONE, timeZone); 2091 long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2092 assertTrue(eventId2 >= 0); 2093 2094 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2095 eventValues.put(Events.DESCRIPTION, "testevent event-three fiddle"); 2096 eventValues.put(Events.DTSTART, startMillis + DateUtils.HOUR_IN_MILLIS * 2); 2097 eventValues.put(Events.DTEND, startMillis + DateUtils.HOUR_IN_MILLIS * 3); 2098 eventValues.put(Events.EVENT_TIMEZONE, timeZone); 2099 long eventId3 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2100 assertTrue(eventId3 >= 0); 2101 2102 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2103 eventValues.put(Events.DESCRIPTION, "nontestevent"); 2104 eventValues.put(Events.DTSTART, startMillis + (long) (DateUtils.HOUR_IN_MILLIS * 1.5f)); 2105 eventValues.put(Events.DTEND, startMillis + DateUtils.HOUR_IN_MILLIS * 2); 2106 eventValues.put(Events.EVENT_TIMEZONE, timeZone); 2107 long eventId4 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2108 assertTrue(eventId4 >= 0); 2109 2110 String rangeStart = "2009-10-01T00:00:00"; 2111 String rangeEnd = "2009-10-01T11:59:59"; 2112 String[] projection = new String[] { Instances.BEGIN }; 2113 2114 if (false) { 2115 Cursor instances = getInstances(timeZone, rangeStart, rangeEnd, projection, 2116 new long[] { calendarId }); 2117 dumpInstances(instances, timeZone, "all"); 2118 instances.close(); 2119 } 2120 2121 Cursor instances; 2122 int count; 2123 2124 // Find all matching "testevent". The search matches on partial strings, so this 2125 // will also pick up "nontestevent". 2126 instances = getInstancesSearch(timeZone, rangeStart, rangeEnd, 2127 "testevent", false, projection, new long[] { calendarId }); 2128 count = instances.getCount(); 2129 instances.close(); 2130 assertEquals(4, count); 2131 2132 // Find all matching "fiddle" and "event". Set the "by day" flag just to be different. 2133 instances = getInstancesSearch(timeZone, rangeStart, rangeEnd, 2134 "fiddle event", true, projection, new long[] { calendarId }); 2135 count = instances.getCount(); 2136 instances.close(); 2137 assertEquals(2, count); 2138 2139 // Find all matching "fiddle" and "baluchitherium". 2140 instances = getInstancesSearch(timeZone, rangeStart, rangeEnd, 2141 "baluchitherium fiddle", false, projection, new long[] { calendarId }); 2142 count = instances.getCount(); 2143 instances.close(); 2144 assertEquals(0, count); 2145 2146 // Find all matching "event-two". 2147 instances = getInstancesSearch(timeZone, rangeStart, rangeEnd, 2148 "event-two", false, projection, new long[] { calendarId }); 2149 count = instances.getCount(); 2150 instances.close(); 2151 assertEquals(1, count); 2152 2153 removeAndVerifyCalendar(account, calendarId); 2154 } 2155 2156 @MediumTest testCalendarUpdateAsApp()2157 public void testCalendarUpdateAsApp() { 2158 String account = "cu1_account"; 2159 int seed = 0; 2160 2161 // Clean up just in case 2162 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2163 2164 // Create a calendar 2165 ContentValues values = CalendarHelper.getNewCalendarValues(account, seed); 2166 long id = createAndVerifyCalendar(account, seed++, values); 2167 2168 Uri uri = ContentUris.withAppendedId(Calendars.CONTENT_URI, id); 2169 2170 // Update the calendar using the direct Uri 2171 ContentValues updateValues = CalendarHelper.getUpdateCalendarValuesWithOriginal( 2172 values, seed++); 2173 assertEquals(1, mContentResolver.update(uri, updateValues, null, null)); 2174 2175 verifyCalendar(account, values, id, 1); 2176 2177 // Update the calendar using selection + args 2178 String selection = Calendars._ID + "=?"; 2179 String[] selectionArgs = new String[] { Long.toString(id) }; 2180 2181 updateValues = CalendarHelper.getUpdateCalendarValuesWithOriginal(values, seed++); 2182 2183 assertEquals(1, mContentResolver.update( 2184 Calendars.CONTENT_URI, updateValues, selection, selectionArgs)); 2185 2186 verifyCalendar(account, values, id, 1); 2187 2188 removeAndVerifyCalendar(account, id); 2189 } 2190 2191 // TODO test calendar updates as sync adapter 2192 2193 /** 2194 * Test access to the "syncstate" table. 2195 */ 2196 @MediumTest testSyncState()2197 public void testSyncState() { 2198 String account = "ss_account"; 2199 int seed = 0; 2200 2201 // Clean up just in case 2202 SyncStateHelper.deleteSyncStateByAccount(mContentResolver, account, true); 2203 2204 // Create a new sync state entry 2205 ContentValues values = SyncStateHelper.getNewSyncStateValues(account); 2206 long id = createAndVerifySyncState(account, values); 2207 2208 // Look it up with the by-ID URI 2209 Cursor c = SyncStateHelper.getSyncStateById(mContentResolver, id); 2210 assertNotNull(c); 2211 assertEquals(1, c.getCount()); 2212 c.close(); 2213 2214 // Try to remove it as non-sync-adapter; expected to fail. 2215 boolean failed; 2216 try { 2217 SyncStateHelper.deleteSyncStateByAccount(mContentResolver, account, false); 2218 failed = false; 2219 } catch (IllegalArgumentException iae) { 2220 failed = true; 2221 } 2222 assertTrue("deletion of sync state by app was allowed", failed); 2223 2224 // Remove it and verify that it's gone 2225 removeAndVerifySyncState(account); 2226 } 2227 2228 verifyEvent(ContentValues values, long eventId)2229 private void verifyEvent(ContentValues values, long eventId) { 2230 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 2231 // Verify 2232 Cursor c = mContentResolver 2233 .query(eventUri, EventHelper.EVENTS_PROJECTION, null, null, null); 2234 assertEquals(1, c.getCount()); 2235 assertTrue(c.moveToFirst()); 2236 assertEquals(eventId, c.getLong(0)); 2237 for (String key : values.keySet()) { 2238 int index = c.getColumnIndex(key); 2239 assertEquals(key, values.getAsString(key), c.getString(index)); 2240 } 2241 c.close(); 2242 } 2243 2244 @MediumTest testEventCreationAndDeletion()2245 public void testEventCreationAndDeletion() { 2246 String account = "ec1_account"; 2247 int seed = 0; 2248 2249 // Clean up just in case 2250 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2251 2252 // Create calendar and event 2253 long calendarId = createAndVerifyCalendar(account, seed++, null); 2254 2255 ContentValues eventValues = EventHelper 2256 .getNewEventValues(account, seed++, calendarId, true); 2257 long eventId = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2258 2259 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 2260 2261 removeAndVerifyEvent(eventUri, eventValues, account); 2262 2263 // Attempt to create an event without a calendar ID. 2264 ContentValues badValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2265 badValues.remove(Events.CALENDAR_ID); 2266 try { 2267 createAndVerifyEvent(account, seed, calendarId, true, badValues); 2268 fail("was allowed to create an event without CALENDAR_ID"); 2269 } catch (IllegalArgumentException iae) { 2270 // expected 2271 } 2272 2273 // Validation may be relaxed for content providers, so test missing timezone as app. 2274 badValues = EventHelper.getNewEventValues(account, seed++, calendarId, false); 2275 badValues.remove(Events.EVENT_TIMEZONE); 2276 try { 2277 createAndVerifyEvent(account, seed, calendarId, false, badValues); 2278 fail("was allowed to create an event without EVENT_TIMEZONE"); 2279 } catch (IllegalArgumentException iae) { 2280 // expected 2281 } 2282 2283 removeAndVerifyCalendar(account, calendarId); 2284 } 2285 2286 @MediumTest testEventUpdateAsApp()2287 public void testEventUpdateAsApp() { 2288 String account = "em1_account"; 2289 int seed = 0; 2290 2291 // Clean up just in case 2292 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2293 2294 // Create calendar 2295 long calendarId = createAndVerifyCalendar(account, seed++, null); 2296 2297 // Create event as sync adapter 2298 ContentValues eventValues = EventHelper 2299 .getNewEventValues(account, seed++, calendarId, true); 2300 long eventId = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2301 2302 // Update event as app 2303 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 2304 2305 ContentValues updateValues = EventHelper.getUpdateEventValuesWithOriginal(eventValues, 2306 seed++, false); 2307 assertEquals(1, mContentResolver.update(eventUri, updateValues, null, null)); 2308 updateValues.put(Events.DIRTY, 1); // provider should have marked as dirty 2309 verifyEvent(updateValues, eventId); 2310 2311 // Try nulling out a required value. 2312 ContentValues badValues = new ContentValues(updateValues); 2313 badValues.putNull(Events.EVENT_TIMEZONE); 2314 badValues.remove(Events.DIRTY); 2315 try { 2316 mContentResolver.update(eventUri, badValues, null, null); 2317 fail("was allowed to null out EVENT_TIMEZONE"); 2318 } catch (IllegalArgumentException iae) { 2319 // good 2320 } 2321 2322 removeAndVerifyEvent(eventUri, eventValues, account); 2323 2324 // delete the calendar 2325 removeAndVerifyCalendar(account, calendarId); 2326 } 2327 2328 /** 2329 * Tests update of multiple events with a single update call. 2330 */ 2331 @MediumTest testBulkUpdate()2332 public void testBulkUpdate() { 2333 String account = "bup_account"; 2334 int seed = 0; 2335 2336 // Clean up just in case 2337 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2338 2339 // Create calendar 2340 long calendarId = createAndVerifyCalendar(account, seed++, null); 2341 String calendarIdStr = String.valueOf(calendarId); 2342 2343 // Create events 2344 ContentValues eventValues; 2345 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2346 long eventId1 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2347 2348 eventValues = EventHelper.getNewEventValues(account, seed++, calendarId, true); 2349 long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2350 2351 // Update the "description" field in all events in this calendar. 2352 String newDescription = "bulk edit"; 2353 ContentValues updateValues = new ContentValues(); 2354 updateValues.put(Events.DESCRIPTION, newDescription); 2355 2356 // Must be sync adapter to do a bulk update. 2357 Uri uri = asSyncAdapter(Events.CONTENT_URI, account, CTS_TEST_TYPE); 2358 int count = mContentResolver.update(uri, updateValues, SQL_WHERE_CALENDAR_ID, 2359 new String[] { calendarIdStr }); 2360 2361 // Check to see if the changes went through. 2362 Uri eventUri = Events.CONTENT_URI; 2363 Cursor c = mContentResolver.query(eventUri, new String[] { Events.DESCRIPTION }, 2364 SQL_WHERE_CALENDAR_ID, new String[] { calendarIdStr }, null); 2365 assertEquals(2, c.getCount()); 2366 while (c.moveToNext()) { 2367 assertEquals(newDescription, c.getString(0)); 2368 } 2369 c.close(); 2370 2371 // delete the calendar 2372 removeAndVerifyCalendar(account, calendarId); 2373 } 2374 2375 /** 2376 * Tests the content provider's enforcement of restrictions on who is allowed to modify 2377 * specific columns in a Calendar. 2378 * <p> 2379 * This attempts to create a new row in the Calendar table, specifying one restricted 2380 * column at a time. 2381 */ 2382 @MediumTest testSyncOnlyInsertEnforcement()2383 public void testSyncOnlyInsertEnforcement() { 2384 // These operations should not succeed, so there should be nothing to clean up after. 2385 // TODO: this should be a new event augmented with an illegal column, not a single 2386 // column. Otherwise we might be tripping over a "DTSTART must exist" test. 2387 ContentValues vals = new ContentValues(); 2388 for (int i = 0; i < Calendars.SYNC_WRITABLE_COLUMNS.length; i++) { 2389 boolean threw = false; 2390 try { 2391 vals.clear(); 2392 vals.put(Calendars.SYNC_WRITABLE_COLUMNS[i], "1"); 2393 mContentResolver.insert(Calendars.CONTENT_URI, vals); 2394 } catch (IllegalArgumentException e) { 2395 threw = true; 2396 } 2397 assertTrue("Only sync adapter should be allowed to insert " 2398 + Calendars.SYNC_WRITABLE_COLUMNS[i], threw); 2399 } 2400 } 2401 2402 /** 2403 * Tests creation of a recurring event. 2404 * <p> 2405 * This (and the other recurrence tests) uses dates well in the past to reduce the likelihood 2406 * of encountering non-test recurring events. (Ideally we would select events associated 2407 * with a specific calendar.) With dates well in the past, it's also important to have a 2408 * fixed maximum count or end date; otherwise, if the metadata min/max instance values are 2409 * large enough, the recurrence recalculation processor could get triggered on an insert or 2410 * update and bump up against the 2000-instance limit. 2411 * 2412 * TODO: need some allDay tests 2413 */ 2414 @MediumTest testRecurrence()2415 public void testRecurrence() { 2416 String account = "re_account"; 2417 int seed = 0; 2418 2419 // Clean up just in case 2420 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2421 2422 // Create calendar 2423 long calendarId = createAndVerifyCalendar(account, seed++, null); 2424 2425 // Create recurring event 2426 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 2427 calendarId, true, "2003-08-05T09:00:00", "PT1H", 2428 "FREQ=WEEKLY;INTERVAL=2;COUNT=4;BYDAY=TU,SU;WKST=SU"); 2429 long eventId = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2430 //Log.d(TAG, "+++ basic recurrence eventId is " + eventId); 2431 2432 // Check to see if we have the expected number of instances 2433 String timeZone = eventValues.getAsString(Events.EVENT_TIMEZONE); 2434 int instanceCount = getInstanceCount(timeZone, "2003-08-05T00:00:00", 2435 "2003-08-31T11:59:59", new long[] { calendarId }); 2436 if (false) { 2437 Cursor instances = getInstances(timeZone, "2003-08-05T00:00:00", "2003-08-31T11:59:59", 2438 new String[] { Instances.BEGIN }, new long[] { calendarId }); 2439 dumpInstances(instances, timeZone, "initial"); 2440 instances.close(); 2441 } 2442 assertEquals("recurrence instance count", 4, instanceCount); 2443 2444 // delete the calendar 2445 removeAndVerifyCalendar(account, calendarId); 2446 } 2447 2448 /** 2449 * Tests conversion of a regular event to a recurring event. 2450 */ 2451 @MediumTest testConversionToRecurring()2452 public void testConversionToRecurring() { 2453 String account = "reconv_account"; 2454 int seed = 0; 2455 2456 // Clean up just in case 2457 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2458 2459 // Create calendar and event 2460 long calendarId = createAndVerifyCalendar(account, seed++, null); 2461 2462 ContentValues eventValues = EventHelper 2463 .getNewEventValues(account, seed++, calendarId, true); 2464 long eventId = createAndVerifyEvent(account, seed, calendarId, true, eventValues); 2465 2466 long dtstart = eventValues.getAsLong(Events.DTSTART); 2467 long dtend = eventValues.getAsLong(Events.DTEND); 2468 long durationSecs = (dtend - dtstart) / 1000; 2469 2470 ContentValues updateValues = new ContentValues(); 2471 updateValues.put(Events.RRULE, "FREQ=WEEKLY"); // recurs forever 2472 updateValues.put(Events.DURATION, "P" + durationSecs + "S"); 2473 updateValues.putNull(Events.DTEND); 2474 2475 // Issue update; do it as app instead of sync adapter to exercise that path. 2476 updateAndVerifyEvent(account, calendarId, eventId, false, updateValues); 2477 2478 // Make sure LAST_DATE got nulled out by our infinitely repeating sequence. 2479 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 2480 Cursor c = mContentResolver.query(eventUri, new String[] { Events.LAST_DATE }, 2481 null, null, null); 2482 assertEquals(1, c.getCount()); 2483 assertTrue(c.moveToFirst()); 2484 assertNull(c.getString(0)); 2485 c.close(); 2486 2487 removeAndVerifyCalendar(account, calendarId); 2488 } 2489 2490 /** 2491 * Tests creation of a recurring event with single-instance exceptions. 2492 */ 2493 @MediumTest testSingleRecurrenceExceptions()2494 public void testSingleRecurrenceExceptions() { 2495 String account = "rex_account"; 2496 int seed = 0; 2497 2498 // Clean up just in case 2499 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2500 2501 // Create calendar 2502 long calendarId = createAndVerifyCalendar(account, seed++, null); 2503 2504 // Create recurring event. 2505 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 2506 calendarId, true, "1999-03-28T09:00:00", "PT1H", "FREQ=WEEKLY;WKST=SU;COUNT=100"); 2507 long eventId = createAndVerifyEvent(account, seed++, calendarId, true, eventValues); 2508 2509 // Add some attendees and reminders. 2510 addAttendees(account, eventId, seed); 2511 addReminders(account, eventId, seed); 2512 2513 // Select a period that gives us 5 instances. We don't want this to straddle a DST 2514 // transition, because we expect the startMinute field to be the same for all 2515 // instances, and it's stored as minutes since midnight in the device's time zone. 2516 // Things won't be consistent if the event and the device have different ideas about DST. 2517 String timeZone = eventValues.getAsString(Events.EVENT_TIMEZONE); 2518 String testStart = "1999-07-02T00:00:00"; 2519 String testEnd = "1999-08-04T23:59:59"; 2520 String[] projection = { Instances.BEGIN, Instances.START_MINUTE, Instances.END_MINUTE }; 2521 2522 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 2523 new long[] { calendarId }); 2524 if (DEBUG_RECURRENCE) { 2525 dumpInstances(instances, timeZone, "initial"); 2526 } 2527 2528 assertEquals("initial recurrence instance count", 5, instances.getCount()); 2529 2530 /* 2531 * Advance the start time of a few instances, and verify. 2532 */ 2533 2534 // Leave first instance alone. 2535 instances.moveToPosition(1); 2536 2537 long startMillis; 2538 ContentValues excepValues; 2539 2540 // Advance the start time of the 2nd instance. 2541 startMillis = instances.getLong(0); 2542 excepValues = EventHelper.getNewExceptionValues(startMillis); 2543 excepValues.put(Events.DTSTART, startMillis + 3600*1000); 2544 long excepEventId2 = createAndVerifyException(account, eventId, excepValues, true); 2545 instances.moveToNext(); 2546 2547 // Advance the start time of the 3rd instance. 2548 startMillis = instances.getLong(0); 2549 excepValues = EventHelper.getNewExceptionValues(startMillis); 2550 excepValues.put(Events.DTSTART, startMillis + 3600*1000*2); 2551 long excepEventId3 = createAndVerifyException(account, eventId, excepValues, true); 2552 instances.moveToNext(); 2553 2554 // Cancel the 4th instance. 2555 startMillis = instances.getLong(0); 2556 excepValues = EventHelper.getNewExceptionValues(startMillis); 2557 excepValues.put(Events.STATUS, Events.STATUS_CANCELED); 2558 long excepEventId4 = createAndVerifyException(account, eventId, excepValues, true); 2559 instances.moveToNext(); 2560 2561 // TODO: try to modify a non-existent instance. 2562 2563 instances.close(); 2564 2565 // TODO: compare Reminders, Attendees, ExtendedProperties on one of the exception events 2566 2567 // Re-query the instances and figure out if they look right. 2568 instances = getInstances(timeZone, testStart, testEnd, projection, 2569 new long[] { calendarId }); 2570 if (DEBUG_RECURRENCE) { 2571 dumpInstances(instances, timeZone, "with DTSTART exceptions"); 2572 } 2573 assertEquals("exceptional recurrence instance count", 4, instances.getCount()); 2574 2575 long prevMinute = -1; 2576 while (instances.moveToNext()) { 2577 // expect the start times for each entry to be different from the previous entry 2578 long startMinute = instances.getLong(1); 2579 assertTrue("instance start times are different", startMinute != prevMinute); 2580 2581 prevMinute = startMinute; 2582 } 2583 instances.close(); 2584 2585 2586 // Delete all of our exceptions, and verify. 2587 int deleteCount = 0; 2588 deleteCount += deleteException(account, eventId, excepEventId2); 2589 deleteCount += deleteException(account, eventId, excepEventId3); 2590 deleteCount += deleteException(account, eventId, excepEventId4); 2591 assertEquals("events deleted", 3, deleteCount); 2592 2593 // Re-query the instances and figure out if they look right. 2594 instances = getInstances(timeZone, testStart, testEnd, projection, 2595 new long[] { calendarId }); 2596 if (DEBUG_RECURRENCE) { 2597 dumpInstances(instances, timeZone, "post exception deletion"); 2598 } 2599 assertEquals("post-exception deletion instance count", 5, instances.getCount()); 2600 2601 prevMinute = -1; 2602 while (instances.moveToNext()) { 2603 // expect the start times for each entry to be the same 2604 long startMinute = instances.getLong(1); 2605 if (prevMinute != -1) { 2606 assertEquals("instance start times are the same", startMinute, prevMinute); 2607 } 2608 prevMinute = startMinute; 2609 } 2610 instances.close(); 2611 2612 /* 2613 * Repeat the test, this time modifying DURATION. 2614 */ 2615 2616 instances = getInstances(timeZone, testStart, testEnd, projection, 2617 new long[] { calendarId }); 2618 if (DEBUG_RECURRENCE) { 2619 dumpInstances(instances, timeZone, "initial"); 2620 } 2621 2622 assertEquals("initial recurrence instance count", 5, instances.getCount()); 2623 2624 // Leave first instance alone. 2625 instances.moveToPosition(1); 2626 2627 // Advance the end time of the 2nd instance. 2628 startMillis = instances.getLong(0); 2629 excepValues = EventHelper.getNewExceptionValues(startMillis); 2630 excepValues.put(Events.DURATION, "P" + 3600*2 + "S"); 2631 excepEventId2 = createAndVerifyException(account, eventId, excepValues, true); 2632 instances.moveToNext(); 2633 2634 // Advance the end time of the 3rd instance, and change the self-attendee status. 2635 startMillis = instances.getLong(0); 2636 excepValues = EventHelper.getNewExceptionValues(startMillis); 2637 excepValues.put(Events.DURATION, "P" + 3600*3 + "S"); 2638 excepValues.put(Events.SELF_ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_DECLINED); 2639 excepEventId3 = createAndVerifyException(account, eventId, excepValues, true); 2640 instances.moveToNext(); 2641 2642 // Advance the start time of the 4th instance, which will also advance the end time. 2643 startMillis = instances.getLong(0); 2644 excepValues = EventHelper.getNewExceptionValues(startMillis); 2645 excepValues.put(Events.DTSTART, startMillis + 3600*1000); 2646 excepEventId4 = createAndVerifyException(account, eventId, excepValues, true); 2647 instances.moveToNext(); 2648 2649 instances.close(); 2650 2651 // TODO: make sure the selfAttendeeStatus change took 2652 2653 // Re-query the instances and figure out if they look right. 2654 instances = getInstances(timeZone, testStart, testEnd, projection, 2655 new long[] { calendarId }); 2656 if (DEBUG_RECURRENCE) { 2657 dumpInstances(instances, timeZone, "with DURATION exceptions"); 2658 } 2659 assertEquals("exceptional recurrence instance count", 5, instances.getCount()); 2660 2661 prevMinute = -1; 2662 while (instances.moveToNext()) { 2663 // expect the start times for each entry to be different from the previous entry 2664 long endMinute = instances.getLong(2); 2665 assertTrue("instance end times are different", endMinute != prevMinute); 2666 2667 prevMinute = endMinute; 2668 } 2669 instances.close(); 2670 2671 // delete the calendar 2672 removeAndVerifyCalendar(account, calendarId); 2673 } 2674 2675 /** 2676 * Tests creation of a simple recurrence exception when not pretending to be the sync 2677 * adapter. One significant consequence is that we don't set the _sync_id field in the 2678 * events, which affects how the provider correlates recurrences and exceptions. 2679 */ 2680 @MediumTest testNonAdapterRecurrenceExceptions()2681 public void testNonAdapterRecurrenceExceptions() { 2682 String account = "rena_account"; 2683 int seed = 0; 2684 2685 // Clean up just in case 2686 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2687 2688 // Create calendar 2689 long calendarId = createAndVerifyCalendar(account, seed++, null); 2690 2691 // Generate recurring event, with "asSyncAdapter" set to false. 2692 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 2693 calendarId, false, "1991-02-03T12:00:00", "PT1H", "FREQ=DAILY;WKST=SU;COUNT=10"); 2694 2695 // Select a period that gives us 3 instances. 2696 String timeZone = eventValues.getAsString(Events.EVENT_TIMEZONE); 2697 String testStart = "1991-02-03T00:00:00"; 2698 String testEnd = "1991-02-05T23:59:59"; 2699 String[] projection = { Instances.BEGIN, Instances.START_MINUTE }; 2700 2701 // Expand the bounds of the instances table so we expand future events as they are added. 2702 expandInstanceRange(account, calendarId, testStart, testEnd, timeZone); 2703 2704 // Create the event in the database. 2705 long eventId = createAndVerifyEvent(account, seed++, calendarId, false, eventValues); 2706 assertTrue(eventId >= 0); 2707 2708 // Add some attendees. 2709 addAttendees(account, eventId, seed); 2710 2711 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 2712 new long[] { calendarId }); 2713 if (DEBUG_RECURRENCE) { 2714 dumpInstances(instances, timeZone, "initial"); 2715 } 2716 assertEquals("initial recurrence instance count", 3, instances.getCount()); 2717 2718 /* 2719 * Alter the attendee status of the second event. This should cause the instances to 2720 * be updated, replacing the previous 2nd instance with the exception instance. If the 2721 * code is broken we'll see four instances (because the original instance didn't get 2722 * removed) or one instance (because the code correctly deleted all related events but 2723 * couldn't correlate the exception with its original recurrence). 2724 */ 2725 2726 // Leave first instance alone. 2727 instances.moveToPosition(1); 2728 2729 long startMillis; 2730 ContentValues excepValues; 2731 2732 // Advance the start time of the 2nd instance. 2733 startMillis = instances.getLong(0); 2734 excepValues = EventHelper.getNewExceptionValues(startMillis); 2735 excepValues.put(Events.SELF_ATTENDEE_STATUS, Attendees.ATTENDEE_STATUS_DECLINED); 2736 long excepEventId2 = createAndVerifyException(account, eventId, excepValues, false); 2737 instances.moveToNext(); 2738 2739 instances.close(); 2740 2741 // Re-query the instances and figure out if they look right. 2742 instances = getInstances(timeZone, testStart, testEnd, projection, 2743 new long[] { calendarId }); 2744 if (DEBUG_RECURRENCE) { 2745 dumpInstances(instances, timeZone, "with exceptions"); 2746 } 2747 2748 // TODO: this test currently fails due to limitations in the provider 2749 //assertEquals("exceptional recurrence instance count", 3, instances.getCount()); 2750 2751 instances.close(); 2752 2753 // delete the calendar 2754 removeAndVerifyCalendar(account, calendarId); 2755 } 2756 2757 /** 2758 * Tests insertion of event exceptions before and after a recurring event is created. 2759 * <p> 2760 * The server may send exceptions down before the event they refer to, so the provider 2761 * fills in the originalId of previously-existing exceptions when a recurring event is 2762 * inserted. Make sure that works. 2763 * <p> 2764 * The _sync_id column is only unique with a given calendar. We create events with 2765 * identical originalSyncId values in two different calendars to verify that the provider 2766 * doesn't update unrelated events. 2767 * <p> 2768 * We can't use the /exception URI, because that only works if the events are created 2769 * in order. 2770 */ 2771 @MediumTest testOutOfOrderRecurrenceExceptions()2772 public void testOutOfOrderRecurrenceExceptions() { 2773 String account1 = "roid1_account"; 2774 String account2 = "roid2_account"; 2775 String startWhen = "1987-08-09T12:00:00"; 2776 int seed = 0; 2777 2778 // Clean up just in case 2779 CalendarHelper.deleteCalendarByAccount(mContentResolver, account1); 2780 CalendarHelper.deleteCalendarByAccount(mContentResolver, account2); 2781 2782 // Create calendars 2783 long calendarId1 = createAndVerifyCalendar(account1, seed++, null); 2784 long calendarId2 = createAndVerifyCalendar(account2, seed++, null); 2785 2786 2787 // Generate base event. 2788 ContentValues recurEventValues = EventHelper.getNewRecurringEventValues(account1, seed++, 2789 calendarId1, true, startWhen, "PT1H", "FREQ=DAILY;WKST=SU;COUNT=10"); 2790 2791 // Select a period that gives us 3 instances. 2792 String timeZone = recurEventValues.getAsString(Events.EVENT_TIMEZONE); 2793 String testStart = "1987-08-09T00:00:00"; 2794 String testEnd = "1987-08-11T23:59:59"; 2795 String[] projection = { Instances.BEGIN, Instances.START_MINUTE, Instances.EVENT_ID }; 2796 2797 /* 2798 * We're interested in exploring what the instance expansion code does with the events 2799 * as they arrive. It won't do anything at event-creation time unless the instance 2800 * range already covers the interesting set of dates, so we need to create and remove 2801 * an instance in the same time frame beforehand. 2802 */ 2803 expandInstanceRange(account1, calendarId1, testStart, testEnd, timeZone); 2804 2805 /* 2806 * Instances table should be expanded. Do the test. 2807 */ 2808 2809 final String MAGIC_SYNC_ID = "MagicSyncId"; 2810 recurEventValues.put(Events._SYNC_ID, MAGIC_SYNC_ID); 2811 2812 // Generate exceptions from base, removing the generated _sync_id and setting the 2813 // base event's _sync_id as originalSyncId. 2814 ContentValues beforeExcepValues, afterExcepValues, unrelatedExcepValues; 2815 beforeExcepValues = new ContentValues(recurEventValues); 2816 afterExcepValues = new ContentValues(recurEventValues); 2817 unrelatedExcepValues = new ContentValues(recurEventValues); 2818 beforeExcepValues.remove(Events._SYNC_ID); 2819 afterExcepValues.remove(Events._SYNC_ID); 2820 unrelatedExcepValues.remove(Events._SYNC_ID); 2821 beforeExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2822 afterExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2823 unrelatedExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2824 2825 // Disassociate the "unrelated" exception by moving it to the other calendar. 2826 unrelatedExcepValues.put(Events.CALENDAR_ID, calendarId2); 2827 2828 // We shift the start time by half an hour, and use the same _sync_id. 2829 final long ONE_DAY_MILLIS = 24 * 60 * 60 * 1000; 2830 final long ONE_HOUR_MILLIS = 60 * 60 * 1000; 2831 final long HALF_HOUR_MILLIS = 30 * 60 * 1000; 2832 long dtstartMillis = recurEventValues.getAsLong(Events.DTSTART) + ONE_DAY_MILLIS; 2833 beforeExcepValues.put(Events.ORIGINAL_INSTANCE_TIME, dtstartMillis); 2834 beforeExcepValues.put(Events.DTSTART, dtstartMillis + HALF_HOUR_MILLIS); 2835 beforeExcepValues.put(Events.DTEND, dtstartMillis + ONE_HOUR_MILLIS); 2836 beforeExcepValues.remove(Events.DURATION); 2837 beforeExcepValues.remove(Events.RRULE); 2838 beforeExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2839 dtstartMillis += ONE_DAY_MILLIS; 2840 afterExcepValues.put(Events.ORIGINAL_INSTANCE_TIME, dtstartMillis); 2841 afterExcepValues.put(Events.DTSTART, dtstartMillis + HALF_HOUR_MILLIS); 2842 afterExcepValues.put(Events.DTEND, dtstartMillis + ONE_HOUR_MILLIS); 2843 afterExcepValues.remove(Events.DURATION); 2844 afterExcepValues.remove(Events.RRULE); 2845 afterExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2846 dtstartMillis += ONE_DAY_MILLIS; 2847 unrelatedExcepValues.put(Events.ORIGINAL_INSTANCE_TIME, dtstartMillis); 2848 unrelatedExcepValues.put(Events.DTSTART, dtstartMillis + HALF_HOUR_MILLIS); 2849 unrelatedExcepValues.put(Events.DTEND, dtstartMillis + ONE_HOUR_MILLIS); 2850 unrelatedExcepValues.remove(Events.DURATION); 2851 unrelatedExcepValues.remove(Events.RRULE); 2852 unrelatedExcepValues.put(Events.ORIGINAL_SYNC_ID, MAGIC_SYNC_ID); 2853 2854 2855 // Create "before" and "unrelated" exceptions. 2856 long beforeEventId = createAndVerifyEvent(account1, seed, calendarId1, true, 2857 beforeExcepValues); 2858 assertTrue(beforeEventId >= 0); 2859 long unrelatedEventId = createAndVerifyEvent(account2, seed, calendarId2, true, 2860 unrelatedExcepValues); 2861 assertTrue(unrelatedEventId >= 0); 2862 2863 // Create recurring event. 2864 long recurEventId = createAndVerifyEvent(account1, seed, calendarId1, true, 2865 recurEventValues); 2866 assertTrue(recurEventId >= 0); 2867 2868 // Create "after" exception. 2869 long afterEventId = createAndVerifyEvent(account1, seed, calendarId1, true, 2870 afterExcepValues); 2871 assertTrue(afterEventId >= 0); 2872 2873 if (Log.isLoggable(TAG, Log.DEBUG)) { 2874 Log.d(TAG, "before=" + beforeEventId + ", unrel=" + unrelatedEventId + 2875 ", recur=" + recurEventId + ", after=" + afterEventId); 2876 } 2877 2878 // Check to see how many instances we get. If the recurrence and the exception don't 2879 // get paired up correctly, we'll see too many instances. 2880 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 2881 new long[] { calendarId1, calendarId2 }); 2882 if (DEBUG_RECURRENCE) { 2883 dumpInstances(instances, timeZone, "with exception"); 2884 } 2885 2886 assertEquals("initial recurrence instance count", 3, instances.getCount()); 2887 2888 instances.close(); 2889 2890 2891 /* 2892 * Now we want to verify that: 2893 * - "before" and "after" have an originalId equal to our recurEventId 2894 * - "unrelated" has no originalId 2895 */ 2896 Cursor c = null; 2897 try { 2898 final String[] PROJECTION = new String[] { Events.ORIGINAL_ID }; 2899 Uri eventUri; 2900 Long originalId; 2901 2902 eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, beforeEventId); 2903 c = mContentResolver.query(eventUri, PROJECTION, null, null, null); 2904 assertEquals(1, c.getCount()); 2905 c.moveToNext(); 2906 originalId = c.getLong(0); 2907 assertNotNull(originalId); 2908 assertEquals(recurEventId, (long) originalId); 2909 c.close(); 2910 2911 eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, afterEventId); 2912 c = mContentResolver.query(eventUri, PROJECTION, null, null, null); 2913 assertEquals(1, c.getCount()); 2914 c.moveToNext(); 2915 originalId = c.getLong(0); 2916 assertNotNull(originalId); 2917 assertEquals(recurEventId, (long) originalId); 2918 c.close(); 2919 2920 eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, unrelatedEventId); 2921 c = mContentResolver.query(eventUri, PROJECTION, null, null, null); 2922 assertEquals(1, c.getCount()); 2923 c.moveToNext(); 2924 assertNull(c.getString(0)); 2925 c.close(); 2926 2927 c = null; 2928 } finally { 2929 if (c != null) { 2930 c.close(); 2931 } 2932 } 2933 2934 // delete the calendars 2935 removeAndVerifyCalendar(account1, calendarId1); 2936 removeAndVerifyCalendar(account2, calendarId2); 2937 } 2938 2939 /** 2940 * Tests exceptions that modify all future instances of a recurring event. 2941 */ 2942 @MediumTest testForwardRecurrenceExceptions()2943 public void testForwardRecurrenceExceptions() { 2944 String account = "refx_account"; 2945 int seed = 0; 2946 2947 // Clean up just in case 2948 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 2949 2950 // Create calendar 2951 long calendarId = createAndVerifyCalendar(account, seed++, null); 2952 2953 // Create recurring event 2954 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 2955 calendarId, true, "1999-01-01T06:00:00", "PT1H", "FREQ=WEEKLY;WKST=SU;COUNT=10"); 2956 long eventId = createAndVerifyEvent(account, seed++, calendarId, true, eventValues); 2957 2958 // Add some attendees and reminders. 2959 addAttendees(account, eventId, seed++); 2960 addReminders(account, eventId, seed++); 2961 2962 // Get some instances. 2963 String timeZone = eventValues.getAsString(Events.EVENT_TIMEZONE); 2964 String testStart = "1999-01-01T00:00:00"; 2965 String testEnd = "1999-01-29T23:59:59"; 2966 String[] projection = { Instances.BEGIN, Instances.START_MINUTE }; 2967 2968 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 2969 new long[] { calendarId }); 2970 if (DEBUG_RECURRENCE) { 2971 dumpInstances(instances, timeZone, "initial"); 2972 } 2973 2974 assertEquals("initial recurrence instance count", 5, instances.getCount()); 2975 2976 // Modify starting from 3rd instance. 2977 instances.moveToPosition(2); 2978 2979 long startMillis; 2980 ContentValues excepValues; 2981 2982 // Replace with a new recurrence rule. We move the start time an hour later, and cap 2983 // it at two instances. 2984 startMillis = instances.getLong(0); 2985 excepValues = EventHelper.getNewExceptionValues(startMillis); 2986 excepValues.put(Events.DTSTART, startMillis + 3600*1000); 2987 excepValues.put(Events.RRULE, "FREQ=WEEKLY;COUNT=2;WKST=SU"); 2988 long excepEventId = createAndVerifyException(account, eventId, excepValues, true); 2989 instances.close(); 2990 2991 2992 // Check to see if it took. 2993 instances = getInstances(timeZone, testStart, testEnd, projection, 2994 new long[] { calendarId }); 2995 if (DEBUG_RECURRENCE) { 2996 dumpInstances(instances, timeZone, "with new rule"); 2997 } 2998 2999 assertEquals("count with exception", 4, instances.getCount()); 3000 3001 long prevMinute = -1; 3002 for (int i = 0; i < 4; i++) { 3003 long startMinute; 3004 instances.moveToNext(); 3005 switch (i) { 3006 case 0: 3007 startMinute = instances.getLong(1); 3008 break; 3009 case 1: 3010 case 3: 3011 startMinute = instances.getLong(1); 3012 assertEquals("first/last pairs match", prevMinute, startMinute); 3013 break; 3014 case 2: 3015 startMinute = instances.getLong(1); 3016 assertFalse("first two != last two", prevMinute == startMinute); 3017 break; 3018 default: 3019 fail(); 3020 startMinute = -1; // make compiler happy 3021 break; 3022 } 3023 3024 prevMinute = startMinute; 3025 } 3026 instances.close(); 3027 3028 // delete the calendar 3029 removeAndVerifyCalendar(account, calendarId); 3030 } 3031 3032 /** 3033 * Tests exceptions that modify all instances of a recurring event. This is not really an 3034 * exception, since it won't create a new event, but supporting it allows us to use the 3035 * exception URI without having to determine whether the "start from here" instance is the 3036 * very first instance. 3037 */ 3038 @MediumTest testFullRecurrenceUpdate()3039 public void testFullRecurrenceUpdate() { 3040 String account = "ref_account"; 3041 int seed = 0; 3042 3043 // Clean up just in case 3044 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3045 3046 // Create calendar 3047 long calendarId = createAndVerifyCalendar(account, seed++, null); 3048 3049 // Create recurring event 3050 String rrule = "FREQ=DAILY;WKST=MO;COUNT=100"; 3051 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 3052 calendarId, true, "1997-08-29T02:14:00", "PT1H", rrule); 3053 long eventId = createAndVerifyEvent(account, seed++, calendarId, true, eventValues); 3054 //Log.i(TAG, "+++ eventId is " + eventId); 3055 3056 // Get some instances. 3057 String timeZone = eventValues.getAsString(Events.EVENT_TIMEZONE); 3058 String testStart = "1997-08-01T00:00:00"; 3059 String testEnd = "1997-08-31T23:59:59"; 3060 String[] projection = { Instances.BEGIN, Instances.EVENT_LOCATION }; 3061 String newLocation = "NEW!"; 3062 3063 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 3064 new long[] { calendarId }); 3065 if (DEBUG_RECURRENCE) { 3066 dumpInstances(instances, timeZone, "initial"); 3067 } 3068 3069 assertEquals("initial recurrence instance count", 3, instances.getCount()); 3070 3071 instances.moveToFirst(); 3072 long startMillis = instances.getLong(0); 3073 ContentValues excepValues = EventHelper.getNewExceptionValues(startMillis); 3074 excepValues.put(Events.RRULE, rrule); // identifies this as an "all future events" excep 3075 excepValues.put(Events.EVENT_LOCATION, newLocation); 3076 long excepEventId = createAndVerifyException(account, eventId, excepValues, true); 3077 instances.close(); 3078 3079 // Check results. 3080 assertEquals("full update does not create new ID", eventId, excepEventId); 3081 3082 instances = getInstances(timeZone, testStart, testEnd, projection, 3083 new long[] { calendarId }); 3084 assertEquals("post-update instance count", 3, instances.getCount()); 3085 while (instances.moveToNext()) { 3086 assertEquals("new location", newLocation, instances.getString(1)); 3087 } 3088 instances.close(); 3089 3090 // delete the calendar 3091 removeAndVerifyCalendar(account, calendarId); 3092 } 3093 3094 @MediumTest testMultiRuleRecurrence()3095 public void testMultiRuleRecurrence() { 3096 String account = "multirule_account"; 3097 int seed = 0; 3098 3099 // Clean up just in case 3100 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3101 3102 // Create calendar 3103 long calendarId = createAndVerifyCalendar(account, seed++, null); 3104 3105 // Create recurring event 3106 String rrule = "FREQ=DAILY;WKST=MO;COUNT=5\nFREQ=WEEKLY;WKST=SU;COUNT=5"; 3107 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 3108 calendarId, true, "1997-08-29T02:14:00", "PT1H", rrule); 3109 long eventId = createAndVerifyEvent(account, seed++, calendarId, true, eventValues); 3110 3111 // TODO: once multi-rule RRULEs are fully supported, verify that they work 3112 3113 // delete the calendar 3114 removeAndVerifyCalendar(account, calendarId); 3115 } 3116 3117 /** 3118 * Issue bad requests and expect them to get rejected. 3119 */ 3120 @MediumTest testBadRequests()3121 public void testBadRequests() { 3122 String account = "neg_account"; 3123 int seed = 0; 3124 3125 // Clean up just in case 3126 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3127 3128 // Create calendar 3129 long calendarId = createAndVerifyCalendar(account, seed++, null); 3130 3131 // Create recurring event 3132 String rrule = "FREQ=OFTEN;WKST=MO"; 3133 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed++, 3134 calendarId, true, "1997-08-29T02:14:00", "PT1H", rrule); 3135 try { 3136 createAndVerifyEvent(account, seed++, calendarId, true, eventValues); 3137 fail("Bad recurrence rule should have been rejected"); 3138 } catch (IllegalArgumentException iae) { 3139 // good 3140 } 3141 3142 // delete the calendar 3143 removeAndVerifyCalendar(account, calendarId); 3144 } 3145 3146 /** 3147 * Tests correct behavior of Calendars.isPrimary column 3148 */ 3149 @MediumTest testCalendarIsPrimary()3150 public void testCalendarIsPrimary() { 3151 String account = "ec_account"; 3152 int seed = 0; 3153 3154 // Clean up just in case 3155 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3156 3157 int isPrimary; 3158 Cursor cursor; 3159 ContentValues values = new ContentValues(); 3160 3161 final long calendarId = createAndVerifyCalendar(account, seed++, null); 3162 final Uri uri = ContentUris.withAppendedId(Calendars.CONTENT_URI, calendarId); 3163 3164 // verify when ownerAccount != account_name && isPrimary IS NULL 3165 cursor = mContentResolver.query(uri, new String[]{Calendars.IS_PRIMARY}, null, null, null); 3166 cursor.moveToFirst(); 3167 isPrimary = cursor.getInt(0); 3168 cursor.close(); 3169 assertEquals("isPrimary should be 0 if ownerAccount != account_name", 0, isPrimary); 3170 3171 // verify when ownerAccount == account_name && isPrimary IS NULL 3172 values.clear(); 3173 values.put(Calendars.OWNER_ACCOUNT, account); 3174 mContentResolver.update(asSyncAdapter(uri, account, CTS_TEST_TYPE), values, null, null); 3175 cursor = mContentResolver.query(uri, new String[]{Calendars.IS_PRIMARY}, null, null, null); 3176 cursor.moveToFirst(); 3177 isPrimary = cursor.getInt(0); 3178 cursor.close(); 3179 assertEquals("isPrimary should be 1 if ownerAccount == account_name", 1, isPrimary); 3180 3181 // verify isPrimary IS NOT NULL 3182 values.clear(); 3183 values.put(Calendars.IS_PRIMARY, SOME_ARBITRARY_INT); 3184 mContentResolver.update(uri, values, null, null); 3185 cursor = mContentResolver.query(uri, new String[]{Calendars.IS_PRIMARY}, null, null, null); 3186 cursor.moveToFirst(); 3187 isPrimary = cursor.getInt(0); 3188 cursor.close(); 3189 assertEquals("isPrimary should be the value it was set to", SOME_ARBITRARY_INT, isPrimary); 3190 3191 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3192 } 3193 3194 /** 3195 * Tests correct behavior of Events.isOrganizer column 3196 */ 3197 @MediumTest testEventsIsOrganizer()3198 public void testEventsIsOrganizer() { 3199 String account = "ec_account"; 3200 int seed = 0; 3201 3202 // Clean up just in case 3203 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3204 3205 int isOrganizer; 3206 Cursor cursor; 3207 ContentValues values = new ContentValues(); 3208 3209 final long calendarId = createAndVerifyCalendar(account, seed++, null); 3210 final long eventId = createAndVerifyEvent(account, seed, calendarId, true, null); 3211 final Uri uri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 3212 3213 // verify when ownerAccount != organizer && isOrganizer IS NULL 3214 cursor = mContentResolver.query(uri, new String[]{Events.IS_ORGANIZER}, null, null, null); 3215 cursor.moveToFirst(); 3216 isOrganizer = cursor.getInt(0); 3217 cursor.close(); 3218 assertEquals("isOrganizer should be 0 if ownerAccount != organizer", 0, isOrganizer); 3219 3220 // verify when ownerAccount == account_name && isOrganizer IS NULL 3221 values.clear(); 3222 values.put(Events.ORGANIZER, CalendarHelper.generateCalendarOwnerEmail(account)); 3223 mContentResolver.update(asSyncAdapter(uri, account, CTS_TEST_TYPE), values, null, null); 3224 cursor = mContentResolver.query(uri, new String[]{Events.IS_ORGANIZER}, null, null, null); 3225 cursor.moveToFirst(); 3226 isOrganizer = cursor.getInt(0); 3227 cursor.close(); 3228 assertEquals("isOrganizer should be 1 if ownerAccount == organizer", 1, isOrganizer); 3229 3230 // verify isOrganizer IS NOT NULL 3231 values.clear(); 3232 values.put(Events.IS_ORGANIZER, SOME_ARBITRARY_INT); 3233 mContentResolver.update(uri, values, null, null); 3234 cursor = mContentResolver.query(uri, new String[]{Events.IS_ORGANIZER}, null, null, null); 3235 cursor.moveToFirst(); 3236 isOrganizer = cursor.getInt(0); 3237 cursor.close(); 3238 assertEquals( 3239 "isPrimary should be the value it was set to", SOME_ARBITRARY_INT, isOrganizer); 3240 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3241 } 3242 3243 /** 3244 * Tests correct behavior of Events.uid2445 column 3245 */ 3246 @MediumTest testEventsUid2445()3247 public void testEventsUid2445() { 3248 String account = "ec_account"; 3249 int seed = 0; 3250 3251 // Clean up just in case 3252 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3253 3254 final String uid = "uid_123"; 3255 Cursor cursor; 3256 ContentValues values = new ContentValues(); 3257 final long calendarId = createAndVerifyCalendar(account, seed++, null); 3258 final long eventId = createAndVerifyEvent(account, seed, calendarId, true, null); 3259 final Uri uri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 3260 3261 // Verify default is null 3262 cursor = mContentResolver.query(uri, new String[] {Events.UID_2445}, null, null, null); 3263 cursor.moveToFirst(); 3264 assertTrue(cursor.isNull(0)); 3265 cursor.close(); 3266 3267 // Write column value and read back 3268 values.clear(); 3269 values.put(Events.UID_2445, uid); 3270 mContentResolver.update(asSyncAdapter(uri, account, CTS_TEST_TYPE), values, null, null); 3271 cursor = mContentResolver.query(uri, new String[] {Events.UID_2445}, null, null, null); 3272 cursor.moveToFirst(); 3273 assertFalse(cursor.isNull(0)); 3274 assertEquals("Column uid_2445 has unexpected value.", uid, cursor.getString(0)); 3275 3276 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3277 } 3278 3279 @MediumTest testMutatorSetCorrectly()3280 public void testMutatorSetCorrectly() { 3281 String account = "ec_account"; 3282 String packageName = "android.provider.cts.calendar"; 3283 int seed = 0; 3284 3285 // Clean up just in case 3286 CalendarHelper.deleteCalendarByAccount(mContentResolver, account); 3287 3288 String mutator; 3289 Cursor cursor; 3290 ContentValues values = new ContentValues(); 3291 final long calendarId = createAndVerifyCalendar(account, seed++, null); 3292 3293 // Verify mutator is set to the package, via: 3294 // Create: 3295 final long eventId = createAndVerifyEvent(account, seed, calendarId, false, null); 3296 final Uri uri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 3297 cursor = mContentResolver.query(uri, new String[] {Events.MUTATORS}, null, null, null); 3298 cursor.moveToFirst(); 3299 mutator = cursor.getString(0); 3300 cursor.close(); 3301 assertEquals(packageName, mutator); 3302 3303 // Edit: 3304 // First clear the mutator column 3305 values.clear(); 3306 values.putNull(Events.MUTATORS); 3307 mContentResolver.update(asSyncAdapter(uri, account, CTS_TEST_TYPE), values, null, null); 3308 cursor = mContentResolver.query(uri, new String[] {Events.MUTATORS}, null, null, null); 3309 cursor.moveToFirst(); 3310 mutator = cursor.getString(0); 3311 cursor.close(); 3312 assertNull(mutator); 3313 // Now edit the event and verify the mutator column 3314 values.clear(); 3315 values.put(Events.TITLE, "New title"); 3316 mContentResolver.update(uri, values, null, null); 3317 cursor = mContentResolver.query(uri, new String[] {Events.MUTATORS}, null, null, null); 3318 cursor.moveToFirst(); 3319 mutator = cursor.getString(0); 3320 cursor.close(); 3321 assertEquals(packageName, mutator); 3322 3323 // Clean up the event 3324 assertEquals(1, EventHelper.deleteEventAsSyncAdapter(mContentResolver, uri, account)); 3325 3326 // Delete: 3327 // First create as sync adapter 3328 final long eventId2 = createAndVerifyEvent(account, seed, calendarId, true, null); 3329 final Uri uri2 = ContentUris.withAppendedId(Events.CONTENT_URI, eventId2); 3330 // Now delete the event and verify 3331 values.clear(); 3332 values.put(Events.MUTATORS, packageName); 3333 removeAndVerifyEvent(uri2, values, account); 3334 3335 3336 // delete the calendar 3337 removeAndVerifyCalendar(account, calendarId); 3338 } 3339 3340 /** 3341 * Acquires the set of instances that appear between the specified start and end points. 3342 * 3343 * @param timeZone Time zone to use when parsing startWhen and endWhen 3344 * @param startWhen Start date/time, in RFC 3339 format 3345 * @param endWhen End date/time, in RFC 3339 format 3346 * @param projection Array of desired column names 3347 * @return Cursor with instances (caller should close when done) 3348 */ getInstances(String timeZone, String startWhen, String endWhen, String[] projection, long[] calendarIds)3349 private Cursor getInstances(String timeZone, String startWhen, String endWhen, 3350 String[] projection, long[] calendarIds) { 3351 Time startTime = new Time(timeZone); 3352 startTime.parse3339(startWhen); 3353 long startMillis = startTime.toMillis(false); 3354 3355 Time endTime = new Time(timeZone); 3356 endTime.parse3339(endWhen); 3357 long endMillis = endTime.toMillis(false); 3358 3359 // We want a list of instances that occur between the specified dates. Use the 3360 // "instances/when" URI. 3361 Uri uri = Uri.withAppendedPath(CalendarContract.Instances.CONTENT_URI, 3362 startMillis + "/" + endMillis); 3363 3364 String where = null; 3365 for (int i = 0; i < calendarIds.length; i++) { 3366 if (i > 0) { 3367 where += " OR "; 3368 } else { 3369 where = ""; 3370 } 3371 where += (Instances.CALENDAR_ID + "=" + calendarIds[i]); 3372 } 3373 Cursor instances = mContentResolver.query(uri, projection, where, null, 3374 projection[0] + " ASC"); 3375 3376 return instances; 3377 } 3378 3379 /** 3380 * Acquires the set of instances that appear between the specified start and end points 3381 * that match the search terms. 3382 * 3383 * @param timeZone Time zone to use when parsing startWhen and endWhen 3384 * @param startWhen Start date/time, in RFC 3339 format 3385 * @param endWhen End date/time, in RFC 3339 format 3386 * @param search A collection of tokens to search for. The columns searched are 3387 * hard-coded in the provider (currently title, description, location, attendee 3388 * name, attendee email). 3389 * @param searchByDay If set, adjust start/end to calendar day boundaries. 3390 * @param projection Array of desired column names 3391 * @return Cursor with instances (caller should close when done) 3392 */ getInstancesSearch(String timeZone, String startWhen, String endWhen, String search, boolean searchByDay, String[] projection, long[] calendarIds)3393 private Cursor getInstancesSearch(String timeZone, String startWhen, String endWhen, 3394 String search, boolean searchByDay, String[] projection, long[] calendarIds) { 3395 Time startTime = new Time(timeZone); 3396 startTime.parse3339(startWhen); 3397 long startMillis = startTime.toMillis(false); 3398 3399 Time endTime = new Time(timeZone); 3400 endTime.parse3339(endWhen); 3401 long endMillis = endTime.toMillis(false); 3402 3403 Uri uri; 3404 if (searchByDay) { 3405 // start/end are Julian day numbers rather than time in milliseconds 3406 int julianStart = Time.getJulianDay(startMillis, startTime.gmtoff); 3407 int julianEnd = Time.getJulianDay(endMillis, endTime.gmtoff); 3408 uri = Uri.withAppendedPath(CalendarContract.Instances.CONTENT_SEARCH_BY_DAY_URI, 3409 julianStart + "/" + julianEnd + "/" + search); 3410 } else { 3411 uri = Uri.withAppendedPath(CalendarContract.Instances.CONTENT_SEARCH_URI, 3412 startMillis + "/" + endMillis + "/" + search); 3413 } 3414 3415 String where = null; 3416 for (int i = 0; i < calendarIds.length; i++) { 3417 if (i > 0) { 3418 where += " OR "; 3419 } else { 3420 where = ""; 3421 } 3422 where += (Instances.CALENDAR_ID + "=" + calendarIds[i]); 3423 } 3424 // We want a list of instances that occur between the specified dates and that match 3425 // the search terms. 3426 3427 Cursor instances = mContentResolver.query(uri, projection, where, null, 3428 projection[0] + " ASC"); 3429 3430 return instances; 3431 } 3432 3433 /** debug -- dump instances cursor */ dumpInstances(Cursor instances, String timeZone, String msg)3434 private static void dumpInstances(Cursor instances, String timeZone, String msg) { 3435 Log.d(TAG, "Instances (" + msg + ")"); 3436 3437 int posn = instances.getPosition(); 3438 instances.moveToPosition(-1); 3439 3440 //Log.d(TAG, "+++ instances has " + instances.getCount() + " rows, " + 3441 // instances.getColumnCount() + " columns"); 3442 while (instances.moveToNext()) { 3443 long beginMil = instances.getLong(0); 3444 Time beginT = new Time(timeZone); 3445 beginT.set(beginMil); 3446 String logMsg = "--> begin=" + beginT.format3339(false) + " (" + beginMil + ")"; 3447 for (int i = 2; i < instances.getColumnCount(); i++) { 3448 logMsg += " [" + instances.getString(i) + "]"; 3449 } 3450 Log.d(TAG, logMsg); 3451 } 3452 instances.moveToPosition(posn); 3453 } 3454 3455 3456 /** 3457 * Counts the number of instances that appear between the specified start and end times. 3458 */ getInstanceCount(String timeZone, String startWhen, String endWhen, long[] calendarIds)3459 private int getInstanceCount(String timeZone, String startWhen, String endWhen, 3460 long[] calendarIds) { 3461 Cursor instances = getInstances(timeZone, startWhen, endWhen, 3462 new String[] { Instances._ID }, calendarIds); 3463 int count = instances.getCount(); 3464 instances.close(); 3465 return count; 3466 } 3467 3468 /** 3469 * Deletes an event as app and sync adapter which removes it from the db and 3470 * verifies after each. 3471 * 3472 * @param eventUri The uri for the event to delete 3473 * @param accountName TODO 3474 */ removeAndVerifyEvent(Uri eventUri, ContentValues eventValues, String accountName)3475 private void removeAndVerifyEvent(Uri eventUri, ContentValues eventValues, String accountName) { 3476 // Delete event 3477 EventHelper.deleteEvent(mContentResolver, eventUri, eventValues); 3478 // Verify 3479 verifyEvent(eventValues, ContentUris.parseId(eventUri)); 3480 // Delete as sync adapter 3481 assertEquals(1, 3482 EventHelper.deleteEventAsSyncAdapter(mContentResolver, eventUri, accountName)); 3483 // Verify 3484 Cursor c = EventHelper.getEventByUri(mContentResolver, eventUri); 3485 assertEquals(0, c.getCount()); 3486 c.close(); 3487 } 3488 3489 /** 3490 * Creates an event on the given calendar and verifies it. 3491 * 3492 * @param account 3493 * @param seed 3494 * @param calendarId 3495 * @param asSyncAdapter 3496 * @param values optional pre created set of values; will have several new entries added 3497 * @return the _id for the new event 3498 */ createAndVerifyEvent(String account, int seed, long calendarId, boolean asSyncAdapter, ContentValues values)3499 private long createAndVerifyEvent(String account, int seed, long calendarId, 3500 boolean asSyncAdapter, ContentValues values) { 3501 // Create an event 3502 if (values == null) { 3503 values = EventHelper.getNewEventValues(account, seed, calendarId, asSyncAdapter); 3504 } 3505 Uri insertUri = Events.CONTENT_URI; 3506 if (asSyncAdapter) { 3507 insertUri = asSyncAdapter(insertUri, account, CTS_TEST_TYPE); 3508 } 3509 Uri uri = mContentResolver.insert(insertUri, values); 3510 assertNotNull(uri); 3511 3512 // Verify 3513 EventHelper.addDefaultReadOnlyValues(values, account, asSyncAdapter); 3514 long eventId = ContentUris.parseId(uri); 3515 assertTrue(eventId >= 0); 3516 3517 verifyEvent(values, eventId); 3518 return eventId; 3519 } 3520 3521 /** 3522 * Updates an event, and verifies that the updates took. 3523 */ updateAndVerifyEvent(String account, long calendarId, long eventId, boolean asSyncAdapter, ContentValues updateValues)3524 private void updateAndVerifyEvent(String account, long calendarId, long eventId, 3525 boolean asSyncAdapter, ContentValues updateValues) { 3526 Uri uri = Uri.withAppendedPath(Events.CONTENT_URI, String.valueOf(eventId)); 3527 if (asSyncAdapter) { 3528 uri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 3529 } 3530 int count = mContentResolver.update(uri, updateValues, null, null); 3531 3532 // Verify 3533 assertEquals(1, count); 3534 verifyEvent(updateValues, eventId); 3535 } 3536 3537 /** 3538 * Creates an exception to a recurring event, and verifies it. 3539 * @param account The account to use. 3540 * @param originalEventId The ID of the original event. 3541 * @param values Values for the exception; must include originalInstanceTime. 3542 * @return The _id for the new event. 3543 */ createAndVerifyException(String account, long originalEventId, ContentValues values, boolean asSyncAdapter)3544 private long createAndVerifyException(String account, long originalEventId, 3545 ContentValues values, boolean asSyncAdapter) { 3546 // Create the exception 3547 Uri uri = Uri.withAppendedPath(Events.CONTENT_EXCEPTION_URI, 3548 String.valueOf(originalEventId)); 3549 if (asSyncAdapter) { 3550 uri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 3551 } 3552 Uri resultUri = mContentResolver.insert(uri, values); 3553 assertNotNull(resultUri); 3554 long eventId = ContentUris.parseId(resultUri); 3555 assertTrue(eventId >= 0); 3556 return eventId; 3557 } 3558 3559 /** 3560 * Deletes an exception to a recurring event. 3561 * @param account The account to use. 3562 * @param eventId The ID of the original recurring event. 3563 * @param excepId The ID of the exception event. 3564 * @return The number of rows deleted. 3565 */ deleteException(String account, long eventId, long excepId)3566 private int deleteException(String account, long eventId, long excepId) { 3567 Uri uri = Uri.withAppendedPath(Events.CONTENT_EXCEPTION_URI, 3568 eventId + "/" + excepId); 3569 uri = asSyncAdapter(uri, account, CTS_TEST_TYPE); 3570 return mContentResolver.delete(uri, null, null); 3571 } 3572 3573 /** 3574 * Add some sample attendees to an event. 3575 */ addAttendees(String account, long eventId, int seed)3576 private void addAttendees(String account, long eventId, int seed) { 3577 assertTrue(eventId >= 0); 3578 AttendeeHelper.addAttendee(mContentResolver, eventId, 3579 "Attender" + seed, 3580 CalendarHelper.generateCalendarOwnerEmail(account), 3581 Attendees.ATTENDEE_STATUS_ACCEPTED, 3582 Attendees.RELATIONSHIP_ORGANIZER, 3583 Attendees.TYPE_NONE); 3584 seed++; 3585 3586 AttendeeHelper.addAttendee(mContentResolver, eventId, 3587 "Attender" + seed, 3588 "attender" + seed + "@example.com", 3589 Attendees.ATTENDEE_STATUS_TENTATIVE, 3590 Attendees.RELATIONSHIP_NONE, 3591 Attendees.TYPE_NONE); 3592 } 3593 3594 /** 3595 * Add some sample reminders to an event. 3596 */ addReminders(String account, long eventId, int seed)3597 private void addReminders(String account, long eventId, int seed) { 3598 ReminderHelper.addReminder(mContentResolver, eventId, seed * 5, Reminders.METHOD_ALERT); 3599 } 3600 3601 /** 3602 * Creates and removes an event that covers a specific range of dates. Call this to 3603 * cause the provider to expand the CalendarMetaData min/max values to include the range. 3604 * Useful when you want to see the provider expand the instances as the events are added. 3605 */ expandInstanceRange(String account, long calendarId, String testStart, String testEnd, String timeZone)3606 private void expandInstanceRange(String account, long calendarId, String testStart, 3607 String testEnd, String timeZone) { 3608 int seed = 0; 3609 3610 // TODO: this should use an UNTIL rule based on testEnd, not a COUNT 3611 ContentValues eventValues = EventHelper.getNewRecurringEventValues(account, seed, 3612 calendarId, true, testStart, "PT1H", "FREQ=DAILY;WKST=SU;COUNT=100"); 3613 3614 /* 3615 * Some of the helper functions modify "eventValues", so we want to make sure we're 3616 * passing a copy of anything we want to re-use. 3617 */ 3618 long eventId = createAndVerifyEvent(account, seed, calendarId, true, 3619 new ContentValues(eventValues)); 3620 assertTrue(eventId >= 0); 3621 3622 String[] projection = { Instances.BEGIN, Instances.START_MINUTE }; 3623 Cursor instances = getInstances(timeZone, testStart, testEnd, projection, 3624 new long[] { calendarId }); 3625 if (DEBUG_RECURRENCE) { 3626 dumpInstances(instances, timeZone, "prep-create"); 3627 } 3628 assertEquals("initial recurrence instance count", 3, instances.getCount()); 3629 instances.close(); 3630 3631 Uri eventUri = ContentUris.withAppendedId(Events.CONTENT_URI, eventId); 3632 removeAndVerifyEvent(eventUri, new ContentValues(eventValues), account); 3633 3634 instances = getInstances(timeZone, testStart, testEnd, projection, 3635 new long[] { calendarId }); 3636 if (DEBUG_RECURRENCE) { 3637 dumpInstances(instances, timeZone, "prep-clear"); 3638 } 3639 assertEquals("initial recurrence instance count", 0, instances.getCount()); 3640 instances.close(); 3641 3642 } 3643 3644 /** 3645 * Inserts a new calendar with the given account and seed and verifies it. 3646 * 3647 * @param account The account to add the calendar to 3648 * @param seed A number to use to generate the values 3649 * @return the created calendar's id 3650 */ createAndVerifyCalendar(String account, int seed, ContentValues values)3651 private long createAndVerifyCalendar(String account, int seed, ContentValues values) { 3652 // Create a calendar 3653 if (values == null) { 3654 values = CalendarHelper.getNewCalendarValues(account, seed); 3655 } 3656 Uri syncUri = asSyncAdapter(Calendars.CONTENT_URI, account, CTS_TEST_TYPE); 3657 Uri uri = mContentResolver.insert(syncUri, values); 3658 long calendarId = ContentUris.parseId(uri); 3659 assertTrue(calendarId >= 0); 3660 3661 verifyCalendar(account, values, calendarId, 1); 3662 return calendarId; 3663 } 3664 3665 /** 3666 * Deletes a given calendar and verifies no calendars remain on that 3667 * account. 3668 * 3669 * @param account 3670 * @param id 3671 */ removeAndVerifyCalendar(String account, long id)3672 private void removeAndVerifyCalendar(String account, long id) { 3673 // TODO Add code to delete as app and sync adapter and test both 3674 3675 // Delete 3676 assertEquals(1, CalendarHelper.deleteCalendarById(mContentResolver, id)); 3677 3678 // Verify 3679 Cursor c = CalendarHelper.getCalendarsByAccount(mContentResolver, account); 3680 assertEquals(0, c.getCount()); 3681 c.close(); 3682 } 3683 3684 /** 3685 * Check all the fields of a calendar contained in values + id. 3686 * 3687 * @param account the account of the calendar 3688 * @param values the values to check against the db 3689 * @param id the _id of the calendar 3690 * @param expectedCount the number of calendars expected on this account 3691 */ verifyCalendar(String account, ContentValues values, long id, int expectedCount)3692 private void verifyCalendar(String account, ContentValues values, long id, int expectedCount) { 3693 // Verify 3694 Cursor c = CalendarHelper.getCalendarsByAccount(mContentResolver, account); 3695 assertEquals(expectedCount, c.getCount()); 3696 assertTrue(c.moveToFirst()); 3697 while (c.getLong(0) != id) { 3698 assertTrue(c.moveToNext()); 3699 } 3700 for (String key : values.keySet()) { 3701 int index = c.getColumnIndex(key); 3702 assertTrue("Key " + key + " not in projection", index >= 0); 3703 assertEquals(key, values.getAsString(key), c.getString(index)); 3704 } 3705 c.close(); 3706 } 3707 3708 /** 3709 * Creates a new _sync_state entry and verifies the contents. 3710 */ createAndVerifySyncState(String account, ContentValues values)3711 private long createAndVerifySyncState(String account, ContentValues values) { 3712 assertNotNull(values); 3713 Uri syncUri = asSyncAdapter(SyncState.CONTENT_URI, account, CTS_TEST_TYPE); 3714 Uri uri = mContentResolver.insert(syncUri, values); 3715 long syncStateId = ContentUris.parseId(uri); 3716 assertTrue(syncStateId >= 0); 3717 3718 verifySyncState(account, values, syncStateId); 3719 return syncStateId; 3720 3721 } 3722 3723 /** 3724 * Removes the _sync_state entry with the specified id, then verifies that it's gone. 3725 */ removeAndVerifySyncState(String account)3726 private void removeAndVerifySyncState(String account) { 3727 assertEquals(1, SyncStateHelper.deleteSyncStateByAccount(mContentResolver, account, true)); 3728 3729 // Verify 3730 Cursor c = SyncStateHelper.getSyncStateByAccount(mContentResolver, account); 3731 try { 3732 assertEquals(0, c.getCount()); 3733 } finally { 3734 if (c != null) { 3735 c.close(); 3736 } 3737 } 3738 } 3739 3740 /** 3741 * Check all the fields of a _sync_state entry contained in values + id. This assumes 3742 * a single _sync_state has been created on the given account. 3743 */ verifySyncState(String account, ContentValues values, long id)3744 private void verifySyncState(String account, ContentValues values, long id) { 3745 // Verify 3746 Cursor c = SyncStateHelper.getSyncStateByAccount(mContentResolver, account); 3747 try { 3748 assertEquals(1, c.getCount()); 3749 assertTrue(c.moveToFirst()); 3750 assertEquals(id, c.getLong(0)); 3751 for (String key : values.keySet()) { 3752 int index = c.getColumnIndex(key); 3753 if (key.equals(SyncState.DATA)) { 3754 // TODO: can't compare as string, so compare as byte[] 3755 } else { 3756 assertEquals(key, values.getAsString(key), c.getString(index)); 3757 } 3758 } 3759 } finally { 3760 if (c != null) { 3761 c.close(); 3762 } 3763 } 3764 } 3765 } 3766