1 /* 2 * Copyright (c) 2015, Motorola Mobility LLC 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions are met: 7 * - Redistributions of source code must retain the above copyright 8 * notice, this list of conditions and the following disclaimer. 9 * - Redistributions in binary form must reproduce the above copyright 10 * notice, this list of conditions and the following disclaimer in the 11 * documentation and/or other materials provided with the distribution. 12 * - Neither the name of Motorola Mobility nor the 13 * names of its contributors may be used to endorse or promote products 14 * derived from this software without specific prior written permission. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, 18 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 19 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MOTOROLA MOBILITY LLC BE LIABLE 20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH 26 * DAMAGE. 27 */ 28 29 package com.android.service.ims.presence; 30 31 import android.content.Context; 32 import android.content.ContentResolver; 33 import android.content.ContentUris; 34 import android.content.ContentValues; 35 import android.content.Context; 36 import android.database.Cursor; 37 import android.database.CursorWrapper; 38 import android.database.Cursor; 39 import android.database.DatabaseUtils; 40 import android.net.Uri; 41 import android.text.format.TimeMigrationUtils; 42 import android.text.TextUtils; 43 44 import com.android.ims.internal.EABContract; 45 import com.android.ims.internal.ContactNumberUtils; 46 import com.android.ims.RcsPresenceInfo; 47 import com.android.ims.internal.Logger; 48 49 import java.util.ArrayList; 50 import java.util.List; 51 52 public class EABContactManager { 53 private Logger logger = Logger.getLogger(this.getClass().getName()); 54 55 /** 56 * An identifier for a particular EAB contact number, unique across the system. 57 * Clients use this ID to make subsequent calls related to the contact. 58 */ 59 public final static String COLUMN_ID = Contacts.Impl._ID; 60 61 /** 62 * Timestamp when the presence was last updated, in {@link System#currentTimeMillis 63 * System.currentTimeMillis()} (wall clock time in UTC). 64 */ 65 public final static String COLUMN_LAST_UPDATED_TIMESTAMP = 66 Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP; 67 68 /** 69 * columns to request from EABProvider. 70 * @hide 71 */ 72 public static final String[] CONTACT_COLUMNS = new String[] { 73 Contacts.Impl._ID, 74 Contacts.Impl.CONTACT_NUMBER, 75 Contacts.Impl.CONTACT_NAME, 76 Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP, 77 Contacts.Impl.VOLTE_CALL_SERVICE_CONTACT_ADDRESS, 78 Contacts.Impl.VOLTE_CALL_CAPABILITY, 79 Contacts.Impl.VOLTE_CALL_CAPABILITY_TIMESTAMP, 80 Contacts.Impl.VOLTE_CALL_AVAILABILITY, 81 Contacts.Impl.VOLTE_CALL_AVAILABILITY_TIMESTAMP, 82 Contacts.Impl.VIDEO_CALL_SERVICE_CONTACT_ADDRESS, 83 Contacts.Impl.VIDEO_CALL_CAPABILITY, 84 Contacts.Impl.VIDEO_CALL_CAPABILITY_TIMESTAMP, 85 Contacts.Impl.VIDEO_CALL_AVAILABILITY, 86 Contacts.Impl.VIDEO_CALL_AVAILABILITY_TIMESTAMP, 87 Contacts.Impl.VOLTE_STATUS 88 }; 89 90 /** 91 * Look up the formatted number and Data ID 92 */ 93 private static final String[] DATA_QUERY_PROJECTION = new String[] { 94 Contacts.Impl._ID, 95 Contacts.Impl.FORMATTED_NUMBER, 96 EABContract.EABColumns.DATA_ID 97 }; 98 // Data Query Columns, which match the DATA_QUERY_PROJECTION 99 private static final int DATA_QUERY_ID = 0; 100 private static final int DATA_QUERY_FORMATTED_NUMBER = 1; 101 private static final int DATA_QUERY_DATA_ID = 2; 102 103 104 /** 105 * This class contains all the information necessary to request a new contact. 106 */ 107 public static class Request { 108 private long mId = -1; 109 private String mContactNumber = null; 110 private String mContactName = null; 111 112 private int mVolteCallCapability = -1; 113 private long mVolteCallCapabilityTimeStamp = -1; 114 private int mVolteCallAvailability = -1; 115 private long mVolteCallAvailabilityTimeStamp = -1; 116 private String mVolteCallServiceContactAddress = null; 117 118 private int mVideoCallCapability = -1; 119 private long mVideoCallCapabilityTimeStamp = -1; 120 private int mVideoCallAvailability = -1; 121 private long mVideoCallAvailabilityTimeStamp = -1; 122 private String mVideoCallServiceContactAddress = null; 123 124 private long mContactLastUpdatedTimeStamp = -1; 125 private int mFieldUpdatedFlags = 0; 126 127 private static int sVolteCallCapabilityFlag = 0x0001; 128 private static int sVolteCallCapabilityTimeStampFlag = 0x0002; 129 private static int sVolteCallAvailabilityFlag = 0x0004; 130 private static int sVolteCallAvailabilityTimeStampFlag = 0x0008; 131 private static int sVolteCallServiceContactAddressFlag = 0x0010; 132 private static int sVideoCallCapabilityFlag = 0x0020; 133 private static int sVideoCallCapabilityTimeStampFlag = 0x0040; 134 private static int sVideoCallAvailabilityFlag = 0x0080; 135 private static int sVideoCallAvailabilityTimeStampFlag = 0x0100; 136 private static int sVideoCallServiceContactAddressFlag = 0x0200; 137 private static int sContactLastUpdatedTimeStampFlag = 0x0400; 138 139 /** 140 * @param id the contact id. 141 */ Request(long id)142 public Request(long id) { 143 if (id < 0) { 144 throw new IllegalArgumentException( 145 "Can't update EAB presence item with id: " + id); 146 } 147 148 mId = id; 149 } 150 Request(String number)151 public Request(String number) { 152 if (TextUtils.isEmpty(number)) { 153 throw new IllegalArgumentException( 154 "Can't update EAB presence item with number: " + number); 155 } 156 157 mContactNumber = number; 158 } 159 getContactId()160 public long getContactId() { 161 return mId; 162 } 163 getContactNumber()164 public String getContactNumber() { 165 return mContactNumber; 166 } 167 168 /** 169 * Set Volte call service contact address. 170 * @param address contact from NOTIFY 171 * @return this object 172 */ setVolteCallServiceContactAddress(String address)173 public Request setVolteCallServiceContactAddress(String address) { 174 mVolteCallServiceContactAddress = address; 175 mFieldUpdatedFlags |= sVolteCallServiceContactAddressFlag; 176 return this; 177 } 178 179 /** 180 * Set Volte call capability. 181 * @param b wheter volte call is supported or not 182 * @return this object 183 */ setVolteCallCapability(boolean b)184 public Request setVolteCallCapability(boolean b) { 185 mVolteCallCapability = b ? 1 : 0; 186 mFieldUpdatedFlags |= sVolteCallCapabilityFlag; 187 return this; 188 } 189 setVolteCallCapability(int val)190 public Request setVolteCallCapability(int val) { 191 mVolteCallCapability = val; 192 mFieldUpdatedFlags |= sVolteCallCapabilityFlag; 193 return this; 194 } 195 196 /** 197 * Set Volte call availability. 198 * @param b wheter volte call is available or not 199 * @return this object 200 */ setVolteCallAvailability(boolean b)201 public Request setVolteCallAvailability(boolean b) { 202 mVolteCallAvailability = b ? 1 : 0; 203 mFieldUpdatedFlags |= sVolteCallAvailabilityFlag; 204 return this; 205 } 206 setVolteCallAvailability(int val)207 public Request setVolteCallAvailability(int val) { 208 mVolteCallAvailability = val; 209 mFieldUpdatedFlags |= sVolteCallAvailabilityFlag; 210 return this; 211 } 212 213 /** 214 * Set Video call service contact address. 215 * @param address contact from NOTIFY. 216 * @return this object 217 */ setVideoCallServiceContactAddress(String address)218 public Request setVideoCallServiceContactAddress(String address) { 219 mVideoCallServiceContactAddress = address; 220 mFieldUpdatedFlags |= sVideoCallServiceContactAddressFlag; 221 return this; 222 } 223 224 /** 225 * Set Video call capability. 226 * @param b wheter volte call is supported or not 227 * @return this object 228 */ setVideoCallCapability(boolean b)229 public Request setVideoCallCapability(boolean b) { 230 mVideoCallCapability = b ? 1 : 0; 231 mFieldUpdatedFlags |= sVideoCallCapabilityFlag; 232 return this; 233 } 234 setVideoCallCapability(int val)235 public Request setVideoCallCapability(int val) { 236 mVideoCallCapability = val; 237 mFieldUpdatedFlags |= sVideoCallCapabilityFlag; 238 return this; 239 } 240 241 /** 242 * Set Video call availability. 243 * @param b wheter volte call is available or not 244 * @return this object 245 */ setVideoCallAvailability(boolean b)246 public Request setVideoCallAvailability(boolean b) { 247 mVideoCallAvailability = b ? 1 : 0; 248 mFieldUpdatedFlags |= sVideoCallAvailabilityFlag; 249 return this; 250 } 251 setVideoCallAvailability(int val)252 public Request setVideoCallAvailability(int val) { 253 mVideoCallAvailability = val; 254 mFieldUpdatedFlags |= sVideoCallAvailabilityFlag; 255 return this; 256 } 257 258 /** 259 * Set the update timestamp. 260 * @param long timestamp the last update timestamp 261 * @return this object 262 */ setLastUpdatedTimeStamp(long timestamp)263 public Request setLastUpdatedTimeStamp(long timestamp) { 264 mContactLastUpdatedTimeStamp = timestamp; 265 mFieldUpdatedFlags |= sContactLastUpdatedTimeStampFlag; 266 return this; 267 } 268 setVolteCallCapabilityTimeStamp(long timestamp)269 public Request setVolteCallCapabilityTimeStamp(long timestamp) { 270 mVolteCallCapabilityTimeStamp = timestamp; 271 mFieldUpdatedFlags |= sVolteCallCapabilityTimeStampFlag; 272 return this; 273 } 274 setVolteCallAvailabilityTimeStamp(long timestamp)275 public Request setVolteCallAvailabilityTimeStamp(long timestamp) { 276 mVolteCallAvailabilityTimeStamp = timestamp; 277 mFieldUpdatedFlags |= sVolteCallAvailabilityTimeStampFlag; 278 return this; 279 } 280 setVideoCallCapabilityTimeStamp(long timestamp)281 public Request setVideoCallCapabilityTimeStamp(long timestamp) { 282 mVideoCallCapabilityTimeStamp = timestamp; 283 mFieldUpdatedFlags |= sVideoCallCapabilityTimeStampFlag; 284 return this; 285 } 286 setVideoCallAvailabilityTimeStamp(long timestamp)287 public Request setVideoCallAvailabilityTimeStamp(long timestamp) { 288 mVideoCallAvailabilityTimeStamp = timestamp; 289 mFieldUpdatedFlags |= sVideoCallAvailabilityTimeStampFlag; 290 return this; 291 } 292 reset()293 public Request reset() { 294 mVolteCallCapability = -1; 295 mVolteCallCapabilityTimeStamp = -1; 296 mVolteCallAvailability = -1; 297 mVolteCallAvailabilityTimeStamp = -1; 298 mVolteCallServiceContactAddress = null; 299 300 mVideoCallCapability = -1; 301 mVideoCallCapabilityTimeStamp = -1; 302 mVideoCallAvailability = -1; 303 mVideoCallAvailabilityTimeStamp = -1; 304 mVideoCallServiceContactAddress = null; 305 306 mContactLastUpdatedTimeStamp = -1; 307 mFieldUpdatedFlags = 0; 308 return this; 309 } 310 311 /** 312 * @return ContentValues to be passed to EABProvider.update() 313 */ toContentValues()314 ContentValues toContentValues() { 315 ContentValues values = new ContentValues(); 316 317 if ((mFieldUpdatedFlags & sVolteCallCapabilityFlag) > 0) { 318 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY, 319 mVolteCallCapability); 320 } 321 if ((mFieldUpdatedFlags & sVolteCallCapabilityTimeStampFlag) > 0) { 322 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY_TIMESTAMP, 323 mVolteCallCapabilityTimeStamp); 324 } 325 if ((mFieldUpdatedFlags & sVolteCallAvailabilityFlag) > 0) { 326 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY, 327 mVolteCallAvailability); 328 } 329 if ((mFieldUpdatedFlags & sVolteCallAvailabilityTimeStampFlag) > 0) { 330 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY_TIMESTAMP, 331 mVolteCallAvailabilityTimeStamp); 332 } 333 if ((mFieldUpdatedFlags & sVolteCallServiceContactAddressFlag) > 0) { 334 values.put(Contacts.Impl.VOLTE_CALL_SERVICE_CONTACT_ADDRESS, 335 mVolteCallServiceContactAddress); 336 } 337 338 if ((mFieldUpdatedFlags & sVideoCallCapabilityFlag) > 0) { 339 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY, 340 mVideoCallCapability); 341 } 342 if ((mFieldUpdatedFlags & sVideoCallCapabilityTimeStampFlag) > 0) { 343 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY_TIMESTAMP, 344 mVideoCallCapabilityTimeStamp); 345 } 346 if ((mFieldUpdatedFlags & sVideoCallAvailabilityFlag) > 0) { 347 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY, 348 mVideoCallAvailability); 349 } 350 if ((mFieldUpdatedFlags & sVideoCallAvailabilityTimeStampFlag) > 0) { 351 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY_TIMESTAMP, 352 mVideoCallAvailabilityTimeStamp); 353 } 354 if ((mFieldUpdatedFlags & sVideoCallServiceContactAddressFlag) > 0) { 355 values.put(Contacts.Impl.VIDEO_CALL_SERVICE_CONTACT_ADDRESS, 356 mVideoCallServiceContactAddress); 357 } 358 359 if ((mFieldUpdatedFlags & sContactLastUpdatedTimeStampFlag) > 0 ) { 360 values.put(Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP, 361 mContactLastUpdatedTimeStamp); 362 } 363 364 return values; 365 } 366 367 @Override toString()368 public String toString() { 369 StringBuilder sb = new StringBuilder(512); 370 sb.append("EABContactManager.Request { "); 371 if (mId != -1) { 372 sb.append("\nId: " + mId); 373 } 374 if (!TextUtils.isEmpty(mContactNumber)) { 375 sb.append("\nContact Number: " + mContactNumber); 376 } 377 if (!TextUtils.isEmpty(mContactName)) { 378 sb.append("\nContact Name: " + mContactName); 379 } 380 381 if ((mFieldUpdatedFlags & sVolteCallCapabilityFlag) > 0) { 382 sb.append("\nVolte call capability: " + mVolteCallCapability); 383 } 384 if ((mFieldUpdatedFlags & sVolteCallCapabilityTimeStampFlag) > 0) { 385 sb.append("\nVolte call capability timestamp: " + mVolteCallCapabilityTimeStamp 386 + "(" + getTimeString(mVolteCallCapabilityTimeStamp) + ")"); 387 } 388 if ((mFieldUpdatedFlags & sVolteCallAvailabilityFlag) > 0) { 389 sb.append("\nVolte call availability: " + mVolteCallAvailability); 390 } 391 if ((mFieldUpdatedFlags & sVolteCallAvailabilityTimeStampFlag) > 0) { 392 sb.append("\nVolte call availablity timestamp: " + mVolteCallAvailabilityTimeStamp 393 + "(" + getTimeString(mVolteCallAvailabilityTimeStamp) + ")"); 394 } 395 if ((mFieldUpdatedFlags & sVolteCallServiceContactAddressFlag) > 0) { 396 sb.append("\nVolte Call Service address: " + mVolteCallServiceContactAddress); 397 } 398 399 if ((mFieldUpdatedFlags & sVideoCallCapabilityFlag) > 0) { 400 sb.append("\nVideo call capability: " + mVideoCallCapability); 401 } 402 if ((mFieldUpdatedFlags & sVideoCallCapabilityTimeStampFlag) > 0) { 403 sb.append("\nVideo call capability timestamp: " + mVideoCallCapabilityTimeStamp 404 + "(" + getTimeString(mVideoCallCapabilityTimeStamp) + ")"); 405 } 406 if ((mFieldUpdatedFlags & sVideoCallAvailabilityFlag) > 0) { 407 sb.append("\nVideo call availability: " + mVideoCallAvailability); 408 } 409 if ((mFieldUpdatedFlags & sVideoCallAvailabilityTimeStampFlag) > 0) { 410 sb.append("\nVideo call availablity timestamp: " + mVideoCallAvailabilityTimeStamp 411 + "(" + getTimeString(mVideoCallAvailabilityTimeStamp) + ")"); 412 } 413 if ((mFieldUpdatedFlags & sVideoCallServiceContactAddressFlag) > 0) { 414 sb.append("\nVideo Call Service address: " + mVideoCallServiceContactAddress); 415 } 416 417 if ((mFieldUpdatedFlags & sContactLastUpdatedTimeStampFlag) > 0 ) { 418 sb.append("\nContact last update time: " + mContactLastUpdatedTimeStamp 419 + "(" + getTimeString(mContactLastUpdatedTimeStamp) + ")"); 420 } 421 422 sb.append(" }"); 423 return sb.toString(); 424 } 425 } 426 427 /** 428 * This class may be used to filter EABProvider queries. 429 */ 430 public static class Query { 431 /** 432 * Constant for use with {@link #orderBy} 433 * @hide 434 */ 435 public static final int ORDER_ASCENDING = 1; 436 437 /** 438 * Constant for use with {@link #orderBy} 439 * @hide 440 */ 441 public static final int ORDER_DESCENDING = 2; 442 443 private long[] mIds = null; 444 private String mContactNumber = null; 445 private List<String> mTimeFilters = null; 446 private String mOrderByColumn = COLUMN_LAST_UPDATED_TIMESTAMP; 447 private int mOrderDirection = ORDER_ASCENDING; 448 449 /** 450 * Include only the contacts with the given IDs. 451 * @return this object 452 */ setFilterById(long... ids)453 public Query setFilterById(long... ids) { 454 mIds = ids; 455 return this; 456 } 457 458 /** 459 * Include only the contacts with the given number. 460 * @return this object 461 */ setFilterByNumber(String number)462 public Query setFilterByNumber(String number) { 463 mContactNumber = number; 464 return this; 465 } 466 467 /** 468 * Include the contacts that meet the specified time condition. 469 * @return this object 470 */ setFilterByTime(String selection)471 public Query setFilterByTime(String selection) { 472 if (mTimeFilters == null) { 473 mTimeFilters = new ArrayList<String>(); 474 } 475 476 mTimeFilters.add(selection); 477 return this; 478 } 479 480 /** 481 * Include only the contacts that has not been updated before the last time. 482 * @return this object 483 */ setFilterByTime(String column, long last)484 public Query setFilterByTime(String column, long last) { 485 if (mTimeFilters == null) { 486 mTimeFilters = new ArrayList<String>(); 487 } 488 489 mTimeFilters.add(column + "<='" + last + "'"); 490 return this; 491 } 492 493 /** 494 * Include only the contacts that has not been updated after the eariest time. 495 * @return this object 496 */ setFilterByEarliestTime(String column, long earliest)497 public Query setFilterByEarliestTime(String column, long earliest) { 498 if (mTimeFilters == null) { 499 mTimeFilters = new ArrayList<String>(); 500 } 501 502 mTimeFilters.add(column + ">='" + earliest + "'"); 503 return this; 504 } 505 506 /** 507 * Change the sort order of the returned Cursor. 508 * 509 * @param column one of the COLUMN_* constants; currently, only 510 * {@link #COLUMN_LAST_MODIFIED_TIMESTAMP} and {@link #COLUMN_TOTAL_SIZE_BYTES} are 511 * supported. 512 * @param direction either {@link #ORDER_ASCENDING} or {@link #ORDER_DESCENDING} 513 * @return this object 514 * @hide 515 */ orderBy(String column, int direction)516 public Query orderBy(String column, int direction) { 517 if (direction != ORDER_ASCENDING && direction != ORDER_DESCENDING) { 518 throw new IllegalArgumentException("Invalid direction: " + direction); 519 } 520 521 if (column.equals(COLUMN_ID)) { 522 mOrderByColumn = Contacts.Impl._ID; 523 } else if (column.equals(COLUMN_LAST_UPDATED_TIMESTAMP)) { 524 mOrderByColumn = Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP; 525 } else { 526 throw new IllegalArgumentException("Cannot order by " + column); 527 } 528 mOrderDirection = direction; 529 return this; 530 } 531 532 /** 533 * Run this query using the given ContentResolver. 534 * @param projection the projection to pass to ContentResolver.query() 535 * @return the Cursor returned by ContentResolver.query() 536 */ runQuery(ContentResolver resolver, String[] projection, Uri baseUri)537 Cursor runQuery(ContentResolver resolver, String[] projection, Uri baseUri) { 538 Uri uri = baseUri; 539 List<String> selectionParts = new ArrayList<String>(); 540 String[] selectionArgs = null; 541 542 if (mIds != null) { 543 selectionParts.add(getWhereClauseForIds(mIds)); 544 selectionArgs = getWhereArgsForIds(mIds); 545 } 546 547 if (!TextUtils.isEmpty(mContactNumber)) { 548 String number = mContactNumber; 549 if (number.startsWith("tel:")) { 550 number = number.substring(4); 551 } 552 String escapedPhoneNumber = DatabaseUtils.sqlEscapeString(number); 553 String cselection = "(" + Contacts.Impl.CONTACT_NUMBER + "=" + escapedPhoneNumber; 554 cselection += " OR PHONE_NUMBERS_EQUAL(" + Contacts.Impl.CONTACT_NUMBER + ", "; 555 cselection += escapedPhoneNumber + ", 0))"; 556 557 selectionParts.add(cselection); 558 } 559 560 if (mTimeFilters != null) { 561 String cselection = joinStrings(" OR ", mTimeFilters); 562 int size = mTimeFilters.size(); 563 if (size > 1) { 564 cselection = "(" + cselection + ")"; 565 } 566 selectionParts.add(cselection); 567 } 568 569 String selection = joinStrings(" AND ", selectionParts); 570 String orderDirection = (mOrderDirection == ORDER_ASCENDING ? "ASC" : "DESC"); 571 String orderBy = mOrderByColumn + " " + orderDirection; 572 573 return resolver.query(uri, projection, selection, selectionArgs, orderBy); 574 } 575 joinStrings(String joiner, Iterable<String> parts)576 private String joinStrings(String joiner, Iterable<String> parts) { 577 StringBuilder builder = new StringBuilder(); 578 boolean first = true; 579 for (String part : parts) { 580 if (!first) { 581 builder.append(joiner); 582 } 583 builder.append(part); 584 first = false; 585 } 586 return builder.toString(); 587 } 588 589 @Override toString()590 public String toString() { 591 List<String> selectionParts = new ArrayList<String>(); 592 String[] selectionArgs = null; 593 594 if (mIds != null) { 595 selectionParts.add(getWhereClauseForIds(mIds)); 596 selectionArgs = getWhereArgsForIds(mIds); 597 } 598 599 if (!TextUtils.isEmpty(mContactNumber)) { 600 String number = mContactNumber; 601 if (number.startsWith("tel:")) { 602 number = number.substring(4); 603 } 604 String escapedPhoneNumber = DatabaseUtils.sqlEscapeString(number); 605 String cselection = "(" + Contacts.Impl.CONTACT_NUMBER + "=" + escapedPhoneNumber; 606 cselection += " OR PHONE_NUMBERS_EQUAL(" + Contacts.Impl.CONTACT_NUMBER + ", "; 607 cselection += escapedPhoneNumber + ", 0))"; 608 609 selectionParts.add(cselection); 610 } 611 612 if (mTimeFilters != null) { 613 String cselection = joinStrings(" OR ", mTimeFilters); 614 int size = mTimeFilters.size(); 615 if (size > 1) { 616 cselection = "(" + cselection + ")"; 617 } 618 selectionParts.add(cselection); 619 } 620 621 String selection = joinStrings(" AND ", selectionParts); 622 String orderDirection = (mOrderDirection == ORDER_ASCENDING ? "ASC" : "DESC"); 623 String orderBy = mOrderByColumn + " " + orderDirection; 624 625 StringBuilder sb = new StringBuilder(512); 626 sb.append("EABContactManager.Query { "); 627 sb.append("\nSelection: " + selection); 628 sb.append("\nSelectionArgs: " + selectionArgs); 629 sb.append("\nOrderBy: " + orderBy); 630 sb.append(" }"); 631 return sb.toString(); 632 } 633 } 634 635 private ContentResolver mResolver; 636 private String mPackageName; 637 private Uri mBaseUri = Contacts.Impl.CONTENT_URI; 638 639 /** 640 * @hide 641 */ EABContactManager(ContentResolver resolver, String packageName)642 public EABContactManager(ContentResolver resolver, String packageName) { 643 mResolver = resolver; 644 mPackageName = packageName; 645 } 646 647 /** 648 * Query the presence manager about contacts that have been requested. 649 * @param query parameters specifying filters for this query 650 * @return a Cursor over the result set of contacts, with columns consisting of all the 651 * COLUMN_* constants. 652 */ query(Query query)653 public Cursor query(Query query) { 654 Cursor underlyingCursor = query.runQuery(mResolver, CONTACT_COLUMNS, mBaseUri); 655 if (underlyingCursor == null) { 656 return null; 657 } 658 659 return new CursorTranslator(underlyingCursor, mBaseUri); 660 } 661 662 /** 663 * Update a contact presence status. 664 * 665 * @param request the parameters specifying this contact presence 666 * @return an ID for the contact, unique across the system. This ID is used to make future 667 * calls related to this contact. 668 */ update(Request request)669 public int update(Request request) { 670 if (request == null) { 671 return 0; 672 } 673 674 long id = request.getContactId(); 675 String number = request.getContactNumber(); 676 if ((id == -1) && TextUtils.isEmpty(number)) { 677 throw new IllegalArgumentException("invalid request for contact update."); 678 } 679 680 ContentValues values = request.toContentValues(); 681 if (id != -1) { 682 logger.debug("Update contact " + id + " with request: " + values); 683 return mResolver.update(ContentUris.withAppendedId(mBaseUri, id), values, 684 null, null); 685 } else { 686 Query query = new Query().setFilterByNumber(number) 687 .orderBy(COLUMN_ID, Query.ORDER_ASCENDING); 688 long[] ids = null; 689 Cursor cursor = null; 690 try { 691 cursor = query(query); 692 if (cursor == null) { 693 return 0; 694 } 695 int count = cursor.getCount(); 696 if (count == 0) { 697 return 0; 698 } 699 700 ids = new long[count]; 701 int idx = 0; 702 for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) { 703 id = cursor.getLong(cursor.getColumnIndex(Contacts.Impl._ID)); 704 ids[idx++] = id; 705 if (idx >= count) { 706 break; 707 } 708 } 709 } finally { 710 if (cursor != null) { 711 cursor.close(); 712 } 713 } 714 715 if ((ids == null) || (ids.length == 0)) { 716 return 0; 717 } 718 719 if (ids.length == 1) { 720 logger.debug("Update contact " + ids[0] + " with request: " + values); 721 return mResolver.update(ContentUris.withAppendedId(mBaseUri, ids[0]), values, 722 null, null); 723 } 724 725 logger.debug("Update contact " + number + " with request: " + values); 726 return mResolver.update(mBaseUri, values, getWhereClauseForIds(ids), 727 getWhereArgsForIds(ids)); 728 } 729 } 730 731 /** 732 * Get the EABProvider URI for the contact with the given ID. 733 * 734 * @hide 735 */ getContactUri(long id)736 public Uri getContactUri(long id) { 737 return ContentUris.withAppendedId(mBaseUri, id); 738 } 739 740 /** 741 * Get a parameterized SQL WHERE clause to select a bunch of IDs. 742 */ getWhereClauseForIds(long[] ids)743 static String getWhereClauseForIds(long[] ids) { 744 StringBuilder whereClause = new StringBuilder(); 745 whereClause.append("("); 746 for (int i = 0; i < ids.length; i++) { 747 if (i > 0) { 748 whereClause.append("OR "); 749 } 750 whereClause.append(COLUMN_ID); 751 whereClause.append(" = ? "); 752 } 753 whereClause.append(")"); 754 return whereClause.toString(); 755 } 756 757 /** 758 * Get the selection args for a clause returned by {@link #getWhereClauseForIds(long[])}. 759 */ getWhereArgsForIds(long[] ids)760 static String[] getWhereArgsForIds(long[] ids) { 761 String[] whereArgs = new String[ids.length]; 762 for (int i = 0; i < ids.length; i++) { 763 whereArgs[i] = Long.toString(ids[i]); 764 } 765 return whereArgs; 766 } 767 getTimeString(long time)768 static String getTimeString(long time) { 769 if (time <= 0) { 770 time = System.currentTimeMillis(); 771 } 772 773 String timeString = TimeMigrationUtils.formatMillisWithFixedFormat(time); 774 return String.format("%s.%s", timeString, time % 1000); 775 } 776 777 /** 778 * This class wraps a cursor returned by EABProvider -- the "underlying cursor" -- and 779 * presents a different set of columns, those defined in the COLUMN_* constants. 780 * Some columns correspond directly to underlying values while others are computed from 781 * underlying data. 782 */ 783 private static class CursorTranslator extends CursorWrapper { 784 private Uri mBaseUri; 785 CursorTranslator(Cursor cursor, Uri baseUri)786 public CursorTranslator(Cursor cursor, Uri baseUri) { 787 super(cursor); 788 mBaseUri = baseUri; 789 } 790 791 @Override getInt(int columnIndex)792 public int getInt(int columnIndex) { 793 return (int) getLong(columnIndex); 794 } 795 796 @Override getLong(int columnIndex)797 public long getLong(int columnIndex) { 798 return super.getLong(columnIndex); 799 } 800 801 @Override getString(int columnIndex)802 public String getString(int columnIndex) { 803 return super.getString(columnIndex); 804 } 805 } 806 updateAllCapabilityToUnknown()807 public void updateAllCapabilityToUnknown() { 808 if (mResolver == null) { 809 logger.error("updateAllCapabilityToUnknown, mResolver=null"); 810 return; 811 } 812 813 ContentValues values = new ContentValues(); 814 values.put(Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP, (String)null); 815 values.put(Contacts.Impl.VOLTE_CALL_SERVICE_CONTACT_ADDRESS, (String)null); 816 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY, (String)null); 817 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY_TIMESTAMP, (String)null); 818 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY, (String)null); 819 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY_TIMESTAMP, (String)null); 820 821 values.put(Contacts.Impl.VIDEO_CALL_SERVICE_CONTACT_ADDRESS, (String)null); 822 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY, (String)null); 823 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY_TIMESTAMP, (String)null); 824 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY, (String)null); 825 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY_TIMESTAMP, (String)null); 826 827 try { 828 int count = ContactDbUtil.resetVtCapability(mResolver); 829 logger.print("update Contact DB: updateAllCapabilityToUnknown count=" + count); 830 831 count = mResolver.update(Contacts.Impl.CONTENT_URI, 832 values, null, null); 833 logger.print("update EAB DB: updateAllCapabilityToUnknown count=" + count); 834 } catch (Exception ex) { 835 logger.error("updateAllCapabilityToUnknown exception: " + ex); 836 } 837 } 838 updateAllVtCapabilityToUnknown()839 public void updateAllVtCapabilityToUnknown() { 840 if (mResolver == null) { 841 logger.error("updateAllVtCapabilityToUnknown mResolver=null"); 842 return; 843 } 844 845 ContentValues values = new ContentValues(); 846 values.put(Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP, (String)null); 847 values.put(Contacts.Impl.VIDEO_CALL_SERVICE_CONTACT_ADDRESS, (String)null); 848 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY, (String)null); 849 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY_TIMESTAMP, (String)null); 850 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY, (String)null); 851 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY_TIMESTAMP, (String)null); 852 853 try { 854 int count = ContactDbUtil.resetVtCapability(mResolver); 855 logger.print("update Contact DB: updateAllVtCapabilityToUnknown count=" + count); 856 857 count = mResolver.update(Contacts.Impl.CONTENT_URI, 858 values, null, null); 859 logger.print("update EAB DB: updateAllVtCapabilityToUnknown count=" + count); 860 } catch (Exception ex) { 861 logger.error("updateAllVtCapabilityToUnknown exception: " + ex); 862 } 863 } 864 865 // if updateLastTimestamp is true, the rcsPresenceInfo is from network. 866 // if the updateLastTimestamp is false, It is used to update the availabilty to unknown only. 867 // And the availability will be updated only when it has expired, so we don't update the 868 // timestamp to make sure the availablity still in expired status and will be subscribed from 869 // network afterwards. update(RcsPresenceInfo rcsPresenceInfo, boolean updateLastTimestamp)870 public int update(RcsPresenceInfo rcsPresenceInfo, boolean updateLastTimestamp) { 871 if (rcsPresenceInfo == null) { 872 return 0; 873 } 874 875 String number = rcsPresenceInfo.getContactNumber(); 876 if (TextUtils.isEmpty(number)) { 877 logger.error("Failed to update for the contact number is empty."); 878 return 0; 879 } 880 881 ContentValues values = new ContentValues(); 882 883 int volteStatus = rcsPresenceInfo.getVolteStatus(); 884 if(volteStatus != RcsPresenceInfo.VolteStatus.VOLTE_UNKNOWN) { 885 values.put(Contacts.Impl.VOLTE_STATUS, volteStatus); 886 } 887 888 if(updateLastTimestamp){ 889 values.put(Contacts.Impl.CONTACT_LAST_UPDATED_TIMESTAMP, 890 (long)System.currentTimeMillis()); 891 } 892 893 int lteCallCapability = rcsPresenceInfo.getServiceState( 894 RcsPresenceInfo.ServiceType.VOLTE_CALL); 895 long lteCallTimestamp = rcsPresenceInfo.getTimeStamp( 896 RcsPresenceInfo.ServiceType.VOLTE_CALL); 897 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY, lteCallCapability); 898 values.put(Contacts.Impl.VOLTE_CALL_AVAILABILITY_TIMESTAMP, (long)lteCallTimestamp); 899 if(rcsPresenceInfo.getServiceState(RcsPresenceInfo.ServiceType.VOLTE_CALL) 900 != RcsPresenceInfo.ServiceState.UNKNOWN){ 901 String lteCallContactAddress = 902 rcsPresenceInfo.getServiceContact(RcsPresenceInfo.ServiceType.VOLTE_CALL); 903 if (!TextUtils.isEmpty(lteCallContactAddress)) { 904 values.put(Contacts.Impl.VOLTE_CALL_SERVICE_CONTACT_ADDRESS, lteCallContactAddress); 905 } 906 907 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY, lteCallCapability); 908 values.put(Contacts.Impl.VOLTE_CALL_CAPABILITY_TIMESTAMP, lteCallTimestamp); 909 } 910 911 int videoCallCapability = rcsPresenceInfo.getServiceState( 912 RcsPresenceInfo.ServiceType.VT_CALL); 913 long videoCallTimestamp = rcsPresenceInfo.getTimeStamp( 914 RcsPresenceInfo.ServiceType.VT_CALL); 915 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY, videoCallCapability); 916 values.put(Contacts.Impl.VIDEO_CALL_AVAILABILITY_TIMESTAMP, (long)videoCallTimestamp); 917 if(rcsPresenceInfo.getServiceState(RcsPresenceInfo.ServiceType.VT_CALL) 918 != RcsPresenceInfo.ServiceState.UNKNOWN){ 919 String videoCallContactAddress = 920 rcsPresenceInfo.getServiceContact(RcsPresenceInfo.ServiceType.VT_CALL); 921 if (!TextUtils.isEmpty(videoCallContactAddress)) { 922 values.put(Contacts.Impl.VIDEO_CALL_SERVICE_CONTACT_ADDRESS, 923 videoCallContactAddress); 924 } 925 926 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY, videoCallCapability); 927 values.put(Contacts.Impl.VIDEO_CALL_CAPABILITY_TIMESTAMP, videoCallTimestamp); 928 } 929 930 int count = 0; 931 Cursor cursor = null; 932 try{ 933 cursor = mResolver.query(Contacts.Impl.CONTENT_URI, DATA_QUERY_PROJECTION, 934 "PHONE_NUMBERS_EQUAL(" + Contacts.Impl.FORMATTED_NUMBER + ", ?, 1)", 935 new String[] {number}, null); 936 if(cursor == null) { 937 logger.print("update rcsPresenceInfo to DB: update count=" + count); 938 return count; 939 } 940 941 ContactNumberUtils contactNumberUtils = ContactNumberUtils.getDefault(); 942 for (cursor.moveToFirst(); !cursor.isAfterLast(); cursor.moveToNext()) { 943 String numberInDB = cursor.getString(DATA_QUERY_FORMATTED_NUMBER); 944 logger.debug("number=" + number + " numberInDB=" + numberInDB + 945 " formatedNumber in DB=" + contactNumberUtils.format(numberInDB)); 946 if(number.equals(contactNumberUtils.format(numberInDB))) { 947 count = ContactDbUtil.updateVtCapability(mResolver, 948 cursor.getLong(DATA_QUERY_DATA_ID), 949 (videoCallCapability == RcsPresenceInfo.ServiceState.ONLINE)); 950 logger.print("update rcsPresenceInfo to Contact DB, count=" + count); 951 952 int id = cursor.getInt(DATA_QUERY_ID); 953 count += mResolver.update(Contacts.Impl.CONTENT_URI, values, 954 Contacts.Impl._ID + "=" + id, null); 955 logger.debug("count=" + count); 956 } 957 } 958 959 logger.print("update rcsPresenceInfo to DB: update count=" + count + 960 " rcsPresenceInfo=" + rcsPresenceInfo); 961 } catch(Exception e){ 962 logger.error("updateCapability exception"); 963 } finally { 964 if(cursor != null) { 965 cursor.close(); 966 cursor = null; 967 } 968 } 969 970 return count; 971 } 972 } 973 974