1 /* 2 * Copyright (C) 2006 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.content; 18 19 import android.accounts.Account; 20 import android.annotation.IntDef; 21 import android.annotation.NonNull; 22 import android.annotation.Nullable; 23 import android.annotation.RequiresPermission; 24 import android.annotation.TestApi; 25 import android.annotation.UserIdInt; 26 import android.app.ActivityManager; 27 import android.app.ActivityThread; 28 import android.app.AppGlobals; 29 import android.content.pm.PackageManager.NameNotFoundException; 30 import android.content.res.AssetFileDescriptor; 31 import android.content.res.Resources; 32 import android.database.ContentObserver; 33 import android.database.CrossProcessCursorWrapper; 34 import android.database.Cursor; 35 import android.database.IContentObserver; 36 import android.graphics.Point; 37 import android.graphics.drawable.Drawable; 38 import android.net.Uri; 39 import android.os.Bundle; 40 import android.os.CancellationSignal; 41 import android.os.DeadObjectException; 42 import android.os.IBinder; 43 import android.os.ICancellationSignal; 44 import android.os.OperationCanceledException; 45 import android.os.ParcelFileDescriptor; 46 import android.os.RemoteException; 47 import android.os.ServiceManager; 48 import android.os.SystemClock; 49 import android.os.UserHandle; 50 import android.text.TextUtils; 51 import android.util.EventLog; 52 import android.util.Log; 53 54 import com.android.internal.util.ArrayUtils; 55 import com.android.internal.util.MimeIconUtils; 56 import com.android.internal.util.Preconditions; 57 58 import dalvik.system.CloseGuard; 59 60 import java.io.File; 61 import java.io.FileInputStream; 62 import java.io.FileNotFoundException; 63 import java.io.IOException; 64 import java.io.InputStream; 65 import java.io.OutputStream; 66 import java.lang.annotation.Retention; 67 import java.lang.annotation.RetentionPolicy; 68 import java.util.ArrayList; 69 import java.util.List; 70 import java.util.Random; 71 import java.util.concurrent.atomic.AtomicBoolean; 72 73 /** 74 * This class provides applications access to the content model. 75 * 76 * <div class="special reference"> 77 * <h3>Developer Guides</h3> 78 * <p>For more information about using a ContentResolver with content providers, read the 79 * <a href="{@docRoot}guide/topics/providers/content-providers.html">Content Providers</a> 80 * developer guide.</p> 81 */ 82 public abstract class ContentResolver { 83 /** 84 * @deprecated instead use 85 * {@link #requestSync(android.accounts.Account, String, android.os.Bundle)} 86 */ 87 @Deprecated 88 public static final String SYNC_EXTRAS_ACCOUNT = "account"; 89 90 /** 91 * If this extra is set to true, the sync request will be scheduled 92 * at the front of the sync request queue and without any delay 93 */ 94 public static final String SYNC_EXTRAS_EXPEDITED = "expedited"; 95 96 /** 97 * If this extra is set to true, the sync request will be scheduled 98 * only when the device is plugged in. This is equivalent to calling 99 * setRequiresCharging(true) on {@link SyncRequest}. 100 */ 101 public static final String SYNC_EXTRAS_REQUIRE_CHARGING = "require_charging"; 102 103 /** 104 * @deprecated instead use 105 * {@link #SYNC_EXTRAS_MANUAL} 106 */ 107 @Deprecated 108 public static final String SYNC_EXTRAS_FORCE = "force"; 109 110 /** 111 * If this extra is set to true then the sync settings (like getSyncAutomatically()) 112 * are ignored by the sync scheduler. 113 */ 114 public static final String SYNC_EXTRAS_IGNORE_SETTINGS = "ignore_settings"; 115 116 /** 117 * If this extra is set to true then any backoffs for the initial attempt (e.g. due to retries) 118 * are ignored by the sync scheduler. If this request fails and gets rescheduled then the 119 * retries will still honor the backoff. 120 */ 121 public static final String SYNC_EXTRAS_IGNORE_BACKOFF = "ignore_backoff"; 122 123 /** 124 * If this extra is set to true then the request will not be retried if it fails. 125 */ 126 public static final String SYNC_EXTRAS_DO_NOT_RETRY = "do_not_retry"; 127 128 /** 129 * Setting this extra is the equivalent of setting both {@link #SYNC_EXTRAS_IGNORE_SETTINGS} 130 * and {@link #SYNC_EXTRAS_IGNORE_BACKOFF} 131 */ 132 public static final String SYNC_EXTRAS_MANUAL = "force"; 133 134 /** 135 * Indicates that this sync is intended to only upload local changes to the server. 136 * For example, this will be set to true if the sync is initiated by a call to 137 * {@link ContentResolver#notifyChange(android.net.Uri, android.database.ContentObserver, boolean)} 138 */ 139 public static final String SYNC_EXTRAS_UPLOAD = "upload"; 140 141 /** 142 * Indicates that the sync adapter should proceed with the delete operations, 143 * even if it determines that there are too many. 144 * See {@link SyncResult#tooManyDeletions} 145 */ 146 public static final String SYNC_EXTRAS_OVERRIDE_TOO_MANY_DELETIONS = "deletions_override"; 147 148 /** 149 * Indicates that the sync adapter should not proceed with the delete operations, 150 * if it determines that there are too many. 151 * See {@link SyncResult#tooManyDeletions} 152 */ 153 public static final String SYNC_EXTRAS_DISCARD_LOCAL_DELETIONS = "discard_deletions"; 154 155 /* Extensions to API. TODO: Not clear if we will keep these as public flags. */ 156 /** {@hide} User-specified flag for expected upload size. */ 157 public static final String SYNC_EXTRAS_EXPECTED_UPLOAD = "expected_upload"; 158 159 /** {@hide} User-specified flag for expected download size. */ 160 public static final String SYNC_EXTRAS_EXPECTED_DOWNLOAD = "expected_download"; 161 162 /** {@hide} Priority of this sync with respect to other syncs scheduled for this application. */ 163 public static final String SYNC_EXTRAS_PRIORITY = "sync_priority"; 164 165 /** {@hide} Flag to allow sync to occur on metered network. */ 166 public static final String SYNC_EXTRAS_DISALLOW_METERED = "allow_metered"; 167 168 /** 169 * Set by the SyncManager to request that the SyncAdapter initialize itself for 170 * the given account/authority pair. One required initialization step is to 171 * ensure that {@link #setIsSyncable(android.accounts.Account, String, int)} has been 172 * called with a >= 0 value. When this flag is set the SyncAdapter does not need to 173 * do a full sync, though it is allowed to do so. 174 */ 175 public static final String SYNC_EXTRAS_INITIALIZE = "initialize"; 176 177 /** @hide */ 178 public static final Intent ACTION_SYNC_CONN_STATUS_CHANGED = 179 new Intent("com.android.sync.SYNC_CONN_STATUS_CHANGED"); 180 181 public static final String SCHEME_CONTENT = "content"; 182 public static final String SCHEME_ANDROID_RESOURCE = "android.resource"; 183 public static final String SCHEME_FILE = "file"; 184 185 /** 186 * An extra {@link Point} describing the optimal size for a requested image 187 * resource, in pixels. If a provider has multiple sizes of the image, it 188 * should return the image closest to this size. 189 * 190 * @see #openTypedAssetFileDescriptor(Uri, String, Bundle) 191 * @see #openTypedAssetFileDescriptor(Uri, String, Bundle, 192 * CancellationSignal) 193 */ 194 public static final String EXTRA_SIZE = "android.content.extra.SIZE"; 195 196 /** 197 * An extra boolean describing whether a particular provider supports refresh 198 * or not. If a provider supports refresh, it should include this key in its 199 * returned Cursor as part of its query call. 200 * 201 */ 202 public static final String EXTRA_REFRESH_SUPPORTED = "android.content.extra.REFRESH_SUPPORTED"; 203 204 /** 205 * Key for an SQL style selection string that may be present in the query Bundle argument 206 * passed to {@link ContentProvider#query(Uri, String[], Bundle, CancellationSignal)} 207 * when called by a legacy client. 208 * 209 * <p>Clients should never include user supplied values directly in the selection string, 210 * as this presents an avenue for SQL injection attacks. In lieu of this, a client 211 * should use standard placeholder notation to represent values in a selection string, 212 * then supply a corresponding value in {@value #QUERY_ARG_SQL_SELECTION_ARGS}. 213 * 214 * <p><b>Apps targeting {@link android.os.Build.VERSION_CODES#O} or higher are strongly 215 * encourage to use structured query arguments in lieu of opaque SQL query clauses.</b> 216 * 217 * @see #QUERY_ARG_SORT_COLUMNS 218 * @see #QUERY_ARG_SORT_DIRECTION 219 * @see #QUERY_ARG_SORT_COLLATION 220 */ 221 public static final String QUERY_ARG_SQL_SELECTION = "android:query-arg-sql-selection"; 222 223 /** 224 * Key for SQL selection string arguments list. 225 * 226 * <p>Clients should never include user supplied values directly in the selection string, 227 * as this presents an avenue for SQL injection attacks. In lieu of this, a client 228 * should use standard placeholder notation to represent values in a selection string, 229 * then supply a corresponding value in {@value #QUERY_ARG_SQL_SELECTION_ARGS}. 230 * 231 * <p><b>Apps targeting {@link android.os.Build.VERSION_CODES#O} or higher are strongly 232 * encourage to use structured query arguments in lieu of opaque SQL query clauses.</b> 233 * 234 * @see #QUERY_ARG_SORT_COLUMNS 235 * @see #QUERY_ARG_SORT_DIRECTION 236 * @see #QUERY_ARG_SORT_COLLATION 237 */ 238 public static final String QUERY_ARG_SQL_SELECTION_ARGS = 239 "android:query-arg-sql-selection-args"; 240 241 /** 242 * Key for an SQL style sort string that may be present in the query Bundle argument 243 * passed to {@link ContentProvider#query(Uri, String[], Bundle, CancellationSignal)} 244 * when called by a legacy client. 245 * 246 * <p><b>Apps targeting {@link android.os.Build.VERSION_CODES#O} or higher are strongly 247 * encourage to use structured query arguments in lieu of opaque SQL query clauses.</b> 248 * 249 * @see #QUERY_ARG_SORT_COLUMNS 250 * @see #QUERY_ARG_SORT_DIRECTION 251 * @see #QUERY_ARG_SORT_COLLATION 252 */ 253 public static final String QUERY_ARG_SQL_SORT_ORDER = "android:query-arg-sql-sort-order"; 254 255 /** 256 * Specifies the list of columns against which to sort results. When first column values 257 * are identical, records are then sorted based on second column values, and so on. 258 * 259 * <p>Columns present in this list must also be included in the projection 260 * supplied to {@link ContentResolver#query(Uri, String[], Bundle, CancellationSignal)}. 261 * 262 * <p>Apps targeting {@link android.os.Build.VERSION_CODES#O} or higher: 263 * 264 * <li>{@link ContentProvider} implementations: When preparing data in 265 * {@link ContentProvider#query(Uri, String[], Bundle, CancellationSignal)}, if sort columns 266 * is reflected in the returned Cursor, it is strongly recommended that 267 * {@link #QUERY_ARG_SORT_COLUMNS} then be included in the array of honored arguments 268 * reflected in {@link Cursor} extras {@link Bundle} under {@link #EXTRA_HONORED_ARGS}. 269 * 270 * <li>When querying a provider, where no QUERY_ARG_SQL* otherwise exists in the 271 * arguments {@link Bundle}, the Content framework will attempt to synthesize 272 * an QUERY_ARG_SQL* argument using the corresponding QUERY_ARG_SORT* values. 273 */ 274 public static final String QUERY_ARG_SORT_COLUMNS = "android:query-arg-sort-columns"; 275 276 /** 277 * Specifies desired sort order. When unspecified a provider may provide a default 278 * sort direction, or choose to return unsorted results. 279 * 280 * <p>Apps targeting {@link android.os.Build.VERSION_CODES#O} or higher: 281 * 282 * <li>{@link ContentProvider} implementations: When preparing data in 283 * {@link ContentProvider#query(Uri, String[], Bundle, CancellationSignal)}, if sort direction 284 * is reflected in the returned Cursor, it is strongly recommended that 285 * {@link #QUERY_ARG_SORT_DIRECTION} then be included in the array of honored arguments 286 * reflected in {@link Cursor} extras {@link Bundle} under {@link #EXTRA_HONORED_ARGS}. 287 * 288 * <li>When querying a provider, where no QUERY_ARG_SQL* otherwise exists in the 289 * arguments {@link Bundle}, the Content framework will attempt to synthesize 290 * a QUERY_ARG_SQL* argument using the corresponding QUERY_ARG_SORT* values. 291 * 292 * @see #QUERY_SORT_DIRECTION_ASCENDING 293 * @see #QUERY_SORT_DIRECTION_DESCENDING 294 */ 295 public static final String QUERY_ARG_SORT_DIRECTION = "android:query-arg-sort-direction"; 296 297 /** 298 * Allows client to specify a hint to the provider declaring which collation 299 * to use when sorting text values. 300 * 301 * <p>Providers may support custom collators. When specifying a custom collator 302 * the value is determined by the Provider. 303 * 304 * <li>{@link ContentProvider} implementations: When preparing data in 305 * {@link ContentProvider#query(Uri, String[], Bundle, CancellationSignal)}, if sort collation 306 * is reflected in the returned Cursor, it is strongly recommended that 307 * {@link #QUERY_ARG_SORT_COLLATION} then be included in the array of honored arguments 308 * reflected in {@link Cursor} extras {@link Bundle} under {@link #EXTRA_HONORED_ARGS}. 309 * 310 * <li>When querying a provider, where no QUERY_ARG_SQL* otherwise exists in the 311 * arguments {@link Bundle}, the Content framework will attempt to synthesize 312 * a QUERY_ARG_SQL* argument using the corresponding QUERY_ARG_SORT* values. 313 * 314 * @see java.text.Collator#PRIMARY 315 * @see java.text.Collator#SECONDARY 316 * @see java.text.Collator#TERTIARY 317 * @see java.text.Collator#IDENTICAL 318 */ 319 public static final String QUERY_ARG_SORT_COLLATION = "android:query-arg-sort-collation"; 320 321 /** 322 * Allows provider to report back to client which query keys are honored in a Cursor. 323 * 324 * <p>Key identifying a {@code String[]} containing all QUERY_ARG_SORT* arguments 325 * honored by the provider. Include this in {@link Cursor} extras {@link Bundle} 326 * when any QUERY_ARG_SORT* value was honored during the preparation of the 327 * results {@link Cursor}. 328 * 329 * <p>If present, ALL honored arguments are enumerated in this extra’s payload. 330 * 331 * @see #QUERY_ARG_SORT_COLUMNS 332 * @see #QUERY_ARG_SORT_DIRECTION 333 * @see #QUERY_ARG_SORT_COLLATION 334 */ 335 public static final String EXTRA_HONORED_ARGS = "android.content.extra.HONORED_ARGS"; 336 337 /** @hide */ 338 @IntDef(flag = false, value = { 339 QUERY_SORT_DIRECTION_ASCENDING, 340 QUERY_SORT_DIRECTION_DESCENDING 341 }) 342 @Retention(RetentionPolicy.SOURCE) 343 public @interface SortDirection {} 344 public static final int QUERY_SORT_DIRECTION_ASCENDING = 0; 345 public static final int QUERY_SORT_DIRECTION_DESCENDING = 1; 346 347 /** 348 * @see {@link java.text.Collector} for details on respective collation strength. 349 * @hide 350 */ 351 @IntDef(flag = false, value = { 352 java.text.Collator.PRIMARY, 353 java.text.Collator.SECONDARY, 354 java.text.Collator.TERTIARY, 355 java.text.Collator.IDENTICAL 356 }) 357 @Retention(RetentionPolicy.SOURCE) 358 public @interface QueryCollator {} 359 360 /** 361 * Specifies the offset row index within a Cursor. 362 */ 363 public static final String QUERY_ARG_OFFSET = "android:query-arg-offset"; 364 365 /** 366 * Specifies the max number of rows to include in a Cursor. 367 */ 368 public static final String QUERY_ARG_LIMIT = "android:query-arg-limit"; 369 370 /** 371 * Added to {@link Cursor} extras {@link Bundle} to indicate total row count of 372 * recordset when paging is supported. Providers must include this when 373 * implementing paging support. 374 * 375 * <p>A provider may return -1 that row count of the recordset is unknown. 376 * 377 * <p>Providers having returned -1 in a previous query are recommended to 378 * send content change notification once (if) full recordset size becomes 379 * known. 380 */ 381 public static final String EXTRA_TOTAL_COUNT = "android.content.extra.TOTAL_COUNT"; 382 383 /** 384 * This is the Android platform's base MIME type for a content: URI 385 * containing a Cursor of a single item. Applications should use this 386 * as the base type along with their own sub-type of their content: URIs 387 * that represent a particular item. For example, hypothetical IMAP email 388 * client may have a URI 389 * <code>content://com.company.provider.imap/inbox/1</code> for a particular 390 * message in the inbox, whose MIME type would be reported as 391 * <code>CURSOR_ITEM_BASE_TYPE + "/vnd.company.imap-msg"</code> 392 * 393 * <p>Compare with {@link #CURSOR_DIR_BASE_TYPE}. 394 */ 395 public static final String CURSOR_ITEM_BASE_TYPE = "vnd.android.cursor.item"; 396 397 /** 398 * This is the Android platform's base MIME type for a content: URI 399 * containing a Cursor of zero or more items. Applications should use this 400 * as the base type along with their own sub-type of their content: URIs 401 * that represent a directory of items. For example, hypothetical IMAP email 402 * client may have a URI 403 * <code>content://com.company.provider.imap/inbox</code> for all of the 404 * messages in its inbox, whose MIME type would be reported as 405 * <code>CURSOR_DIR_BASE_TYPE + "/vnd.company.imap-msg"</code> 406 * 407 * <p>Note how the base MIME type varies between this and 408 * {@link #CURSOR_ITEM_BASE_TYPE} depending on whether there is 409 * one single item or multiple items in the data set, while the sub-type 410 * remains the same because in either case the data structure contained 411 * in the cursor is the same. 412 */ 413 public static final String CURSOR_DIR_BASE_TYPE = "vnd.android.cursor.dir"; 414 415 /** 416 * This is the Android platform's generic MIME type to match any MIME 417 * type of the form "{@link #CURSOR_ITEM_BASE_TYPE}/{@code SUB_TYPE}". 418 * {@code SUB_TYPE} is the sub-type of the application-dependent 419 * content, e.g., "audio", "video", "playlist". 420 */ 421 public static final String ANY_CURSOR_ITEM_TYPE = "vnd.android.cursor.item/*"; 422 423 /** @hide */ 424 public static final int SYNC_ERROR_SYNC_ALREADY_IN_PROGRESS = 1; 425 /** @hide */ 426 public static final int SYNC_ERROR_AUTHENTICATION = 2; 427 /** @hide */ 428 public static final int SYNC_ERROR_IO = 3; 429 /** @hide */ 430 public static final int SYNC_ERROR_PARSE = 4; 431 /** @hide */ 432 public static final int SYNC_ERROR_CONFLICT = 5; 433 /** @hide */ 434 public static final int SYNC_ERROR_TOO_MANY_DELETIONS = 6; 435 /** @hide */ 436 public static final int SYNC_ERROR_TOO_MANY_RETRIES = 7; 437 /** @hide */ 438 public static final int SYNC_ERROR_INTERNAL = 8; 439 440 private static final String[] SYNC_ERROR_NAMES = new String[] { 441 "already-in-progress", 442 "authentication-error", 443 "io-error", 444 "parse-error", 445 "conflict", 446 "too-many-deletions", 447 "too-many-retries", 448 "internal-error", 449 }; 450 451 /** @hide */ syncErrorToString(int error)452 public static String syncErrorToString(int error) { 453 if (error < 1 || error > SYNC_ERROR_NAMES.length) { 454 return String.valueOf(error); 455 } 456 return SYNC_ERROR_NAMES[error - 1]; 457 } 458 459 /** @hide */ syncErrorStringToInt(String error)460 public static int syncErrorStringToInt(String error) { 461 for (int i = 0, n = SYNC_ERROR_NAMES.length; i < n; i++) { 462 if (SYNC_ERROR_NAMES[i].equals(error)) { 463 return i + 1; 464 } 465 } 466 if (error != null) { 467 try { 468 return Integer.parseInt(error); 469 } catch (NumberFormatException e) { 470 Log.d(TAG, "error parsing sync error: " + error); 471 } 472 } 473 return 0; 474 } 475 476 public static final int SYNC_OBSERVER_TYPE_SETTINGS = 1<<0; 477 public static final int SYNC_OBSERVER_TYPE_PENDING = 1<<1; 478 public static final int SYNC_OBSERVER_TYPE_ACTIVE = 1<<2; 479 /** @hide */ 480 public static final int SYNC_OBSERVER_TYPE_STATUS = 1<<3; 481 /** @hide */ 482 public static final int SYNC_OBSERVER_TYPE_ALL = 0x7fffffff; 483 484 /** @hide */ 485 @IntDef(flag = true, 486 value = { 487 NOTIFY_SYNC_TO_NETWORK, 488 NOTIFY_SKIP_NOTIFY_FOR_DESCENDANTS 489 }) 490 @Retention(RetentionPolicy.SOURCE) 491 public @interface NotifyFlags {} 492 493 /** 494 * Flag for {@link #notifyChange(Uri, ContentObserver, int)}: attempt to sync the change 495 * to the network. 496 */ 497 public static final int NOTIFY_SYNC_TO_NETWORK = 1<<0; 498 499 /** 500 * Flag for {@link #notifyChange(Uri, ContentObserver, int)}: if set, this notification 501 * will be skipped if it is being delivered to the root URI of a ContentObserver that is 502 * using "notify for descendants." The purpose of this is to allow the provide to send 503 * a general notification of "something under X" changed that observers of that specific 504 * URI can receive, while also sending a specific URI under X. It would use this flag 505 * when sending the former, so that observers of "X and descendants" only see the latter. 506 */ 507 public static final int NOTIFY_SKIP_NOTIFY_FOR_DESCENDANTS = 1<<1; 508 509 // Always log queries which take 500ms+; shorter queries are 510 // sampled accordingly. 511 private static final boolean ENABLE_CONTENT_SAMPLE = false; 512 private static final int SLOW_THRESHOLD_MILLIS = 500; 513 private final Random mRandom = new Random(); // guarded by itself 514 ContentResolver(Context context)515 public ContentResolver(Context context) { 516 mContext = context != null ? context : ActivityThread.currentApplication(); 517 mPackageName = mContext.getOpPackageName(); 518 mTargetSdkVersion = mContext.getApplicationInfo().targetSdkVersion; 519 } 520 521 /** @hide */ acquireProvider(Context c, String name)522 protected abstract IContentProvider acquireProvider(Context c, String name); 523 524 /** 525 * Providing a default implementation of this, to avoid having to change a 526 * lot of other things, but implementations of ContentResolver should 527 * implement it. 528 * 529 * @hide 530 */ acquireExistingProvider(Context c, String name)531 protected IContentProvider acquireExistingProvider(Context c, String name) { 532 return acquireProvider(c, name); 533 } 534 535 /** @hide */ releaseProvider(IContentProvider icp)536 public abstract boolean releaseProvider(IContentProvider icp); 537 /** @hide */ acquireUnstableProvider(Context c, String name)538 protected abstract IContentProvider acquireUnstableProvider(Context c, String name); 539 /** @hide */ releaseUnstableProvider(IContentProvider icp)540 public abstract boolean releaseUnstableProvider(IContentProvider icp); 541 /** @hide */ unstableProviderDied(IContentProvider icp)542 public abstract void unstableProviderDied(IContentProvider icp); 543 544 /** @hide */ appNotRespondingViaProvider(IContentProvider icp)545 public void appNotRespondingViaProvider(IContentProvider icp) { 546 throw new UnsupportedOperationException("appNotRespondingViaProvider"); 547 } 548 549 /** 550 * Return the MIME type of the given content URL. 551 * 552 * @param url A Uri identifying content (either a list or specific type), 553 * using the content:// scheme. 554 * @return A MIME type for the content, or null if the URL is invalid or the type is unknown 555 */ getType(@onNull Uri url)556 public final @Nullable String getType(@NonNull Uri url) { 557 Preconditions.checkNotNull(url, "url"); 558 559 // XXX would like to have an acquireExistingUnstableProvider for this. 560 IContentProvider provider = acquireExistingProvider(url); 561 if (provider != null) { 562 try { 563 return provider.getType(url); 564 } catch (RemoteException e) { 565 return null; 566 } catch (java.lang.Exception e) { 567 Log.w(TAG, "Failed to get type for: " + url + " (" + e.getMessage() + ")"); 568 return null; 569 } finally { 570 releaseProvider(provider); 571 } 572 } 573 574 if (!SCHEME_CONTENT.equals(url.getScheme())) { 575 return null; 576 } 577 578 try { 579 String type = ActivityManager.getService().getProviderMimeType( 580 ContentProvider.getUriWithoutUserId(url), resolveUserId(url)); 581 return type; 582 } catch (RemoteException e) { 583 // Arbitrary and not worth documenting, as Activity 584 // Manager will kill this process shortly anyway. 585 return null; 586 } catch (java.lang.Exception e) { 587 Log.w(TAG, "Failed to get type for: " + url + " (" + e.getMessage() + ")"); 588 return null; 589 } 590 } 591 592 /** 593 * Query for the possible MIME types for the representations the given 594 * content URL can be returned when opened as as stream with 595 * {@link #openTypedAssetFileDescriptor}. Note that the types here are 596 * not necessarily a superset of the type returned by {@link #getType} -- 597 * many content providers cannot return a raw stream for the structured 598 * data that they contain. 599 * 600 * @param url A Uri identifying content (either a list or specific type), 601 * using the content:// scheme. 602 * @param mimeTypeFilter The desired MIME type. This may be a pattern, 603 * such as */*, to query for all available MIME types that match the 604 * pattern. 605 * @return Returns an array of MIME type strings for all available 606 * data streams that match the given mimeTypeFilter. If there are none, 607 * null is returned. 608 */ getStreamTypes(@onNull Uri url, @NonNull String mimeTypeFilter)609 public @Nullable String[] getStreamTypes(@NonNull Uri url, @NonNull String mimeTypeFilter) { 610 Preconditions.checkNotNull(url, "url"); 611 Preconditions.checkNotNull(mimeTypeFilter, "mimeTypeFilter"); 612 613 IContentProvider provider = acquireProvider(url); 614 if (provider == null) { 615 return null; 616 } 617 618 try { 619 return provider.getStreamTypes(url, mimeTypeFilter); 620 } catch (RemoteException e) { 621 // Arbitrary and not worth documenting, as Activity 622 // Manager will kill this process shortly anyway. 623 return null; 624 } finally { 625 releaseProvider(provider); 626 } 627 } 628 629 /** 630 * Query the given URI, returning a {@link Cursor} over the result set. 631 * <p> 632 * For best performance, the caller should follow these guidelines: 633 * <ul> 634 * <li>Provide an explicit projection, to prevent 635 * reading data from storage that aren't going to be used.</li> 636 * <li>Use question mark parameter markers such as 'phone=?' instead of 637 * explicit values in the {@code selection} parameter, so that queries 638 * that differ only by those values will be recognized as the same 639 * for caching purposes.</li> 640 * </ul> 641 * </p> 642 * 643 * @param uri The URI, using the content:// scheme, for the content to 644 * retrieve. 645 * @param projection A list of which columns to return. Passing null will 646 * return all columns, which is inefficient. 647 * @param selection A filter declaring which rows to return, formatted as an 648 * SQL WHERE clause (excluding the WHERE itself). Passing null will 649 * return all rows for the given URI. 650 * @param selectionArgs You may include ?s in selection, which will be 651 * replaced by the values from selectionArgs, in the order that they 652 * appear in the selection. The values will be bound as Strings. 653 * @param sortOrder How to order the rows, formatted as an SQL ORDER BY 654 * clause (excluding the ORDER BY itself). Passing null will use the 655 * default sort order, which may be unordered. 656 * @return A Cursor object, which is positioned before the first entry, or null 657 * @see Cursor 658 */ query(@equiresPermission.Read @onNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder)659 public final @Nullable Cursor query(@RequiresPermission.Read @NonNull Uri uri, 660 @Nullable String[] projection, @Nullable String selection, 661 @Nullable String[] selectionArgs, @Nullable String sortOrder) { 662 return query(uri, projection, selection, selectionArgs, sortOrder, null); 663 } 664 665 /** 666 * Query the given URI, returning a {@link Cursor} over the result set 667 * with optional support for cancellation. 668 * <p> 669 * For best performance, the caller should follow these guidelines: 670 * <ul> 671 * <li>Provide an explicit projection, to prevent 672 * reading data from storage that aren't going to be used.</li> 673 * <li>Use question mark parameter markers such as 'phone=?' instead of 674 * explicit values in the {@code selection} parameter, so that queries 675 * that differ only by those values will be recognized as the same 676 * for caching purposes.</li> 677 * </ul> 678 * </p> 679 * 680 * @param uri The URI, using the content:// scheme, for the content to 681 * retrieve. 682 * @param projection A list of which columns to return. Passing null will 683 * return all columns, which is inefficient. 684 * @param selection A filter declaring which rows to return, formatted as an 685 * SQL WHERE clause (excluding the WHERE itself). Passing null will 686 * return all rows for the given URI. 687 * @param selectionArgs You may include ?s in selection, which will be 688 * replaced by the values from selectionArgs, in the order that they 689 * appear in the selection. The values will be bound as Strings. 690 * @param sortOrder How to order the rows, formatted as an SQL ORDER BY 691 * clause (excluding the ORDER BY itself). Passing null will use the 692 * default sort order, which may be unordered. 693 * @param cancellationSignal A signal to cancel the operation in progress, or null if none. 694 * If the operation is canceled, then {@link OperationCanceledException} will be thrown 695 * when the query is executed. 696 * @return A Cursor object, which is positioned before the first entry, or null 697 * @see Cursor 698 */ query(@equiresPermission.Read @onNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder, @Nullable CancellationSignal cancellationSignal)699 public final @Nullable Cursor query(@RequiresPermission.Read @NonNull Uri uri, 700 @Nullable String[] projection, @Nullable String selection, 701 @Nullable String[] selectionArgs, @Nullable String sortOrder, 702 @Nullable CancellationSignal cancellationSignal) { 703 Bundle queryArgs = createSqlQueryBundle(selection, selectionArgs, sortOrder); 704 return query(uri, projection, queryArgs, cancellationSignal); 705 } 706 707 /** 708 * Query the given URI, returning a {@link Cursor} over the result set 709 * with support for cancellation. 710 * 711 * <p>For best performance, the caller should follow these guidelines: 712 * 713 * <li>Provide an explicit projection, to prevent reading data from storage 714 * that aren't going to be used. 715 * 716 * Provider must identify which QUERY_ARG_SORT* arguments were honored during 717 * the preparation of the result set by including the respective argument keys 718 * in the {@link Cursor} extras {@link Bundle}. See {@link #EXTRA_HONORED_ARGS} 719 * for details. 720 * 721 * @see #QUERY_ARG_SORT_COLUMNS, #QUERY_ARG_SORT_DIRECTION, #QUERY_ARG_SORT_COLLATION. 722 * 723 * @param uri The URI, using the content:// scheme, for the content to 724 * retrieve. 725 * @param projection A list of which columns to return. Passing null will 726 * return all columns, which is inefficient. 727 * @param queryArgs A Bundle containing any arguments to the query. 728 * @param cancellationSignal A signal to cancel the operation in progress, or null if none. 729 * If the operation is canceled, then {@link OperationCanceledException} will be thrown 730 * when the query is executed. 731 * @return A Cursor object, which is positioned before the first entry, or null 732 * @see Cursor 733 */ query(final @RequiresPermission.Read @NonNull Uri uri, @Nullable String[] projection, @Nullable Bundle queryArgs, @Nullable CancellationSignal cancellationSignal)734 public final @Nullable Cursor query(final @RequiresPermission.Read @NonNull Uri uri, 735 @Nullable String[] projection, @Nullable Bundle queryArgs, 736 @Nullable CancellationSignal cancellationSignal) { 737 Preconditions.checkNotNull(uri, "uri"); 738 IContentProvider unstableProvider = acquireUnstableProvider(uri); 739 if (unstableProvider == null) { 740 return null; 741 } 742 IContentProvider stableProvider = null; 743 Cursor qCursor = null; 744 try { 745 long startTime = SystemClock.uptimeMillis(); 746 747 ICancellationSignal remoteCancellationSignal = null; 748 if (cancellationSignal != null) { 749 cancellationSignal.throwIfCanceled(); 750 remoteCancellationSignal = unstableProvider.createCancellationSignal(); 751 cancellationSignal.setRemote(remoteCancellationSignal); 752 } 753 try { 754 qCursor = unstableProvider.query(mPackageName, uri, projection, 755 queryArgs, remoteCancellationSignal); 756 } catch (DeadObjectException e) { 757 // The remote process has died... but we only hold an unstable 758 // reference though, so we might recover!!! Let's try!!!! 759 // This is exciting!!1!!1!!!!1 760 unstableProviderDied(unstableProvider); 761 stableProvider = acquireProvider(uri); 762 if (stableProvider == null) { 763 return null; 764 } 765 qCursor = stableProvider.query( 766 mPackageName, uri, projection, queryArgs, remoteCancellationSignal); 767 } 768 if (qCursor == null) { 769 return null; 770 } 771 772 // Force query execution. Might fail and throw a runtime exception here. 773 qCursor.getCount(); 774 long durationMillis = SystemClock.uptimeMillis() - startTime; 775 maybeLogQueryToEventLog(durationMillis, uri, projection, queryArgs); 776 777 // Wrap the cursor object into CursorWrapperInner object. 778 final IContentProvider provider = (stableProvider != null) ? stableProvider 779 : acquireProvider(uri); 780 final CursorWrapperInner wrapper = new CursorWrapperInner(qCursor, provider); 781 stableProvider = null; 782 qCursor = null; 783 return wrapper; 784 } catch (RemoteException e) { 785 // Arbitrary and not worth documenting, as Activity 786 // Manager will kill this process shortly anyway. 787 return null; 788 } finally { 789 if (qCursor != null) { 790 qCursor.close(); 791 } 792 if (cancellationSignal != null) { 793 cancellationSignal.setRemote(null); 794 } 795 if (unstableProvider != null) { 796 releaseUnstableProvider(unstableProvider); 797 } 798 if (stableProvider != null) { 799 releaseProvider(stableProvider); 800 } 801 } 802 } 803 804 /** 805 * Transform the given <var>url</var> to a canonical representation of 806 * its referenced resource, which can be used across devices, persisted, 807 * backed up and restored, etc. The returned Uri is still a fully capable 808 * Uri for use with its content provider, allowing you to do all of the 809 * same content provider operations as with the original Uri -- 810 * {@link #query}, {@link #openInputStream(android.net.Uri)}, etc. The 811 * only difference in behavior between the original and new Uris is that 812 * the content provider may need to do some additional work at each call 813 * using it to resolve it to the correct resource, especially if the 814 * canonical Uri has been moved to a different environment. 815 * 816 * <p>If you are moving a canonical Uri between environments, you should 817 * perform another call to {@link #canonicalize} with that original Uri to 818 * re-canonicalize it for the current environment. Alternatively, you may 819 * want to use {@link #uncanonicalize} to transform it to a non-canonical 820 * Uri that works only in the current environment but potentially more 821 * efficiently than the canonical representation.</p> 822 * 823 * @param url The {@link Uri} that is to be transformed to a canonical 824 * representation. Like all resolver calls, the input can be either 825 * a non-canonical or canonical Uri. 826 * 827 * @return Returns the official canonical representation of <var>url</var>, 828 * or null if the content provider does not support a canonical representation 829 * of the given Uri. Many providers may not support canonicalization of some 830 * or all of their Uris. 831 * 832 * @see #uncanonicalize 833 */ canonicalize(@onNull Uri url)834 public final @Nullable Uri canonicalize(@NonNull Uri url) { 835 Preconditions.checkNotNull(url, "url"); 836 IContentProvider provider = acquireProvider(url); 837 if (provider == null) { 838 return null; 839 } 840 841 try { 842 return provider.canonicalize(mPackageName, url); 843 } catch (RemoteException e) { 844 // Arbitrary and not worth documenting, as Activity 845 // Manager will kill this process shortly anyway. 846 return null; 847 } finally { 848 releaseProvider(provider); 849 } 850 } 851 852 /** 853 * Given a canonical Uri previously generated by {@link #canonicalize}, convert 854 * it to its local non-canonical form. This can be useful in some cases where 855 * you know that you will only be using the Uri in the current environment and 856 * want to avoid any possible overhead when using it with the content 857 * provider or want to verify that the referenced data exists at all in the 858 * new environment. 859 * 860 * @param url The canonical {@link Uri} that is to be convered back to its 861 * non-canonical form. 862 * 863 * @return Returns the non-canonical representation of <var>url</var>. This will 864 * return null if data identified by the canonical Uri can not be found in 865 * the current environment; callers must always check for null and deal with 866 * that by appropriately falling back to an alternative. 867 * 868 * @see #canonicalize 869 */ uncanonicalize(@onNull Uri url)870 public final @Nullable Uri uncanonicalize(@NonNull Uri url) { 871 Preconditions.checkNotNull(url, "url"); 872 IContentProvider provider = acquireProvider(url); 873 if (provider == null) { 874 return null; 875 } 876 877 try { 878 return provider.uncanonicalize(mPackageName, url); 879 } catch (RemoteException e) { 880 // Arbitrary and not worth documenting, as Activity 881 // Manager will kill this process shortly anyway. 882 return null; 883 } finally { 884 releaseProvider(provider); 885 } 886 } 887 888 /** 889 * This allows clients to request an explicit refresh of content identified by {@code uri}. 890 * <p> 891 * Client code should only invoke this method when there is a strong indication (such as a user 892 * initiated pull to refresh gesture) that the content is stale. 893 * <p> 894 * 895 * @param url The Uri identifying the data to refresh. 896 * @param args Additional options from the client. The definitions of these are specific to the 897 * content provider being called. 898 * @param cancellationSignal A signal to cancel the operation in progress, or {@code null} if 899 * none. For example, if you called refresh on a particular uri, you should call 900 * {@link CancellationSignal#throwIfCanceled()} to check whether the client has 901 * canceled the refresh request. 902 * @return true if the provider actually tried refreshing. 903 */ refresh(@onNull Uri url, @Nullable Bundle args, @Nullable CancellationSignal cancellationSignal)904 public final boolean refresh(@NonNull Uri url, @Nullable Bundle args, 905 @Nullable CancellationSignal cancellationSignal) { 906 Preconditions.checkNotNull(url, "url"); 907 IContentProvider provider = acquireProvider(url); 908 if (provider == null) { 909 return false; 910 } 911 912 try { 913 ICancellationSignal remoteCancellationSignal = null; 914 if (cancellationSignal != null) { 915 cancellationSignal.throwIfCanceled(); 916 remoteCancellationSignal = provider.createCancellationSignal(); 917 cancellationSignal.setRemote(remoteCancellationSignal); 918 } 919 return provider.refresh(mPackageName, url, args, remoteCancellationSignal); 920 } catch (RemoteException e) { 921 // Arbitrary and not worth documenting, as Activity 922 // Manager will kill this process shortly anyway. 923 return false; 924 } finally { 925 releaseProvider(provider); 926 } 927 } 928 929 /** 930 * Open a stream on to the content associated with a content URI. If there 931 * is no data associated with the URI, FileNotFoundException is thrown. 932 * 933 * <h5>Accepts the following URI schemes:</h5> 934 * <ul> 935 * <li>content ({@link #SCHEME_CONTENT})</li> 936 * <li>android.resource ({@link #SCHEME_ANDROID_RESOURCE})</li> 937 * <li>file ({@link #SCHEME_FILE})</li> 938 * </ul> 939 * 940 * <p>See {@link #openAssetFileDescriptor(Uri, String)} for more information 941 * on these schemes. 942 * 943 * @param uri The desired URI. 944 * @return InputStream 945 * @throws FileNotFoundException if the provided URI could not be opened. 946 * @see #openAssetFileDescriptor(Uri, String) 947 */ openInputStream(@onNull Uri uri)948 public final @Nullable InputStream openInputStream(@NonNull Uri uri) 949 throws FileNotFoundException { 950 Preconditions.checkNotNull(uri, "uri"); 951 String scheme = uri.getScheme(); 952 if (SCHEME_ANDROID_RESOURCE.equals(scheme)) { 953 // Note: left here to avoid breaking compatibility. May be removed 954 // with sufficient testing. 955 OpenResourceIdResult r = getResourceId(uri); 956 try { 957 InputStream stream = r.r.openRawResource(r.id); 958 return stream; 959 } catch (Resources.NotFoundException ex) { 960 throw new FileNotFoundException("Resource does not exist: " + uri); 961 } 962 } else if (SCHEME_FILE.equals(scheme)) { 963 // Note: left here to avoid breaking compatibility. May be removed 964 // with sufficient testing. 965 return new FileInputStream(uri.getPath()); 966 } else { 967 AssetFileDescriptor fd = openAssetFileDescriptor(uri, "r", null); 968 try { 969 return fd != null ? fd.createInputStream() : null; 970 } catch (IOException e) { 971 throw new FileNotFoundException("Unable to create stream"); 972 } 973 } 974 } 975 976 /** 977 * Synonym for {@link #openOutputStream(Uri, String) 978 * openOutputStream(uri, "w")}. 979 * @throws FileNotFoundException if the provided URI could not be opened. 980 */ openOutputStream(@onNull Uri uri)981 public final @Nullable OutputStream openOutputStream(@NonNull Uri uri) 982 throws FileNotFoundException { 983 return openOutputStream(uri, "w"); 984 } 985 986 /** 987 * Open a stream on to the content associated with a content URI. If there 988 * is no data associated with the URI, FileNotFoundException is thrown. 989 * 990 * <h5>Accepts the following URI schemes:</h5> 991 * <ul> 992 * <li>content ({@link #SCHEME_CONTENT})</li> 993 * <li>file ({@link #SCHEME_FILE})</li> 994 * </ul> 995 * 996 * <p>See {@link #openAssetFileDescriptor(Uri, String)} for more information 997 * on these schemes. 998 * 999 * @param uri The desired URI. 1000 * @param mode May be "w", "wa", "rw", or "rwt". 1001 * @return OutputStream 1002 * @throws FileNotFoundException if the provided URI could not be opened. 1003 * @see #openAssetFileDescriptor(Uri, String) 1004 */ openOutputStream(@onNull Uri uri, @NonNull String mode)1005 public final @Nullable OutputStream openOutputStream(@NonNull Uri uri, @NonNull String mode) 1006 throws FileNotFoundException { 1007 AssetFileDescriptor fd = openAssetFileDescriptor(uri, mode, null); 1008 try { 1009 return fd != null ? fd.createOutputStream() : null; 1010 } catch (IOException e) { 1011 throw new FileNotFoundException("Unable to create stream"); 1012 } 1013 } 1014 1015 /** 1016 * Open a raw file descriptor to access data under a URI. This 1017 * is like {@link #openAssetFileDescriptor(Uri, String)}, but uses the 1018 * underlying {@link ContentProvider#openFile} 1019 * ContentProvider.openFile()} method, so will <em>not</em> work with 1020 * providers that return sub-sections of files. If at all possible, 1021 * you should use {@link #openAssetFileDescriptor(Uri, String)}. You 1022 * will receive a FileNotFoundException exception if the provider returns a 1023 * sub-section of a file. 1024 * 1025 * <h5>Accepts the following URI schemes:</h5> 1026 * <ul> 1027 * <li>content ({@link #SCHEME_CONTENT})</li> 1028 * <li>file ({@link #SCHEME_FILE})</li> 1029 * </ul> 1030 * 1031 * <p>See {@link #openAssetFileDescriptor(Uri, String)} for more information 1032 * on these schemes. 1033 * <p> 1034 * If opening with the exclusive "r" or "w" modes, the returned 1035 * ParcelFileDescriptor could be a pipe or socket pair to enable streaming 1036 * of data. Opening with the "rw" mode implies a file on disk that supports 1037 * seeking. If possible, always use an exclusive mode to give the underlying 1038 * {@link ContentProvider} the most flexibility. 1039 * <p> 1040 * If you are writing a file, and need to communicate an error to the 1041 * provider, use {@link ParcelFileDescriptor#closeWithError(String)}. 1042 * 1043 * @param uri The desired URI to open. 1044 * @param mode The file mode to use, as per {@link ContentProvider#openFile 1045 * ContentProvider.openFile}. 1046 * @return Returns a new ParcelFileDescriptor pointing to the file. You 1047 * own this descriptor and are responsible for closing it when done. 1048 * @throws FileNotFoundException Throws FileNotFoundException if no 1049 * file exists under the URI or the mode is invalid. 1050 * @see #openAssetFileDescriptor(Uri, String) 1051 */ openFileDescriptor(@onNull Uri uri, @NonNull String mode)1052 public final @Nullable ParcelFileDescriptor openFileDescriptor(@NonNull Uri uri, 1053 @NonNull String mode) throws FileNotFoundException { 1054 return openFileDescriptor(uri, mode, null); 1055 } 1056 1057 /** 1058 * Open a raw file descriptor to access data under a URI. This 1059 * is like {@link #openAssetFileDescriptor(Uri, String)}, but uses the 1060 * underlying {@link ContentProvider#openFile} 1061 * ContentProvider.openFile()} method, so will <em>not</em> work with 1062 * providers that return sub-sections of files. If at all possible, 1063 * you should use {@link #openAssetFileDescriptor(Uri, String)}. You 1064 * will receive a FileNotFoundException exception if the provider returns a 1065 * sub-section of a file. 1066 * 1067 * <h5>Accepts the following URI schemes:</h5> 1068 * <ul> 1069 * <li>content ({@link #SCHEME_CONTENT})</li> 1070 * <li>file ({@link #SCHEME_FILE})</li> 1071 * </ul> 1072 * 1073 * <p>See {@link #openAssetFileDescriptor(Uri, String)} for more information 1074 * on these schemes. 1075 * <p> 1076 * If opening with the exclusive "r" or "w" modes, the returned 1077 * ParcelFileDescriptor could be a pipe or socket pair to enable streaming 1078 * of data. Opening with the "rw" mode implies a file on disk that supports 1079 * seeking. If possible, always use an exclusive mode to give the underlying 1080 * {@link ContentProvider} the most flexibility. 1081 * <p> 1082 * If you are writing a file, and need to communicate an error to the 1083 * provider, use {@link ParcelFileDescriptor#closeWithError(String)}. 1084 * 1085 * @param uri The desired URI to open. 1086 * @param mode The file mode to use, as per {@link ContentProvider#openFile 1087 * ContentProvider.openFile}. 1088 * @param cancellationSignal A signal to cancel the operation in progress, 1089 * or null if none. If the operation is canceled, then 1090 * {@link OperationCanceledException} will be thrown. 1091 * @return Returns a new ParcelFileDescriptor pointing to the file. You 1092 * own this descriptor and are responsible for closing it when done. 1093 * @throws FileNotFoundException Throws FileNotFoundException if no 1094 * file exists under the URI or the mode is invalid. 1095 * @see #openAssetFileDescriptor(Uri, String) 1096 */ openFileDescriptor(@onNull Uri uri, @NonNull String mode, @Nullable CancellationSignal cancellationSignal)1097 public final @Nullable ParcelFileDescriptor openFileDescriptor(@NonNull Uri uri, 1098 @NonNull String mode, @Nullable CancellationSignal cancellationSignal) 1099 throws FileNotFoundException { 1100 AssetFileDescriptor afd = openAssetFileDescriptor(uri, mode, cancellationSignal); 1101 if (afd == null) { 1102 return null; 1103 } 1104 1105 if (afd.getDeclaredLength() < 0) { 1106 // This is a full file! 1107 return afd.getParcelFileDescriptor(); 1108 } 1109 1110 // Client can't handle a sub-section of a file, so close what 1111 // we got and bail with an exception. 1112 try { 1113 afd.close(); 1114 } catch (IOException e) { 1115 } 1116 1117 throw new FileNotFoundException("Not a whole file"); 1118 } 1119 1120 /** 1121 * Open a raw file descriptor to access data under a URI. This 1122 * interacts with the underlying {@link ContentProvider#openAssetFile} 1123 * method of the provider associated with the given URI, to retrieve any file stored there. 1124 * 1125 * <h5>Accepts the following URI schemes:</h5> 1126 * <ul> 1127 * <li>content ({@link #SCHEME_CONTENT})</li> 1128 * <li>android.resource ({@link #SCHEME_ANDROID_RESOURCE})</li> 1129 * <li>file ({@link #SCHEME_FILE})</li> 1130 * </ul> 1131 * <h5>The android.resource ({@link #SCHEME_ANDROID_RESOURCE}) Scheme</h5> 1132 * <p> 1133 * A Uri object can be used to reference a resource in an APK file. The 1134 * Uri should be one of the following formats: 1135 * <ul> 1136 * <li><code>android.resource://package_name/id_number</code><br/> 1137 * <code>package_name</code> is your package name as listed in your AndroidManifest.xml. 1138 * For example <code>com.example.myapp</code><br/> 1139 * <code>id_number</code> is the int form of the ID.<br/> 1140 * The easiest way to construct this form is 1141 * <pre>Uri uri = Uri.parse("android.resource://com.example.myapp/" + R.raw.my_resource");</pre> 1142 * </li> 1143 * <li><code>android.resource://package_name/type/name</code><br/> 1144 * <code>package_name</code> is your package name as listed in your AndroidManifest.xml. 1145 * For example <code>com.example.myapp</code><br/> 1146 * <code>type</code> is the string form of the resource type. For example, <code>raw</code> 1147 * or <code>drawable</code>. 1148 * <code>name</code> is the string form of the resource name. That is, whatever the file 1149 * name was in your res directory, without the type extension. 1150 * The easiest way to construct this form is 1151 * <pre>Uri uri = Uri.parse("android.resource://com.example.myapp/raw/my_resource");</pre> 1152 * </li> 1153 * </ul> 1154 * 1155 * <p>Note that if this function is called for read-only input (mode is "r") 1156 * on a content: URI, it will instead call {@link #openTypedAssetFileDescriptor} 1157 * for you with a MIME type of "*/*". This allows such callers to benefit 1158 * from any built-in data conversion that a provider implements. 1159 * 1160 * @param uri The desired URI to open. 1161 * @param mode The file mode to use, as per {@link ContentProvider#openAssetFile 1162 * ContentProvider.openAssetFile}. 1163 * @return Returns a new ParcelFileDescriptor pointing to the file. You 1164 * own this descriptor and are responsible for closing it when done. 1165 * @throws FileNotFoundException Throws FileNotFoundException of no 1166 * file exists under the URI or the mode is invalid. 1167 */ openAssetFileDescriptor(@onNull Uri uri, @NonNull String mode)1168 public final @Nullable AssetFileDescriptor openAssetFileDescriptor(@NonNull Uri uri, 1169 @NonNull String mode) throws FileNotFoundException { 1170 return openAssetFileDescriptor(uri, mode, null); 1171 } 1172 1173 /** 1174 * Open a raw file descriptor to access data under a URI. This 1175 * interacts with the underlying {@link ContentProvider#openAssetFile} 1176 * method of the provider associated with the given URI, to retrieve any file stored there. 1177 * 1178 * <h5>Accepts the following URI schemes:</h5> 1179 * <ul> 1180 * <li>content ({@link #SCHEME_CONTENT})</li> 1181 * <li>android.resource ({@link #SCHEME_ANDROID_RESOURCE})</li> 1182 * <li>file ({@link #SCHEME_FILE})</li> 1183 * </ul> 1184 * <h5>The android.resource ({@link #SCHEME_ANDROID_RESOURCE}) Scheme</h5> 1185 * <p> 1186 * A Uri object can be used to reference a resource in an APK file. The 1187 * Uri should be one of the following formats: 1188 * <ul> 1189 * <li><code>android.resource://package_name/id_number</code><br/> 1190 * <code>package_name</code> is your package name as listed in your AndroidManifest.xml. 1191 * For example <code>com.example.myapp</code><br/> 1192 * <code>id_number</code> is the int form of the ID.<br/> 1193 * The easiest way to construct this form is 1194 * <pre>Uri uri = Uri.parse("android.resource://com.example.myapp/" + R.raw.my_resource");</pre> 1195 * </li> 1196 * <li><code>android.resource://package_name/type/name</code><br/> 1197 * <code>package_name</code> is your package name as listed in your AndroidManifest.xml. 1198 * For example <code>com.example.myapp</code><br/> 1199 * <code>type</code> is the string form of the resource type. For example, <code>raw</code> 1200 * or <code>drawable</code>. 1201 * <code>name</code> is the string form of the resource name. That is, whatever the file 1202 * name was in your res directory, without the type extension. 1203 * The easiest way to construct this form is 1204 * <pre>Uri uri = Uri.parse("android.resource://com.example.myapp/raw/my_resource");</pre> 1205 * </li> 1206 * </ul> 1207 * 1208 * <p>Note that if this function is called for read-only input (mode is "r") 1209 * on a content: URI, it will instead call {@link #openTypedAssetFileDescriptor} 1210 * for you with a MIME type of "*/*". This allows such callers to benefit 1211 * from any built-in data conversion that a provider implements. 1212 * 1213 * @param uri The desired URI to open. 1214 * @param mode The file mode to use, as per {@link ContentProvider#openAssetFile 1215 * ContentProvider.openAssetFile}. 1216 * @param cancellationSignal A signal to cancel the operation in progress, or null if 1217 * none. If the operation is canceled, then 1218 * {@link OperationCanceledException} will be thrown. 1219 * @return Returns a new ParcelFileDescriptor pointing to the file. You 1220 * own this descriptor and are responsible for closing it when done. 1221 * @throws FileNotFoundException Throws FileNotFoundException of no 1222 * file exists under the URI or the mode is invalid. 1223 */ openAssetFileDescriptor(@onNull Uri uri, @NonNull String mode, @Nullable CancellationSignal cancellationSignal)1224 public final @Nullable AssetFileDescriptor openAssetFileDescriptor(@NonNull Uri uri, 1225 @NonNull String mode, @Nullable CancellationSignal cancellationSignal) 1226 throws FileNotFoundException { 1227 Preconditions.checkNotNull(uri, "uri"); 1228 Preconditions.checkNotNull(mode, "mode"); 1229 1230 String scheme = uri.getScheme(); 1231 if (SCHEME_ANDROID_RESOURCE.equals(scheme)) { 1232 if (!"r".equals(mode)) { 1233 throw new FileNotFoundException("Can't write resources: " + uri); 1234 } 1235 OpenResourceIdResult r = getResourceId(uri); 1236 try { 1237 return r.r.openRawResourceFd(r.id); 1238 } catch (Resources.NotFoundException ex) { 1239 throw new FileNotFoundException("Resource does not exist: " + uri); 1240 } 1241 } else if (SCHEME_FILE.equals(scheme)) { 1242 ParcelFileDescriptor pfd = ParcelFileDescriptor.open( 1243 new File(uri.getPath()), ParcelFileDescriptor.parseMode(mode)); 1244 return new AssetFileDescriptor(pfd, 0, -1); 1245 } else { 1246 if ("r".equals(mode)) { 1247 return openTypedAssetFileDescriptor(uri, "*/*", null, cancellationSignal); 1248 } else { 1249 IContentProvider unstableProvider = acquireUnstableProvider(uri); 1250 if (unstableProvider == null) { 1251 throw new FileNotFoundException("No content provider: " + uri); 1252 } 1253 IContentProvider stableProvider = null; 1254 AssetFileDescriptor fd = null; 1255 1256 try { 1257 ICancellationSignal remoteCancellationSignal = null; 1258 if (cancellationSignal != null) { 1259 cancellationSignal.throwIfCanceled(); 1260 remoteCancellationSignal = unstableProvider.createCancellationSignal(); 1261 cancellationSignal.setRemote(remoteCancellationSignal); 1262 } 1263 1264 try { 1265 fd = unstableProvider.openAssetFile( 1266 mPackageName, uri, mode, remoteCancellationSignal); 1267 if (fd == null) { 1268 // The provider will be released by the finally{} clause 1269 return null; 1270 } 1271 } catch (DeadObjectException e) { 1272 // The remote process has died... but we only hold an unstable 1273 // reference though, so we might recover!!! Let's try!!!! 1274 // This is exciting!!1!!1!!!!1 1275 unstableProviderDied(unstableProvider); 1276 stableProvider = acquireProvider(uri); 1277 if (stableProvider == null) { 1278 throw new FileNotFoundException("No content provider: " + uri); 1279 } 1280 fd = stableProvider.openAssetFile( 1281 mPackageName, uri, mode, remoteCancellationSignal); 1282 if (fd == null) { 1283 // The provider will be released by the finally{} clause 1284 return null; 1285 } 1286 } 1287 1288 if (stableProvider == null) { 1289 stableProvider = acquireProvider(uri); 1290 } 1291 releaseUnstableProvider(unstableProvider); 1292 unstableProvider = null; 1293 ParcelFileDescriptor pfd = new ParcelFileDescriptorInner( 1294 fd.getParcelFileDescriptor(), stableProvider); 1295 1296 // Success! Don't release the provider when exiting, let 1297 // ParcelFileDescriptorInner do that when it is closed. 1298 stableProvider = null; 1299 1300 return new AssetFileDescriptor(pfd, fd.getStartOffset(), 1301 fd.getDeclaredLength()); 1302 1303 } catch (RemoteException e) { 1304 // Whatever, whatever, we'll go away. 1305 throw new FileNotFoundException( 1306 "Failed opening content provider: " + uri); 1307 } catch (FileNotFoundException e) { 1308 throw e; 1309 } finally { 1310 if (cancellationSignal != null) { 1311 cancellationSignal.setRemote(null); 1312 } 1313 if (stableProvider != null) { 1314 releaseProvider(stableProvider); 1315 } 1316 if (unstableProvider != null) { 1317 releaseUnstableProvider(unstableProvider); 1318 } 1319 } 1320 } 1321 } 1322 } 1323 1324 /** 1325 * Open a raw file descriptor to access (potentially type transformed) 1326 * data from a "content:" URI. This interacts with the underlying 1327 * {@link ContentProvider#openTypedAssetFile} method of the provider 1328 * associated with the given URI, to retrieve retrieve any appropriate 1329 * data stream for the data stored there. 1330 * 1331 * <p>Unlike {@link #openAssetFileDescriptor}, this function only works 1332 * with "content:" URIs, because content providers are the only facility 1333 * with an associated MIME type to ensure that the returned data stream 1334 * is of the desired type. 1335 * 1336 * <p>All text/* streams are encoded in UTF-8. 1337 * 1338 * @param uri The desired URI to open. 1339 * @param mimeType The desired MIME type of the returned data. This can 1340 * be a pattern such as */*, which will allow the content provider to 1341 * select a type, though there is no way for you to determine what type 1342 * it is returning. 1343 * @param opts Additional provider-dependent options. 1344 * @return Returns a new ParcelFileDescriptor from which you can read the 1345 * data stream from the provider. Note that this may be a pipe, meaning 1346 * you can't seek in it. The only seek you should do is if the 1347 * AssetFileDescriptor contains an offset, to move to that offset before 1348 * reading. You own this descriptor and are responsible for closing it when done. 1349 * @throws FileNotFoundException Throws FileNotFoundException of no 1350 * data of the desired type exists under the URI. 1351 */ openTypedAssetFileDescriptor(@onNull Uri uri, @NonNull String mimeType, @Nullable Bundle opts)1352 public final @Nullable AssetFileDescriptor openTypedAssetFileDescriptor(@NonNull Uri uri, 1353 @NonNull String mimeType, @Nullable Bundle opts) throws FileNotFoundException { 1354 return openTypedAssetFileDescriptor(uri, mimeType, opts, null); 1355 } 1356 1357 /** 1358 * Open a raw file descriptor to access (potentially type transformed) 1359 * data from a "content:" URI. This interacts with the underlying 1360 * {@link ContentProvider#openTypedAssetFile} method of the provider 1361 * associated with the given URI, to retrieve retrieve any appropriate 1362 * data stream for the data stored there. 1363 * 1364 * <p>Unlike {@link #openAssetFileDescriptor}, this function only works 1365 * with "content:" URIs, because content providers are the only facility 1366 * with an associated MIME type to ensure that the returned data stream 1367 * is of the desired type. 1368 * 1369 * <p>All text/* streams are encoded in UTF-8. 1370 * 1371 * @param uri The desired URI to open. 1372 * @param mimeType The desired MIME type of the returned data. This can 1373 * be a pattern such as */*, which will allow the content provider to 1374 * select a type, though there is no way for you to determine what type 1375 * it is returning. 1376 * @param opts Additional provider-dependent options. 1377 * @param cancellationSignal A signal to cancel the operation in progress, 1378 * or null if none. If the operation is canceled, then 1379 * {@link OperationCanceledException} will be thrown. 1380 * @return Returns a new ParcelFileDescriptor from which you can read the 1381 * data stream from the provider. Note that this may be a pipe, meaning 1382 * you can't seek in it. The only seek you should do is if the 1383 * AssetFileDescriptor contains an offset, to move to that offset before 1384 * reading. You own this descriptor and are responsible for closing it when done. 1385 * @throws FileNotFoundException Throws FileNotFoundException of no 1386 * data of the desired type exists under the URI. 1387 */ openTypedAssetFileDescriptor(@onNull Uri uri, @NonNull String mimeType, @Nullable Bundle opts, @Nullable CancellationSignal cancellationSignal)1388 public final @Nullable AssetFileDescriptor openTypedAssetFileDescriptor(@NonNull Uri uri, 1389 @NonNull String mimeType, @Nullable Bundle opts, 1390 @Nullable CancellationSignal cancellationSignal) throws FileNotFoundException { 1391 Preconditions.checkNotNull(uri, "uri"); 1392 Preconditions.checkNotNull(mimeType, "mimeType"); 1393 1394 IContentProvider unstableProvider = acquireUnstableProvider(uri); 1395 if (unstableProvider == null) { 1396 throw new FileNotFoundException("No content provider: " + uri); 1397 } 1398 IContentProvider stableProvider = null; 1399 AssetFileDescriptor fd = null; 1400 1401 try { 1402 ICancellationSignal remoteCancellationSignal = null; 1403 if (cancellationSignal != null) { 1404 cancellationSignal.throwIfCanceled(); 1405 remoteCancellationSignal = unstableProvider.createCancellationSignal(); 1406 cancellationSignal.setRemote(remoteCancellationSignal); 1407 } 1408 1409 try { 1410 fd = unstableProvider.openTypedAssetFile( 1411 mPackageName, uri, mimeType, opts, remoteCancellationSignal); 1412 if (fd == null) { 1413 // The provider will be released by the finally{} clause 1414 return null; 1415 } 1416 } catch (DeadObjectException e) { 1417 // The remote process has died... but we only hold an unstable 1418 // reference though, so we might recover!!! Let's try!!!! 1419 // This is exciting!!1!!1!!!!1 1420 unstableProviderDied(unstableProvider); 1421 stableProvider = acquireProvider(uri); 1422 if (stableProvider == null) { 1423 throw new FileNotFoundException("No content provider: " + uri); 1424 } 1425 fd = stableProvider.openTypedAssetFile( 1426 mPackageName, uri, mimeType, opts, remoteCancellationSignal); 1427 if (fd == null) { 1428 // The provider will be released by the finally{} clause 1429 return null; 1430 } 1431 } 1432 1433 if (stableProvider == null) { 1434 stableProvider = acquireProvider(uri); 1435 } 1436 releaseUnstableProvider(unstableProvider); 1437 unstableProvider = null; 1438 ParcelFileDescriptor pfd = new ParcelFileDescriptorInner( 1439 fd.getParcelFileDescriptor(), stableProvider); 1440 1441 // Success! Don't release the provider when exiting, let 1442 // ParcelFileDescriptorInner do that when it is closed. 1443 stableProvider = null; 1444 1445 return new AssetFileDescriptor(pfd, fd.getStartOffset(), 1446 fd.getDeclaredLength()); 1447 1448 } catch (RemoteException e) { 1449 // Whatever, whatever, we'll go away. 1450 throw new FileNotFoundException( 1451 "Failed opening content provider: " + uri); 1452 } catch (FileNotFoundException e) { 1453 throw e; 1454 } finally { 1455 if (cancellationSignal != null) { 1456 cancellationSignal.setRemote(null); 1457 } 1458 if (stableProvider != null) { 1459 releaseProvider(stableProvider); 1460 } 1461 if (unstableProvider != null) { 1462 releaseUnstableProvider(unstableProvider); 1463 } 1464 } 1465 } 1466 1467 /** 1468 * A resource identified by the {@link Resources} that contains it, and a resource id. 1469 * 1470 * @hide 1471 */ 1472 public class OpenResourceIdResult { 1473 public Resources r; 1474 public int id; 1475 } 1476 1477 /** 1478 * Resolves an android.resource URI to a {@link Resources} and a resource id. 1479 * 1480 * @hide 1481 */ getResourceId(Uri uri)1482 public OpenResourceIdResult getResourceId(Uri uri) throws FileNotFoundException { 1483 String authority = uri.getAuthority(); 1484 Resources r; 1485 if (TextUtils.isEmpty(authority)) { 1486 throw new FileNotFoundException("No authority: " + uri); 1487 } else { 1488 try { 1489 r = mContext.getPackageManager().getResourcesForApplication(authority); 1490 } catch (NameNotFoundException ex) { 1491 throw new FileNotFoundException("No package found for authority: " + uri); 1492 } 1493 } 1494 List<String> path = uri.getPathSegments(); 1495 if (path == null) { 1496 throw new FileNotFoundException("No path: " + uri); 1497 } 1498 int len = path.size(); 1499 int id; 1500 if (len == 1) { 1501 try { 1502 id = Integer.parseInt(path.get(0)); 1503 } catch (NumberFormatException e) { 1504 throw new FileNotFoundException("Single path segment is not a resource ID: " + uri); 1505 } 1506 } else if (len == 2) { 1507 id = r.getIdentifier(path.get(1), path.get(0), authority); 1508 } else { 1509 throw new FileNotFoundException("More than two path segments: " + uri); 1510 } 1511 if (id == 0) { 1512 throw new FileNotFoundException("No resource found for: " + uri); 1513 } 1514 OpenResourceIdResult res = new OpenResourceIdResult(); 1515 res.r = r; 1516 res.id = id; 1517 return res; 1518 } 1519 1520 /** 1521 * Inserts a row into a table at the given URL. 1522 * 1523 * If the content provider supports transactions the insertion will be atomic. 1524 * 1525 * @param url The URL of the table to insert into. 1526 * @param values The initial values for the newly inserted row. The key is the column name for 1527 * the field. Passing an empty ContentValues will create an empty row. 1528 * @return the URL of the newly created row. 1529 */ insert(@equiresPermission.Write @onNull Uri url, @Nullable ContentValues values)1530 public final @Nullable Uri insert(@RequiresPermission.Write @NonNull Uri url, 1531 @Nullable ContentValues values) { 1532 Preconditions.checkNotNull(url, "url"); 1533 IContentProvider provider = acquireProvider(url); 1534 if (provider == null) { 1535 throw new IllegalArgumentException("Unknown URL " + url); 1536 } 1537 try { 1538 long startTime = SystemClock.uptimeMillis(); 1539 Uri createdRow = provider.insert(mPackageName, url, values); 1540 long durationMillis = SystemClock.uptimeMillis() - startTime; 1541 maybeLogUpdateToEventLog(durationMillis, url, "insert", null /* where */); 1542 return createdRow; 1543 } catch (RemoteException e) { 1544 // Arbitrary and not worth documenting, as Activity 1545 // Manager will kill this process shortly anyway. 1546 return null; 1547 } finally { 1548 releaseProvider(provider); 1549 } 1550 } 1551 1552 /** 1553 * Applies each of the {@link ContentProviderOperation} objects and returns an array 1554 * of their results. Passes through OperationApplicationException, which may be thrown 1555 * by the call to {@link ContentProviderOperation#apply}. 1556 * If all the applications succeed then a {@link ContentProviderResult} array with the 1557 * same number of elements as the operations will be returned. It is implementation-specific 1558 * how many, if any, operations will have been successfully applied if a call to 1559 * apply results in a {@link OperationApplicationException}. 1560 * @param authority the authority of the ContentProvider to which this batch should be applied 1561 * @param operations the operations to apply 1562 * @return the results of the applications 1563 * @throws OperationApplicationException thrown if an application fails. 1564 * See {@link ContentProviderOperation#apply} for more information. 1565 * @throws RemoteException thrown if a RemoteException is encountered while attempting 1566 * to communicate with a remote provider. 1567 */ applyBatch(@onNull String authority, @NonNull ArrayList<ContentProviderOperation> operations)1568 public @NonNull ContentProviderResult[] applyBatch(@NonNull String authority, 1569 @NonNull ArrayList<ContentProviderOperation> operations) 1570 throws RemoteException, OperationApplicationException { 1571 Preconditions.checkNotNull(authority, "authority"); 1572 Preconditions.checkNotNull(operations, "operations"); 1573 ContentProviderClient provider = acquireContentProviderClient(authority); 1574 if (provider == null) { 1575 throw new IllegalArgumentException("Unknown authority " + authority); 1576 } 1577 try { 1578 return provider.applyBatch(operations); 1579 } finally { 1580 provider.release(); 1581 } 1582 } 1583 1584 /** 1585 * Inserts multiple rows into a table at the given URL. 1586 * 1587 * This function make no guarantees about the atomicity of the insertions. 1588 * 1589 * @param url The URL of the table to insert into. 1590 * @param values The initial values for the newly inserted rows. The key is the column name for 1591 * the field. Passing null will create an empty row. 1592 * @return the number of newly created rows. 1593 */ bulkInsert(@equiresPermission.Write @onNull Uri url, @NonNull ContentValues[] values)1594 public final int bulkInsert(@RequiresPermission.Write @NonNull Uri url, 1595 @NonNull ContentValues[] values) { 1596 Preconditions.checkNotNull(url, "url"); 1597 Preconditions.checkNotNull(values, "values"); 1598 IContentProvider provider = acquireProvider(url); 1599 if (provider == null) { 1600 throw new IllegalArgumentException("Unknown URL " + url); 1601 } 1602 try { 1603 long startTime = SystemClock.uptimeMillis(); 1604 int rowsCreated = provider.bulkInsert(mPackageName, url, values); 1605 long durationMillis = SystemClock.uptimeMillis() - startTime; 1606 maybeLogUpdateToEventLog(durationMillis, url, "bulkinsert", null /* where */); 1607 return rowsCreated; 1608 } catch (RemoteException e) { 1609 // Arbitrary and not worth documenting, as Activity 1610 // Manager will kill this process shortly anyway. 1611 return 0; 1612 } finally { 1613 releaseProvider(provider); 1614 } 1615 } 1616 1617 /** 1618 * Deletes row(s) specified by a content URI. 1619 * 1620 * If the content provider supports transactions, the deletion will be atomic. 1621 * 1622 * @param url The URL of the row to delete. 1623 * @param where A filter to apply to rows before deleting, formatted as an SQL WHERE clause 1624 (excluding the WHERE itself). 1625 * @return The number of rows deleted. 1626 */ delete(@equiresPermission.Write @onNull Uri url, @Nullable String where, @Nullable String[] selectionArgs)1627 public final int delete(@RequiresPermission.Write @NonNull Uri url, @Nullable String where, 1628 @Nullable String[] selectionArgs) { 1629 Preconditions.checkNotNull(url, "url"); 1630 IContentProvider provider = acquireProvider(url); 1631 if (provider == null) { 1632 throw new IllegalArgumentException("Unknown URL " + url); 1633 } 1634 try { 1635 long startTime = SystemClock.uptimeMillis(); 1636 int rowsDeleted = provider.delete(mPackageName, url, where, selectionArgs); 1637 long durationMillis = SystemClock.uptimeMillis() - startTime; 1638 maybeLogUpdateToEventLog(durationMillis, url, "delete", where); 1639 return rowsDeleted; 1640 } catch (RemoteException e) { 1641 // Arbitrary and not worth documenting, as Activity 1642 // Manager will kill this process shortly anyway. 1643 return -1; 1644 } finally { 1645 releaseProvider(provider); 1646 } 1647 } 1648 1649 /** 1650 * Update row(s) in a content URI. 1651 * 1652 * If the content provider supports transactions the update will be atomic. 1653 * 1654 * @param uri The URI to modify. 1655 * @param values The new field values. The key is the column name for the field. 1656 A null value will remove an existing field value. 1657 * @param where A filter to apply to rows before updating, formatted as an SQL WHERE clause 1658 (excluding the WHERE itself). 1659 * @return the number of rows updated. 1660 * @throws NullPointerException if uri or values are null 1661 */ update(@equiresPermission.Write @onNull Uri uri, @Nullable ContentValues values, @Nullable String where, @Nullable String[] selectionArgs)1662 public final int update(@RequiresPermission.Write @NonNull Uri uri, 1663 @Nullable ContentValues values, @Nullable String where, 1664 @Nullable String[] selectionArgs) { 1665 Preconditions.checkNotNull(uri, "uri"); 1666 IContentProvider provider = acquireProvider(uri); 1667 if (provider == null) { 1668 throw new IllegalArgumentException("Unknown URI " + uri); 1669 } 1670 try { 1671 long startTime = SystemClock.uptimeMillis(); 1672 int rowsUpdated = provider.update(mPackageName, uri, values, where, selectionArgs); 1673 long durationMillis = SystemClock.uptimeMillis() - startTime; 1674 maybeLogUpdateToEventLog(durationMillis, uri, "update", where); 1675 return rowsUpdated; 1676 } catch (RemoteException e) { 1677 // Arbitrary and not worth documenting, as Activity 1678 // Manager will kill this process shortly anyway. 1679 return -1; 1680 } finally { 1681 releaseProvider(provider); 1682 } 1683 } 1684 1685 /** 1686 * Call a provider-defined method. This can be used to implement 1687 * read or write interfaces which are cheaper than using a Cursor and/or 1688 * do not fit into the traditional table model. 1689 * 1690 * @param method provider-defined method name to call. Opaque to 1691 * framework, but must be non-null. 1692 * @param arg provider-defined String argument. May be null. 1693 * @param extras provider-defined Bundle argument. May be null. 1694 * @return a result Bundle, possibly null. Will be null if the ContentProvider 1695 * does not implement call. 1696 * @throws NullPointerException if uri or method is null 1697 * @throws IllegalArgumentException if uri is not known 1698 */ call(@onNull Uri uri, @NonNull String method, @Nullable String arg, @Nullable Bundle extras)1699 public final @Nullable Bundle call(@NonNull Uri uri, @NonNull String method, 1700 @Nullable String arg, @Nullable Bundle extras) { 1701 Preconditions.checkNotNull(uri, "uri"); 1702 Preconditions.checkNotNull(method, "method"); 1703 IContentProvider provider = acquireProvider(uri); 1704 if (provider == null) { 1705 throw new IllegalArgumentException("Unknown URI " + uri); 1706 } 1707 try { 1708 final Bundle res = provider.call(mPackageName, method, arg, extras); 1709 Bundle.setDefusable(res, true); 1710 return res; 1711 } catch (RemoteException e) { 1712 // Arbitrary and not worth documenting, as Activity 1713 // Manager will kill this process shortly anyway. 1714 return null; 1715 } finally { 1716 releaseProvider(provider); 1717 } 1718 } 1719 1720 /** 1721 * Returns the content provider for the given content URI. 1722 * 1723 * @param uri The URI to a content provider 1724 * @return The ContentProvider for the given URI, or null if no content provider is found. 1725 * @hide 1726 */ acquireProvider(Uri uri)1727 public final IContentProvider acquireProvider(Uri uri) { 1728 if (!SCHEME_CONTENT.equals(uri.getScheme())) { 1729 return null; 1730 } 1731 final String auth = uri.getAuthority(); 1732 if (auth != null) { 1733 return acquireProvider(mContext, auth); 1734 } 1735 return null; 1736 } 1737 1738 /** 1739 * Returns the content provider for the given content URI if the process 1740 * already has a reference on it. 1741 * 1742 * @param uri The URI to a content provider 1743 * @return The ContentProvider for the given URI, or null if no content provider is found. 1744 * @hide 1745 */ acquireExistingProvider(Uri uri)1746 public final IContentProvider acquireExistingProvider(Uri uri) { 1747 if (!SCHEME_CONTENT.equals(uri.getScheme())) { 1748 return null; 1749 } 1750 final String auth = uri.getAuthority(); 1751 if (auth != null) { 1752 return acquireExistingProvider(mContext, auth); 1753 } 1754 return null; 1755 } 1756 1757 /** 1758 * @hide 1759 */ acquireProvider(String name)1760 public final IContentProvider acquireProvider(String name) { 1761 if (name == null) { 1762 return null; 1763 } 1764 return acquireProvider(mContext, name); 1765 } 1766 1767 /** 1768 * Returns the content provider for the given content URI. 1769 * 1770 * @param uri The URI to a content provider 1771 * @return The ContentProvider for the given URI, or null if no content provider is found. 1772 * @hide 1773 */ acquireUnstableProvider(Uri uri)1774 public final IContentProvider acquireUnstableProvider(Uri uri) { 1775 if (!SCHEME_CONTENT.equals(uri.getScheme())) { 1776 return null; 1777 } 1778 String auth = uri.getAuthority(); 1779 if (auth != null) { 1780 return acquireUnstableProvider(mContext, uri.getAuthority()); 1781 } 1782 return null; 1783 } 1784 1785 /** 1786 * @hide 1787 */ acquireUnstableProvider(String name)1788 public final IContentProvider acquireUnstableProvider(String name) { 1789 if (name == null) { 1790 return null; 1791 } 1792 return acquireUnstableProvider(mContext, name); 1793 } 1794 1795 /** 1796 * Returns a {@link ContentProviderClient} that is associated with the {@link ContentProvider} 1797 * that services the content at uri, starting the provider if necessary. Returns 1798 * null if there is no provider associated wih the uri. The caller must indicate that they are 1799 * done with the provider by calling {@link ContentProviderClient#release} which will allow 1800 * the system to release the provider it it determines that there is no other reason for 1801 * keeping it active. 1802 * @param uri specifies which provider should be acquired 1803 * @return a {@link ContentProviderClient} that is associated with the {@link ContentProvider} 1804 * that services the content at uri or null if there isn't one. 1805 */ acquireContentProviderClient(@onNull Uri uri)1806 public final @Nullable ContentProviderClient acquireContentProviderClient(@NonNull Uri uri) { 1807 Preconditions.checkNotNull(uri, "uri"); 1808 IContentProvider provider = acquireProvider(uri); 1809 if (provider != null) { 1810 return new ContentProviderClient(this, provider, true); 1811 } 1812 return null; 1813 } 1814 1815 /** 1816 * Returns a {@link ContentProviderClient} that is associated with the {@link ContentProvider} 1817 * with the authority of name, starting the provider if necessary. Returns 1818 * null if there is no provider associated wih the uri. The caller must indicate that they are 1819 * done with the provider by calling {@link ContentProviderClient#release} which will allow 1820 * the system to release the provider it it determines that there is no other reason for 1821 * keeping it active. 1822 * @param name specifies which provider should be acquired 1823 * @return a {@link ContentProviderClient} that is associated with the {@link ContentProvider} 1824 * with the authority of name or null if there isn't one. 1825 */ acquireContentProviderClient( @onNull String name)1826 public final @Nullable ContentProviderClient acquireContentProviderClient( 1827 @NonNull String name) { 1828 Preconditions.checkNotNull(name, "name"); 1829 IContentProvider provider = acquireProvider(name); 1830 if (provider != null) { 1831 return new ContentProviderClient(this, provider, true); 1832 } 1833 1834 return null; 1835 } 1836 1837 /** 1838 * Like {@link #acquireContentProviderClient(Uri)}, but for use when you do 1839 * not trust the stability of the target content provider. This turns off 1840 * the mechanism in the platform clean up processes that are dependent on 1841 * a content provider if that content provider's process goes away. Normally 1842 * you can safely assume that once you have acquired a provider, you can freely 1843 * use it as needed and it won't disappear, even if your process is in the 1844 * background. If using this method, you need to take care to deal with any 1845 * failures when communicating with the provider, and be sure to close it 1846 * so that it can be re-opened later. In particular, catching a 1847 * {@link android.os.DeadObjectException} from the calls there will let you 1848 * know that the content provider has gone away; at that point the current 1849 * ContentProviderClient object is invalid, and you should release it. You 1850 * can acquire a new one if you would like to try to restart the provider 1851 * and perform new operations on it. 1852 */ acquireUnstableContentProviderClient( @onNull Uri uri)1853 public final @Nullable ContentProviderClient acquireUnstableContentProviderClient( 1854 @NonNull Uri uri) { 1855 Preconditions.checkNotNull(uri, "uri"); 1856 IContentProvider provider = acquireUnstableProvider(uri); 1857 if (provider != null) { 1858 return new ContentProviderClient(this, provider, false); 1859 } 1860 1861 return null; 1862 } 1863 1864 /** 1865 * Like {@link #acquireContentProviderClient(String)}, but for use when you do 1866 * not trust the stability of the target content provider. This turns off 1867 * the mechanism in the platform clean up processes that are dependent on 1868 * a content provider if that content provider's process goes away. Normally 1869 * you can safely assume that once you have acquired a provider, you can freely 1870 * use it as needed and it won't disappear, even if your process is in the 1871 * background. If using this method, you need to take care to deal with any 1872 * failures when communicating with the provider, and be sure to close it 1873 * so that it can be re-opened later. In particular, catching a 1874 * {@link android.os.DeadObjectException} from the calls there will let you 1875 * know that the content provider has gone away; at that point the current 1876 * ContentProviderClient object is invalid, and you should release it. You 1877 * can acquire a new one if you would like to try to restart the provider 1878 * and perform new operations on it. 1879 */ acquireUnstableContentProviderClient( @onNull String name)1880 public final @Nullable ContentProviderClient acquireUnstableContentProviderClient( 1881 @NonNull String name) { 1882 Preconditions.checkNotNull(name, "name"); 1883 IContentProvider provider = acquireUnstableProvider(name); 1884 if (provider != null) { 1885 return new ContentProviderClient(this, provider, false); 1886 } 1887 1888 return null; 1889 } 1890 1891 /** 1892 * Register an observer class that gets callbacks when data identified by a 1893 * given content URI changes. 1894 * <p> 1895 * Starting in {@link android.os.Build.VERSION_CODES#O}, all content 1896 * notifications must be backed by a valid {@link ContentProvider}. 1897 * 1898 * @param uri The URI to watch for changes. This can be a specific row URI, 1899 * or a base URI for a whole class of content. 1900 * @param notifyForDescendants When false, the observer will be notified 1901 * whenever a change occurs to the exact URI specified by 1902 * <code>uri</code> or to one of the URI's ancestors in the path 1903 * hierarchy. When true, the observer will also be notified 1904 * whenever a change occurs to the URI's descendants in the path 1905 * hierarchy. 1906 * @param observer The object that receives callbacks when changes occur. 1907 * @see #unregisterContentObserver 1908 */ registerContentObserver(@onNull Uri uri, boolean notifyForDescendants, @NonNull ContentObserver observer)1909 public final void registerContentObserver(@NonNull Uri uri, boolean notifyForDescendants, 1910 @NonNull ContentObserver observer) { 1911 Preconditions.checkNotNull(uri, "uri"); 1912 Preconditions.checkNotNull(observer, "observer"); 1913 registerContentObserver( 1914 ContentProvider.getUriWithoutUserId(uri), 1915 notifyForDescendants, 1916 observer, 1917 ContentProvider.getUserIdFromUri(uri, mContext.getUserId())); 1918 } 1919 1920 /** @hide - designated user version */ registerContentObserver(Uri uri, boolean notifyForDescendents, ContentObserver observer, @UserIdInt int userHandle)1921 public final void registerContentObserver(Uri uri, boolean notifyForDescendents, 1922 ContentObserver observer, @UserIdInt int userHandle) { 1923 try { 1924 getContentService().registerContentObserver(uri, notifyForDescendents, 1925 observer.getContentObserver(), userHandle, mTargetSdkVersion); 1926 } catch (RemoteException e) { 1927 } 1928 } 1929 1930 /** 1931 * Unregisters a change observer. 1932 * 1933 * @param observer The previously registered observer that is no longer needed. 1934 * @see #registerContentObserver 1935 */ unregisterContentObserver(@onNull ContentObserver observer)1936 public final void unregisterContentObserver(@NonNull ContentObserver observer) { 1937 Preconditions.checkNotNull(observer, "observer"); 1938 try { 1939 IContentObserver contentObserver = observer.releaseContentObserver(); 1940 if (contentObserver != null) { 1941 getContentService().unregisterContentObserver( 1942 contentObserver); 1943 } 1944 } catch (RemoteException e) { 1945 } 1946 } 1947 1948 /** 1949 * Notify registered observers that a row was updated and attempt to sync 1950 * changes to the network. 1951 * <p> 1952 * To observe events sent through this call, use 1953 * {@link #registerContentObserver(Uri, boolean, ContentObserver)}. 1954 * <p> 1955 * Starting in {@link android.os.Build.VERSION_CODES#O}, all content 1956 * notifications must be backed by a valid {@link ContentProvider}. 1957 * 1958 * @param uri The uri of the content that was changed. 1959 * @param observer The observer that originated the change, may be 1960 * <code>null</null>. The observer that originated the change 1961 * will only receive the notification if it has requested to 1962 * receive self-change notifications by implementing 1963 * {@link ContentObserver#deliverSelfNotifications()} to return 1964 * true. 1965 */ notifyChange(@onNull Uri uri, @Nullable ContentObserver observer)1966 public void notifyChange(@NonNull Uri uri, @Nullable ContentObserver observer) { 1967 notifyChange(uri, observer, true /* sync to network */); 1968 } 1969 1970 /** 1971 * Notify registered observers that a row was updated. 1972 * <p> 1973 * To observe events sent through this call, use 1974 * {@link #registerContentObserver(Uri, boolean, ContentObserver)}. 1975 * <p> 1976 * If syncToNetwork is true, this will attempt to schedule a local sync 1977 * using the sync adapter that's registered for the authority of the 1978 * provided uri. No account will be passed to the sync adapter, so all 1979 * matching accounts will be synchronized. 1980 * <p> 1981 * Starting in {@link android.os.Build.VERSION_CODES#O}, all content 1982 * notifications must be backed by a valid {@link ContentProvider}. 1983 * 1984 * @param uri The uri of the content that was changed. 1985 * @param observer The observer that originated the change, may be 1986 * <code>null</null>. The observer that originated the change 1987 * will only receive the notification if it has requested to 1988 * receive self-change notifications by implementing 1989 * {@link ContentObserver#deliverSelfNotifications()} to return 1990 * true. 1991 * @param syncToNetwork If true, same as {@link #NOTIFY_SYNC_TO_NETWORK}. 1992 * @see #requestSync(android.accounts.Account, String, android.os.Bundle) 1993 */ notifyChange(@onNull Uri uri, @Nullable ContentObserver observer, boolean syncToNetwork)1994 public void notifyChange(@NonNull Uri uri, @Nullable ContentObserver observer, 1995 boolean syncToNetwork) { 1996 Preconditions.checkNotNull(uri, "uri"); 1997 notifyChange( 1998 ContentProvider.getUriWithoutUserId(uri), 1999 observer, 2000 syncToNetwork, 2001 ContentProvider.getUserIdFromUri(uri, mContext.getUserId())); 2002 } 2003 2004 /** 2005 * Notify registered observers that a row was updated. 2006 * <p> 2007 * To observe events sent through this call, use 2008 * {@link #registerContentObserver(Uri, boolean, ContentObserver)}. 2009 * <p> 2010 * If syncToNetwork is true, this will attempt to schedule a local sync 2011 * using the sync adapter that's registered for the authority of the 2012 * provided uri. No account will be passed to the sync adapter, so all 2013 * matching accounts will be synchronized. 2014 * <p> 2015 * Starting in {@link android.os.Build.VERSION_CODES#O}, all content 2016 * notifications must be backed by a valid {@link ContentProvider}. 2017 * 2018 * @param uri The uri of the content that was changed. 2019 * @param observer The observer that originated the change, may be 2020 * <code>null</null>. The observer that originated the change 2021 * will only receive the notification if it has requested to 2022 * receive self-change notifications by implementing 2023 * {@link ContentObserver#deliverSelfNotifications()} to return 2024 * true. 2025 * @param flags Additional flags: {@link #NOTIFY_SYNC_TO_NETWORK}. 2026 * @see #requestSync(android.accounts.Account, String, android.os.Bundle) 2027 */ notifyChange(@onNull Uri uri, @Nullable ContentObserver observer, @NotifyFlags int flags)2028 public void notifyChange(@NonNull Uri uri, @Nullable ContentObserver observer, 2029 @NotifyFlags int flags) { 2030 Preconditions.checkNotNull(uri, "uri"); 2031 notifyChange( 2032 ContentProvider.getUriWithoutUserId(uri), 2033 observer, 2034 flags, 2035 ContentProvider.getUserIdFromUri(uri, mContext.getUserId())); 2036 } 2037 2038 /** 2039 * Notify registered observers within the designated user(s) that a row was updated. 2040 * 2041 * @hide 2042 */ notifyChange(@onNull Uri uri, ContentObserver observer, boolean syncToNetwork, @UserIdInt int userHandle)2043 public void notifyChange(@NonNull Uri uri, ContentObserver observer, boolean syncToNetwork, 2044 @UserIdInt int userHandle) { 2045 try { 2046 getContentService().notifyChange( 2047 uri, observer == null ? null : observer.getContentObserver(), 2048 observer != null && observer.deliverSelfNotifications(), 2049 syncToNetwork ? NOTIFY_SYNC_TO_NETWORK : 0, 2050 userHandle, mTargetSdkVersion); 2051 } catch (RemoteException e) { 2052 } 2053 } 2054 2055 /** 2056 * Notify registered observers within the designated user(s) that a row was updated. 2057 * 2058 * @hide 2059 */ notifyChange(@onNull Uri uri, ContentObserver observer, @NotifyFlags int flags, @UserIdInt int userHandle)2060 public void notifyChange(@NonNull Uri uri, ContentObserver observer, @NotifyFlags int flags, 2061 @UserIdInt int userHandle) { 2062 try { 2063 getContentService().notifyChange( 2064 uri, observer == null ? null : observer.getContentObserver(), 2065 observer != null && observer.deliverSelfNotifications(), flags, 2066 userHandle, mTargetSdkVersion); 2067 } catch (RemoteException e) { 2068 } 2069 } 2070 2071 /** 2072 * Take a persistable URI permission grant that has been offered. Once 2073 * taken, the permission grant will be remembered across device reboots. 2074 * Only URI permissions granted with 2075 * {@link Intent#FLAG_GRANT_PERSISTABLE_URI_PERMISSION} can be persisted. If 2076 * the grant has already been persisted, taking it again will touch 2077 * {@link UriPermission#getPersistedTime()}. 2078 * 2079 * @see #getPersistedUriPermissions() 2080 */ takePersistableUriPermission(@onNull Uri uri, @Intent.AccessUriMode int modeFlags)2081 public void takePersistableUriPermission(@NonNull Uri uri, 2082 @Intent.AccessUriMode int modeFlags) { 2083 Preconditions.checkNotNull(uri, "uri"); 2084 try { 2085 ActivityManager.getService().takePersistableUriPermission( 2086 ContentProvider.getUriWithoutUserId(uri), modeFlags, resolveUserId(uri)); 2087 } catch (RemoteException e) { 2088 } 2089 } 2090 2091 /** 2092 * Relinquish a persisted URI permission grant. The URI must have been 2093 * previously made persistent with 2094 * {@link #takePersistableUriPermission(Uri, int)}. Any non-persistent 2095 * grants to the calling package will remain intact. 2096 * 2097 * @see #getPersistedUriPermissions() 2098 */ releasePersistableUriPermission(@onNull Uri uri, @Intent.AccessUriMode int modeFlags)2099 public void releasePersistableUriPermission(@NonNull Uri uri, 2100 @Intent.AccessUriMode int modeFlags) { 2101 Preconditions.checkNotNull(uri, "uri"); 2102 try { 2103 ActivityManager.getService().releasePersistableUriPermission( 2104 ContentProvider.getUriWithoutUserId(uri), modeFlags, resolveUserId(uri)); 2105 } catch (RemoteException e) { 2106 } 2107 } 2108 2109 /** 2110 * Return list of all URI permission grants that have been persisted by the 2111 * calling app. That is, the returned permissions have been granted 2112 * <em>to</em> the calling app. Only persistable grants taken with 2113 * {@link #takePersistableUriPermission(Uri, int)} are returned. 2114 * <p>Note: Some of the returned URIs may not be usable until after the user is unlocked. 2115 * 2116 * @see #takePersistableUriPermission(Uri, int) 2117 * @see #releasePersistableUriPermission(Uri, int) 2118 */ getPersistedUriPermissions()2119 public @NonNull List<UriPermission> getPersistedUriPermissions() { 2120 try { 2121 return ActivityManager.getService() 2122 .getPersistedUriPermissions(mPackageName, true).getList(); 2123 } catch (RemoteException e) { 2124 throw new RuntimeException("Activity manager has died", e); 2125 } 2126 } 2127 2128 /** 2129 * Return list of all persisted URI permission grants that are hosted by the 2130 * calling app. That is, the returned permissions have been granted 2131 * <em>from</em> the calling app. Only grants taken with 2132 * {@link #takePersistableUriPermission(Uri, int)} are returned. 2133 * <p>Note: Some of the returned URIs may not be usable until after the user is unlocked. 2134 */ getOutgoingPersistedUriPermissions()2135 public @NonNull List<UriPermission> getOutgoingPersistedUriPermissions() { 2136 try { 2137 return ActivityManager.getService() 2138 .getPersistedUriPermissions(mPackageName, false).getList(); 2139 } catch (RemoteException e) { 2140 throw new RuntimeException("Activity manager has died", e); 2141 } 2142 } 2143 2144 /** 2145 * Start an asynchronous sync operation. If you want to monitor the progress 2146 * of the sync you may register a SyncObserver. Only values of the following 2147 * types may be used in the extras bundle: 2148 * <ul> 2149 * <li>Integer</li> 2150 * <li>Long</li> 2151 * <li>Boolean</li> 2152 * <li>Float</li> 2153 * <li>Double</li> 2154 * <li>String</li> 2155 * <li>Account</li> 2156 * <li>null</li> 2157 * </ul> 2158 * 2159 * @param uri the uri of the provider to sync or null to sync all providers. 2160 * @param extras any extras to pass to the SyncAdapter. 2161 * @deprecated instead use 2162 * {@link #requestSync(android.accounts.Account, String, android.os.Bundle)} 2163 */ 2164 @Deprecated startSync(Uri uri, Bundle extras)2165 public void startSync(Uri uri, Bundle extras) { 2166 Account account = null; 2167 if (extras != null) { 2168 String accountName = extras.getString(SYNC_EXTRAS_ACCOUNT); 2169 if (!TextUtils.isEmpty(accountName)) { 2170 // TODO: No references to Google in AOSP 2171 account = new Account(accountName, "com.google"); 2172 } 2173 extras.remove(SYNC_EXTRAS_ACCOUNT); 2174 } 2175 requestSync(account, uri != null ? uri.getAuthority() : null, extras); 2176 } 2177 2178 /** 2179 * Start an asynchronous sync operation. If you want to monitor the progress 2180 * of the sync you may register a SyncObserver. Only values of the following 2181 * types may be used in the extras bundle: 2182 * <ul> 2183 * <li>Integer</li> 2184 * <li>Long</li> 2185 * <li>Boolean</li> 2186 * <li>Float</li> 2187 * <li>Double</li> 2188 * <li>String</li> 2189 * <li>Account</li> 2190 * <li>null</li> 2191 * </ul> 2192 * 2193 * @param account which account should be synced 2194 * @param authority which authority should be synced 2195 * @param extras any extras to pass to the SyncAdapter. 2196 */ requestSync(Account account, String authority, Bundle extras)2197 public static void requestSync(Account account, String authority, Bundle extras) { 2198 requestSyncAsUser(account, authority, UserHandle.myUserId(), extras); 2199 } 2200 2201 /** 2202 * @see #requestSync(Account, String, Bundle) 2203 * @hide 2204 */ requestSyncAsUser(Account account, String authority, @UserIdInt int userId, Bundle extras)2205 public static void requestSyncAsUser(Account account, String authority, @UserIdInt int userId, 2206 Bundle extras) { 2207 if (extras == null) { 2208 throw new IllegalArgumentException("Must specify extras."); 2209 } 2210 SyncRequest request = 2211 new SyncRequest.Builder() 2212 .setSyncAdapter(account, authority) 2213 .setExtras(extras) 2214 .syncOnce() // Immediate sync. 2215 .build(); 2216 try { 2217 getContentService().syncAsUser(request, userId); 2218 } catch(RemoteException e) { 2219 // Shouldn't happen. 2220 } 2221 } 2222 2223 /** 2224 * Register a sync with the SyncManager. These requests are built using the 2225 * {@link SyncRequest.Builder}. 2226 */ requestSync(SyncRequest request)2227 public static void requestSync(SyncRequest request) { 2228 try { 2229 getContentService().sync(request); 2230 } catch(RemoteException e) { 2231 // Shouldn't happen. 2232 } 2233 } 2234 2235 /** 2236 * Check that only values of the following types are in the Bundle: 2237 * <ul> 2238 * <li>Integer</li> 2239 * <li>Long</li> 2240 * <li>Boolean</li> 2241 * <li>Float</li> 2242 * <li>Double</li> 2243 * <li>String</li> 2244 * <li>Account</li> 2245 * <li>null</li> 2246 * </ul> 2247 * @param extras the Bundle to check 2248 */ validateSyncExtrasBundle(Bundle extras)2249 public static void validateSyncExtrasBundle(Bundle extras) { 2250 try { 2251 for (String key : extras.keySet()) { 2252 Object value = extras.get(key); 2253 if (value == null) continue; 2254 if (value instanceof Long) continue; 2255 if (value instanceof Integer) continue; 2256 if (value instanceof Boolean) continue; 2257 if (value instanceof Float) continue; 2258 if (value instanceof Double) continue; 2259 if (value instanceof String) continue; 2260 if (value instanceof Account) continue; 2261 throw new IllegalArgumentException("unexpected value type: " 2262 + value.getClass().getName()); 2263 } 2264 } catch (IllegalArgumentException e) { 2265 throw e; 2266 } catch (RuntimeException exc) { 2267 throw new IllegalArgumentException("error unparceling Bundle", exc); 2268 } 2269 } 2270 2271 /** 2272 * Cancel any active or pending syncs that match the Uri. If the uri is null then 2273 * all syncs will be canceled. 2274 * 2275 * @param uri the uri of the provider to sync or null to sync all providers. 2276 * @deprecated instead use {@link #cancelSync(android.accounts.Account, String)} 2277 */ 2278 @Deprecated cancelSync(Uri uri)2279 public void cancelSync(Uri uri) { 2280 cancelSync(null /* all accounts */, uri != null ? uri.getAuthority() : null); 2281 } 2282 2283 /** 2284 * Cancel any active or pending syncs that match account and authority. The account and 2285 * authority can each independently be set to null, which means that syncs with any account 2286 * or authority, respectively, will match. 2287 * 2288 * @param account filters the syncs that match by this account 2289 * @param authority filters the syncs that match by this authority 2290 */ cancelSync(Account account, String authority)2291 public static void cancelSync(Account account, String authority) { 2292 try { 2293 getContentService().cancelSync(account, authority, null); 2294 } catch (RemoteException e) { 2295 } 2296 } 2297 2298 /** 2299 * @see #cancelSync(Account, String) 2300 * @hide 2301 */ cancelSyncAsUser(Account account, String authority, @UserIdInt int userId)2302 public static void cancelSyncAsUser(Account account, String authority, @UserIdInt int userId) { 2303 try { 2304 getContentService().cancelSyncAsUser(account, authority, null, userId); 2305 } catch (RemoteException e) { 2306 } 2307 } 2308 2309 /** 2310 * Get information about the SyncAdapters that are known to the system. 2311 * @return an array of SyncAdapters that have registered with the system 2312 */ getSyncAdapterTypes()2313 public static SyncAdapterType[] getSyncAdapterTypes() { 2314 try { 2315 return getContentService().getSyncAdapterTypes(); 2316 } catch (RemoteException e) { 2317 throw new RuntimeException("the ContentService should always be reachable", e); 2318 } 2319 } 2320 2321 /** 2322 * @see #getSyncAdapterTypes() 2323 * @hide 2324 */ getSyncAdapterTypesAsUser(@serIdInt int userId)2325 public static SyncAdapterType[] getSyncAdapterTypesAsUser(@UserIdInt int userId) { 2326 try { 2327 return getContentService().getSyncAdapterTypesAsUser(userId); 2328 } catch (RemoteException e) { 2329 throw new RuntimeException("the ContentService should always be reachable", e); 2330 } 2331 } 2332 2333 /** 2334 * @hide 2335 * Returns the package names of syncadapters that match a given user and authority. 2336 */ 2337 @TestApi getSyncAdapterPackagesForAuthorityAsUser(String authority, @UserIdInt int userId)2338 public static String[] getSyncAdapterPackagesForAuthorityAsUser(String authority, 2339 @UserIdInt int userId) { 2340 try { 2341 return getContentService().getSyncAdapterPackagesForAuthorityAsUser(authority, userId); 2342 } catch (RemoteException e) { 2343 } 2344 return ArrayUtils.emptyArray(String.class); 2345 } 2346 2347 /** 2348 * Check if the provider should be synced when a network tickle is received 2349 * <p>This method requires the caller to hold the permission 2350 * {@link android.Manifest.permission#READ_SYNC_SETTINGS}. 2351 * 2352 * @param account the account whose setting we are querying 2353 * @param authority the provider whose setting we are querying 2354 * @return true if the provider should be synced when a network tickle is received 2355 */ getSyncAutomatically(Account account, String authority)2356 public static boolean getSyncAutomatically(Account account, String authority) { 2357 try { 2358 return getContentService().getSyncAutomatically(account, authority); 2359 } catch (RemoteException e) { 2360 throw new RuntimeException("the ContentService should always be reachable", e); 2361 } 2362 } 2363 2364 /** 2365 * @see #getSyncAutomatically(Account, String) 2366 * @hide 2367 */ getSyncAutomaticallyAsUser(Account account, String authority, @UserIdInt int userId)2368 public static boolean getSyncAutomaticallyAsUser(Account account, String authority, 2369 @UserIdInt int userId) { 2370 try { 2371 return getContentService().getSyncAutomaticallyAsUser(account, authority, userId); 2372 } catch (RemoteException e) { 2373 throw new RuntimeException("the ContentService should always be reachable", e); 2374 } 2375 } 2376 2377 /** 2378 * Set whether or not the provider is synced when it receives a network tickle. 2379 * <p>This method requires the caller to hold the permission 2380 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2381 * 2382 * @param account the account whose setting we are querying 2383 * @param authority the provider whose behavior is being controlled 2384 * @param sync true if the provider should be synced when tickles are received for it 2385 */ setSyncAutomatically(Account account, String authority, boolean sync)2386 public static void setSyncAutomatically(Account account, String authority, boolean sync) { 2387 setSyncAutomaticallyAsUser(account, authority, sync, UserHandle.myUserId()); 2388 } 2389 2390 /** 2391 * @see #setSyncAutomatically(Account, String, boolean) 2392 * @hide 2393 */ setSyncAutomaticallyAsUser(Account account, String authority, boolean sync, @UserIdInt int userId)2394 public static void setSyncAutomaticallyAsUser(Account account, String authority, boolean sync, 2395 @UserIdInt int userId) { 2396 try { 2397 getContentService().setSyncAutomaticallyAsUser(account, authority, sync, userId); 2398 } catch (RemoteException e) { 2399 // exception ignored; if this is thrown then it means the runtime is in the midst of 2400 // being restarted 2401 } 2402 } 2403 2404 /** 2405 * Specifies that a sync should be requested with the specified the account, authority, 2406 * and extras at the given frequency. If there is already another periodic sync scheduled 2407 * with the account, authority and extras then a new periodic sync won't be added, instead 2408 * the frequency of the previous one will be updated. 2409 * <p> 2410 * These periodic syncs honor the "syncAutomatically" and "masterSyncAutomatically" settings. 2411 * Although these sync are scheduled at the specified frequency, it may take longer for it to 2412 * actually be started if other syncs are ahead of it in the sync operation queue. This means 2413 * that the actual start time may drift. 2414 * <p> 2415 * Periodic syncs are not allowed to have any of {@link #SYNC_EXTRAS_DO_NOT_RETRY}, 2416 * {@link #SYNC_EXTRAS_IGNORE_BACKOFF}, {@link #SYNC_EXTRAS_IGNORE_SETTINGS}, 2417 * {@link #SYNC_EXTRAS_INITIALIZE}, {@link #SYNC_EXTRAS_FORCE}, 2418 * {@link #SYNC_EXTRAS_EXPEDITED}, {@link #SYNC_EXTRAS_MANUAL} set to true. 2419 * If any are supplied then an {@link IllegalArgumentException} will be thrown. 2420 * 2421 * <p>This method requires the caller to hold the permission 2422 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2423 * <p>The bundle for a periodic sync can be queried by applications with the correct 2424 * permissions using 2425 * {@link ContentResolver#getPeriodicSyncs(Account account, String provider)}, so no 2426 * sensitive data should be transferred here. 2427 * 2428 * @param account the account to specify in the sync 2429 * @param authority the provider to specify in the sync request 2430 * @param extras extra parameters to go along with the sync request 2431 * @param pollFrequency how frequently the sync should be performed, in seconds. A minimum value 2432 * of 1 hour is enforced. 2433 * @throws IllegalArgumentException if an illegal extra was set or if any of the parameters 2434 * are null. 2435 */ addPeriodicSync(Account account, String authority, Bundle extras, long pollFrequency)2436 public static void addPeriodicSync(Account account, String authority, Bundle extras, 2437 long pollFrequency) { 2438 validateSyncExtrasBundle(extras); 2439 if (extras.getBoolean(SYNC_EXTRAS_MANUAL, false) 2440 || extras.getBoolean(SYNC_EXTRAS_DO_NOT_RETRY, false) 2441 || extras.getBoolean(SYNC_EXTRAS_IGNORE_BACKOFF, false) 2442 || extras.getBoolean(SYNC_EXTRAS_IGNORE_SETTINGS, false) 2443 || extras.getBoolean(SYNC_EXTRAS_INITIALIZE, false) 2444 || extras.getBoolean(SYNC_EXTRAS_FORCE, false) 2445 || extras.getBoolean(SYNC_EXTRAS_EXPEDITED, false)) { 2446 throw new IllegalArgumentException("illegal extras were set"); 2447 } 2448 try { 2449 getContentService().addPeriodicSync(account, authority, extras, pollFrequency); 2450 } catch (RemoteException e) { 2451 // exception ignored; if this is thrown then it means the runtime is in the midst of 2452 // being restarted 2453 } 2454 } 2455 2456 /** 2457 * {@hide} 2458 * Helper function to throw an <code>IllegalArgumentException</code> if any illegal 2459 * extras were set for a periodic sync. 2460 * 2461 * @param extras bundle to validate. 2462 */ invalidPeriodicExtras(Bundle extras)2463 public static boolean invalidPeriodicExtras(Bundle extras) { 2464 if (extras.getBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, false) 2465 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_DO_NOT_RETRY, false) 2466 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF, false) 2467 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_SETTINGS, false) 2468 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_INITIALIZE, false) 2469 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_FORCE, false) 2470 || extras.getBoolean(ContentResolver.SYNC_EXTRAS_EXPEDITED, false)) { 2471 return true; 2472 } 2473 return false; 2474 } 2475 2476 /** 2477 * Remove a periodic sync. Has no affect if account, authority and extras don't match 2478 * an existing periodic sync. 2479 * <p>This method requires the caller to hold the permission 2480 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2481 * 2482 * @param account the account of the periodic sync to remove 2483 * @param authority the provider of the periodic sync to remove 2484 * @param extras the extras of the periodic sync to remove 2485 */ removePeriodicSync(Account account, String authority, Bundle extras)2486 public static void removePeriodicSync(Account account, String authority, Bundle extras) { 2487 validateSyncExtrasBundle(extras); 2488 try { 2489 getContentService().removePeriodicSync(account, authority, extras); 2490 } catch (RemoteException e) { 2491 throw new RuntimeException("the ContentService should always be reachable", e); 2492 } 2493 } 2494 2495 /** 2496 * Remove the specified sync. This will cancel any pending or active syncs. If the request is 2497 * for a periodic sync, this call will remove any future occurrences. 2498 * <p> 2499 * If a periodic sync is specified, the caller must hold the permission 2500 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2501 *</p> 2502 * It is possible to cancel a sync using a SyncRequest object that is not the same object 2503 * with which you requested the sync. Do so by building a SyncRequest with the same 2504 * adapter, frequency, <b>and</b> extras bundle. 2505 * 2506 * @param request SyncRequest object containing information about sync to cancel. 2507 */ cancelSync(SyncRequest request)2508 public static void cancelSync(SyncRequest request) { 2509 if (request == null) { 2510 throw new IllegalArgumentException("request cannot be null"); 2511 } 2512 try { 2513 getContentService().cancelRequest(request); 2514 } catch (RemoteException e) { 2515 // exception ignored; if this is thrown then it means the runtime is in the midst of 2516 // being restarted 2517 } 2518 } 2519 2520 /** 2521 * Get the list of information about the periodic syncs for the given account and authority. 2522 * <p>This method requires the caller to hold the permission 2523 * {@link android.Manifest.permission#READ_SYNC_SETTINGS}. 2524 * 2525 * @param account the account whose periodic syncs we are querying 2526 * @param authority the provider whose periodic syncs we are querying 2527 * @return a list of PeriodicSync objects. This list may be empty but will never be null. 2528 */ getPeriodicSyncs(Account account, String authority)2529 public static List<PeriodicSync> getPeriodicSyncs(Account account, String authority) { 2530 try { 2531 return getContentService().getPeriodicSyncs(account, authority, null); 2532 } catch (RemoteException e) { 2533 throw new RuntimeException("the ContentService should always be reachable", e); 2534 } 2535 } 2536 2537 /** 2538 * Check if this account/provider is syncable. 2539 * <p>This method requires the caller to hold the permission 2540 * {@link android.Manifest.permission#READ_SYNC_SETTINGS}. 2541 * @return >0 if it is syncable, 0 if not, and <0 if the state isn't known yet. 2542 */ getIsSyncable(Account account, String authority)2543 public static int getIsSyncable(Account account, String authority) { 2544 try { 2545 return getContentService().getIsSyncable(account, authority); 2546 } catch (RemoteException e) { 2547 throw new RuntimeException("the ContentService should always be reachable", e); 2548 } 2549 } 2550 2551 /** 2552 * @see #getIsSyncable(Account, String) 2553 * @hide 2554 */ getIsSyncableAsUser(Account account, String authority, @UserIdInt int userId)2555 public static int getIsSyncableAsUser(Account account, String authority, 2556 @UserIdInt int userId) { 2557 try { 2558 return getContentService().getIsSyncableAsUser(account, authority, userId); 2559 } catch (RemoteException e) { 2560 throw new RuntimeException("the ContentService should always be reachable", e); 2561 } 2562 } 2563 2564 /** 2565 * Set whether this account/provider is syncable. 2566 * <p>This method requires the caller to hold the permission 2567 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2568 * @param syncable >0 denotes syncable, 0 means not syncable, <0 means unknown 2569 */ setIsSyncable(Account account, String authority, int syncable)2570 public static void setIsSyncable(Account account, String authority, int syncable) { 2571 try { 2572 getContentService().setIsSyncable(account, authority, syncable); 2573 } catch (RemoteException e) { 2574 // exception ignored; if this is thrown then it means the runtime is in the midst of 2575 // being restarted 2576 } 2577 } 2578 2579 /** 2580 * Gets the master auto-sync setting that applies to all the providers and accounts. 2581 * If this is false then the per-provider auto-sync setting is ignored. 2582 * <p>This method requires the caller to hold the permission 2583 * {@link android.Manifest.permission#READ_SYNC_SETTINGS}. 2584 * 2585 * @return the master auto-sync setting that applies to all the providers and accounts 2586 */ getMasterSyncAutomatically()2587 public static boolean getMasterSyncAutomatically() { 2588 try { 2589 return getContentService().getMasterSyncAutomatically(); 2590 } catch (RemoteException e) { 2591 throw new RuntimeException("the ContentService should always be reachable", e); 2592 } 2593 } 2594 2595 /** 2596 * @see #getMasterSyncAutomatically() 2597 * @hide 2598 */ getMasterSyncAutomaticallyAsUser(@serIdInt int userId)2599 public static boolean getMasterSyncAutomaticallyAsUser(@UserIdInt int userId) { 2600 try { 2601 return getContentService().getMasterSyncAutomaticallyAsUser(userId); 2602 } catch (RemoteException e) { 2603 throw new RuntimeException("the ContentService should always be reachable", e); 2604 } 2605 } 2606 2607 /** 2608 * Sets the master auto-sync setting that applies to all the providers and accounts. 2609 * If this is false then the per-provider auto-sync setting is ignored. 2610 * <p>This method requires the caller to hold the permission 2611 * {@link android.Manifest.permission#WRITE_SYNC_SETTINGS}. 2612 * 2613 * @param sync the master auto-sync setting that applies to all the providers and accounts 2614 */ setMasterSyncAutomatically(boolean sync)2615 public static void setMasterSyncAutomatically(boolean sync) { 2616 setMasterSyncAutomaticallyAsUser(sync, UserHandle.myUserId()); 2617 } 2618 2619 /** 2620 * @see #setMasterSyncAutomatically(boolean) 2621 * @hide 2622 */ setMasterSyncAutomaticallyAsUser(boolean sync, @UserIdInt int userId)2623 public static void setMasterSyncAutomaticallyAsUser(boolean sync, @UserIdInt int userId) { 2624 try { 2625 getContentService().setMasterSyncAutomaticallyAsUser(sync, userId); 2626 } catch (RemoteException e) { 2627 // exception ignored; if this is thrown then it means the runtime is in the midst of 2628 // being restarted 2629 } 2630 } 2631 2632 /** 2633 * Returns true if there is currently a sync operation for the given account or authority 2634 * actively being processed. 2635 * <p>This method requires the caller to hold the permission 2636 * {@link android.Manifest.permission#READ_SYNC_STATS}. 2637 * @param account the account whose setting we are querying 2638 * @param authority the provider whose behavior is being queried 2639 * @return true if a sync is active for the given account or authority. 2640 */ isSyncActive(Account account, String authority)2641 public static boolean isSyncActive(Account account, String authority) { 2642 if (account == null) { 2643 throw new IllegalArgumentException("account must not be null"); 2644 } 2645 if (authority == null) { 2646 throw new IllegalArgumentException("authority must not be null"); 2647 } 2648 2649 try { 2650 return getContentService().isSyncActive(account, authority, null); 2651 } catch (RemoteException e) { 2652 throw new RuntimeException("the ContentService should always be reachable", e); 2653 } 2654 } 2655 2656 /** 2657 * If a sync is active returns the information about it, otherwise returns null. 2658 * <p> 2659 * This method requires the caller to hold the permission 2660 * {@link android.Manifest.permission#READ_SYNC_STATS}. 2661 * <p> 2662 * @return the SyncInfo for the currently active sync or null if one is not active. 2663 * @deprecated 2664 * Since multiple concurrent syncs are now supported you should use 2665 * {@link #getCurrentSyncs()} to get the accurate list of current syncs. 2666 * This method returns the first item from the list of current syncs 2667 * or null if there are none. 2668 */ 2669 @Deprecated getCurrentSync()2670 public static SyncInfo getCurrentSync() { 2671 try { 2672 final List<SyncInfo> syncs = getContentService().getCurrentSyncs(); 2673 if (syncs.isEmpty()) { 2674 return null; 2675 } 2676 return syncs.get(0); 2677 } catch (RemoteException e) { 2678 throw new RuntimeException("the ContentService should always be reachable", e); 2679 } 2680 } 2681 2682 /** 2683 * Returns a list with information about all the active syncs. This list will be empty 2684 * if there are no active syncs. 2685 * <p> 2686 * This method requires the caller to hold the permission 2687 * {@link android.Manifest.permission#READ_SYNC_STATS}. 2688 * <p> 2689 * @return a List of SyncInfo objects for the currently active syncs. 2690 */ getCurrentSyncs()2691 public static List<SyncInfo> getCurrentSyncs() { 2692 try { 2693 return getContentService().getCurrentSyncs(); 2694 } catch (RemoteException e) { 2695 throw new RuntimeException("the ContentService should always be reachable", e); 2696 } 2697 } 2698 2699 /** 2700 * @see #getCurrentSyncs() 2701 * @hide 2702 */ getCurrentSyncsAsUser(@serIdInt int userId)2703 public static List<SyncInfo> getCurrentSyncsAsUser(@UserIdInt int userId) { 2704 try { 2705 return getContentService().getCurrentSyncsAsUser(userId); 2706 } catch (RemoteException e) { 2707 throw new RuntimeException("the ContentService should always be reachable", e); 2708 } 2709 } 2710 2711 /** 2712 * Returns the status that matches the authority. 2713 * @param account the account whose setting we are querying 2714 * @param authority the provider whose behavior is being queried 2715 * @return the SyncStatusInfo for the authority, or null if none exists 2716 * @hide 2717 */ getSyncStatus(Account account, String authority)2718 public static SyncStatusInfo getSyncStatus(Account account, String authority) { 2719 try { 2720 return getContentService().getSyncStatus(account, authority, null); 2721 } catch (RemoteException e) { 2722 throw new RuntimeException("the ContentService should always be reachable", e); 2723 } 2724 } 2725 2726 /** 2727 * @see #getSyncStatus(Account, String) 2728 * @hide 2729 */ getSyncStatusAsUser(Account account, String authority, @UserIdInt int userId)2730 public static SyncStatusInfo getSyncStatusAsUser(Account account, String authority, 2731 @UserIdInt int userId) { 2732 try { 2733 return getContentService().getSyncStatusAsUser(account, authority, null, userId); 2734 } catch (RemoteException e) { 2735 throw new RuntimeException("the ContentService should always be reachable", e); 2736 } 2737 } 2738 2739 /** 2740 * Return true if the pending status is true of any matching authorities. 2741 * <p>This method requires the caller to hold the permission 2742 * {@link android.Manifest.permission#READ_SYNC_STATS}. 2743 * @param account the account whose setting we are querying 2744 * @param authority the provider whose behavior is being queried 2745 * @return true if there is a pending sync with the matching account and authority 2746 */ isSyncPending(Account account, String authority)2747 public static boolean isSyncPending(Account account, String authority) { 2748 return isSyncPendingAsUser(account, authority, UserHandle.myUserId()); 2749 } 2750 2751 /** 2752 * @see #requestSync(Account, String, Bundle) 2753 * @hide 2754 */ isSyncPendingAsUser(Account account, String authority, @UserIdInt int userId)2755 public static boolean isSyncPendingAsUser(Account account, String authority, 2756 @UserIdInt int userId) { 2757 try { 2758 return getContentService().isSyncPendingAsUser(account, authority, null, userId); 2759 } catch (RemoteException e) { 2760 throw new RuntimeException("the ContentService should always be reachable", e); 2761 } 2762 } 2763 2764 /** 2765 * Request notifications when the different aspects of the SyncManager change. The 2766 * different items that can be requested are: 2767 * <ul> 2768 * <li> {@link #SYNC_OBSERVER_TYPE_PENDING} 2769 * <li> {@link #SYNC_OBSERVER_TYPE_ACTIVE} 2770 * <li> {@link #SYNC_OBSERVER_TYPE_SETTINGS} 2771 * </ul> 2772 * The caller can set one or more of the status types in the mask for any 2773 * given listener registration. 2774 * @param mask the status change types that will cause the callback to be invoked 2775 * @param callback observer to be invoked when the status changes 2776 * @return a handle that can be used to remove the listener at a later time 2777 */ addStatusChangeListener(int mask, final SyncStatusObserver callback)2778 public static Object addStatusChangeListener(int mask, final SyncStatusObserver callback) { 2779 if (callback == null) { 2780 throw new IllegalArgumentException("you passed in a null callback"); 2781 } 2782 try { 2783 ISyncStatusObserver.Stub observer = new ISyncStatusObserver.Stub() { 2784 @Override 2785 public void onStatusChanged(int which) throws RemoteException { 2786 callback.onStatusChanged(which); 2787 } 2788 }; 2789 getContentService().addStatusChangeListener(mask, observer); 2790 return observer; 2791 } catch (RemoteException e) { 2792 throw new RuntimeException("the ContentService should always be reachable", e); 2793 } 2794 } 2795 2796 /** 2797 * Remove a previously registered status change listener. 2798 * @param handle the handle that was returned by {@link #addStatusChangeListener} 2799 */ removeStatusChangeListener(Object handle)2800 public static void removeStatusChangeListener(Object handle) { 2801 if (handle == null) { 2802 throw new IllegalArgumentException("you passed in a null handle"); 2803 } 2804 try { 2805 getContentService().removeStatusChangeListener((ISyncStatusObserver.Stub) handle); 2806 } catch (RemoteException e) { 2807 // exception ignored; if this is thrown then it means the runtime is in the midst of 2808 // being restarted 2809 } 2810 } 2811 2812 /** {@hide} */ putCache(Uri key, Bundle value)2813 public void putCache(Uri key, Bundle value) { 2814 try { 2815 getContentService().putCache(mContext.getPackageName(), key, value, 2816 mContext.getUserId()); 2817 } catch (RemoteException e) { 2818 throw e.rethrowFromSystemServer(); 2819 } 2820 } 2821 2822 /** {@hide} */ getCache(Uri key)2823 public Bundle getCache(Uri key) { 2824 try { 2825 final Bundle bundle = getContentService().getCache(mContext.getPackageName(), key, 2826 mContext.getUserId()); 2827 if (bundle != null) bundle.setClassLoader(mContext.getClassLoader()); 2828 return bundle; 2829 } catch (RemoteException e) { 2830 throw e.rethrowFromSystemServer(); 2831 } 2832 } 2833 2834 /** {@hide} */ getTargetSdkVersion()2835 public int getTargetSdkVersion() { 2836 return mTargetSdkVersion; 2837 } 2838 2839 /** 2840 * Returns sampling percentage for a given duration. 2841 * 2842 * Always returns at least 1%. 2843 */ samplePercentForDuration(long durationMillis)2844 private int samplePercentForDuration(long durationMillis) { 2845 if (durationMillis >= SLOW_THRESHOLD_MILLIS) { 2846 return 100; 2847 } 2848 return (int) (100 * durationMillis / SLOW_THRESHOLD_MILLIS) + 1; 2849 } 2850 maybeLogQueryToEventLog( long durationMillis, Uri uri, String[] projection, @Nullable Bundle queryArgs)2851 private void maybeLogQueryToEventLog( 2852 long durationMillis, Uri uri, String[] projection, @Nullable Bundle queryArgs) { 2853 if (!ENABLE_CONTENT_SAMPLE) return; 2854 int samplePercent = samplePercentForDuration(durationMillis); 2855 if (samplePercent < 100) { 2856 synchronized (mRandom) { 2857 if (mRandom.nextInt(100) >= samplePercent) { 2858 return; 2859 } 2860 } 2861 } 2862 2863 // Ensure a non-null bundle. 2864 queryArgs = (queryArgs != null) ? queryArgs : Bundle.EMPTY; 2865 2866 StringBuilder projectionBuffer = new StringBuilder(100); 2867 if (projection != null) { 2868 for (int i = 0; i < projection.length; ++i) { 2869 // Note: not using a comma delimiter here, as the 2870 // multiple arguments to EventLog.writeEvent later 2871 // stringify with a comma delimiter, which would make 2872 // parsing uglier later. 2873 if (i != 0) projectionBuffer.append('/'); 2874 projectionBuffer.append(projection[i]); 2875 } 2876 } 2877 2878 // ActivityThread.currentPackageName() only returns non-null if the 2879 // current thread is an application main thread. This parameter tells 2880 // us whether an event loop is blocked, and if so, which app it is. 2881 String blockingPackage = AppGlobals.getInitialPackage(); 2882 2883 EventLog.writeEvent( 2884 EventLogTags.CONTENT_QUERY_SAMPLE, 2885 uri.toString(), 2886 projectionBuffer.toString(), 2887 queryArgs.getString(QUERY_ARG_SQL_SELECTION, ""), 2888 queryArgs.getString(QUERY_ARG_SQL_SORT_ORDER, ""), 2889 durationMillis, 2890 blockingPackage != null ? blockingPackage : "", 2891 samplePercent); 2892 } 2893 maybeLogUpdateToEventLog( long durationMillis, Uri uri, String operation, String selection)2894 private void maybeLogUpdateToEventLog( 2895 long durationMillis, Uri uri, String operation, String selection) { 2896 if (!ENABLE_CONTENT_SAMPLE) return; 2897 int samplePercent = samplePercentForDuration(durationMillis); 2898 if (samplePercent < 100) { 2899 synchronized (mRandom) { 2900 if (mRandom.nextInt(100) >= samplePercent) { 2901 return; 2902 } 2903 } 2904 } 2905 String blockingPackage = AppGlobals.getInitialPackage(); 2906 EventLog.writeEvent( 2907 EventLogTags.CONTENT_UPDATE_SAMPLE, 2908 uri.toString(), 2909 operation, 2910 selection != null ? selection : "", 2911 durationMillis, 2912 blockingPackage != null ? blockingPackage : "", 2913 samplePercent); 2914 } 2915 2916 private final class CursorWrapperInner extends CrossProcessCursorWrapper { 2917 private final IContentProvider mContentProvider; 2918 private final AtomicBoolean mProviderReleased = new AtomicBoolean(); 2919 2920 private final CloseGuard mCloseGuard = CloseGuard.get(); 2921 CursorWrapperInner(Cursor cursor, IContentProvider contentProvider)2922 CursorWrapperInner(Cursor cursor, IContentProvider contentProvider) { 2923 super(cursor); 2924 mContentProvider = contentProvider; 2925 mCloseGuard.open("close"); 2926 } 2927 2928 @Override close()2929 public void close() { 2930 mCloseGuard.close(); 2931 super.close(); 2932 2933 if (mProviderReleased.compareAndSet(false, true)) { 2934 ContentResolver.this.releaseProvider(mContentProvider); 2935 } 2936 } 2937 2938 @Override finalize()2939 protected void finalize() throws Throwable { 2940 try { 2941 mCloseGuard.warnIfOpen(); 2942 close(); 2943 } finally { 2944 super.finalize(); 2945 } 2946 } 2947 } 2948 2949 private final class ParcelFileDescriptorInner extends ParcelFileDescriptor { 2950 private final IContentProvider mContentProvider; 2951 private final AtomicBoolean mProviderReleased = new AtomicBoolean(); 2952 ParcelFileDescriptorInner(ParcelFileDescriptor pfd, IContentProvider icp)2953 ParcelFileDescriptorInner(ParcelFileDescriptor pfd, IContentProvider icp) { 2954 super(pfd); 2955 mContentProvider = icp; 2956 } 2957 2958 @Override releaseResources()2959 public void releaseResources() { 2960 if (mProviderReleased.compareAndSet(false, true)) { 2961 ContentResolver.this.releaseProvider(mContentProvider); 2962 } 2963 } 2964 } 2965 2966 /** @hide */ 2967 public static final String CONTENT_SERVICE_NAME = "content"; 2968 2969 /** @hide */ getContentService()2970 public static IContentService getContentService() { 2971 if (sContentService != null) { 2972 return sContentService; 2973 } 2974 IBinder b = ServiceManager.getService(CONTENT_SERVICE_NAME); 2975 if (false) Log.v("ContentService", "default service binder = " + b); 2976 sContentService = IContentService.Stub.asInterface(b); 2977 if (false) Log.v("ContentService", "default service = " + sContentService); 2978 return sContentService; 2979 } 2980 2981 /** @hide */ getPackageName()2982 public String getPackageName() { 2983 return mPackageName; 2984 } 2985 2986 private static IContentService sContentService; 2987 private final Context mContext; 2988 2989 final String mPackageName; 2990 final int mTargetSdkVersion; 2991 2992 private static final String TAG = "ContentResolver"; 2993 2994 /** @hide */ resolveUserId(Uri uri)2995 public int resolveUserId(Uri uri) { 2996 return ContentProvider.getUserIdFromUri(uri, mContext.getUserId()); 2997 } 2998 2999 /** @hide */ getTypeDrawable(String mimeType)3000 public Drawable getTypeDrawable(String mimeType) { 3001 return MimeIconUtils.loadMimeIcon(mContext, mimeType); 3002 } 3003 3004 /** 3005 * @hide 3006 */ createSqlQueryBundle( @ullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder)3007 public static @Nullable Bundle createSqlQueryBundle( 3008 @Nullable String selection, 3009 @Nullable String[] selectionArgs, 3010 @Nullable String sortOrder) { 3011 3012 if (selection == null && selectionArgs == null && sortOrder == null) { 3013 return null; 3014 } 3015 3016 Bundle queryArgs = new Bundle(); 3017 if (selection != null) { 3018 queryArgs.putString(QUERY_ARG_SQL_SELECTION, selection); 3019 } 3020 if (selectionArgs != null) { 3021 queryArgs.putStringArray(QUERY_ARG_SQL_SELECTION_ARGS, selectionArgs); 3022 } 3023 if (sortOrder != null) { 3024 queryArgs.putString(QUERY_ARG_SQL_SORT_ORDER, sortOrder); 3025 } 3026 return queryArgs; 3027 } 3028 3029 /** 3030 * Returns structured sort args formatted as an SQL sort clause. 3031 * 3032 * NOTE: Collator clauses are suitable for use with non text fields. We might 3033 * choose to omit any collation clause since we don't know the underlying 3034 * type of data to be collated. Imperical testing shows that sqlite3 doesn't 3035 * appear to care much about the presence of collate clauses in queries 3036 * when ordering by numeric fields. For this reason we include collate 3037 * clause unilaterally when {@link #QUERY_ARG_SORT_COLLATION} is present 3038 * in query args bundle. 3039 * 3040 * TODO: Would be nice to explicitly validate that colums referenced in 3041 * {@link #QUERY_ARG_SORT_COLUMNS} are present in the associated projection. 3042 * 3043 * @hide 3044 */ createSqlSortClause(Bundle queryArgs)3045 public static String createSqlSortClause(Bundle queryArgs) { 3046 String[] columns = queryArgs.getStringArray(QUERY_ARG_SORT_COLUMNS); 3047 if (columns == null || columns.length == 0) { 3048 throw new IllegalArgumentException("Can't create sort clause without columns."); 3049 } 3050 3051 String query = TextUtils.join(", ", columns); 3052 3053 // Interpret PRIMARY and SECONDARY collation strength as no-case collation based 3054 // on their javadoc descriptions. 3055 int collation = queryArgs.getInt( 3056 ContentResolver.QUERY_ARG_SORT_COLLATION, java.text.Collator.IDENTICAL); 3057 if (collation == java.text.Collator.PRIMARY || collation == java.text.Collator.SECONDARY) { 3058 query += " COLLATE NOCASE"; 3059 } 3060 3061 int sortDir = queryArgs.getInt(QUERY_ARG_SORT_DIRECTION, Integer.MIN_VALUE); 3062 if (sortDir != Integer.MIN_VALUE) { 3063 switch (sortDir) { 3064 case QUERY_SORT_DIRECTION_ASCENDING: 3065 query += " ASC"; 3066 break; 3067 case QUERY_SORT_DIRECTION_DESCENDING: 3068 query += " DESC"; 3069 break; 3070 default: 3071 throw new IllegalArgumentException("Unsupported sort direction value." 3072 + " See ContentResolver documentation for details."); 3073 } 3074 } 3075 return query; 3076 } 3077 } 3078