1 /* 2 * Copyright (C) 2010 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 com.android.server.content; 18 19 import android.accounts.Account; 20 import android.app.job.JobInfo; 21 import android.content.ContentResolver; 22 import android.content.ContentResolver.SyncExemption; 23 import android.content.pm.PackageManager; 24 import android.os.Bundle; 25 import android.os.PersistableBundle; 26 import android.os.SystemClock; 27 import android.os.UserHandle; 28 import android.util.Slog; 29 30 /** 31 * Value type that represents a sync operation. 32 * This holds all information related to a sync operation - both one off and periodic. 33 * Data stored in this is used to schedule a job with the JobScheduler. 34 * {@hide} 35 */ 36 public class SyncOperation { 37 public static final String TAG = "SyncManager"; 38 39 /** 40 * This is used in the {@link #sourcePeriodicId} field if the operation is not initiated by a failed 41 * periodic sync. 42 */ 43 public static final int NO_JOB_ID = -1; 44 45 public static final int REASON_BACKGROUND_DATA_SETTINGS_CHANGED = -1; 46 public static final int REASON_ACCOUNTS_UPDATED = -2; 47 public static final int REASON_SERVICE_CHANGED = -3; 48 public static final int REASON_PERIODIC = -4; 49 /** Sync started because it has just been set to isSyncable. */ 50 public static final int REASON_IS_SYNCABLE = -5; 51 /** Sync started because it has just been set to sync automatically. */ 52 public static final int REASON_SYNC_AUTO = -6; 53 /** Sync started because global sync automatically has been set to true. */ 54 public static final int REASON_MASTER_SYNC_AUTO = -7; 55 public static final int REASON_USER_START = -8; 56 57 private static String[] REASON_NAMES = new String[] { 58 "DataSettingsChanged", 59 "AccountsUpdated", 60 "ServiceChanged", 61 "Periodic", 62 "IsSyncable", 63 "AutoSync", 64 "MasterSyncAuto", 65 "UserStart", 66 }; 67 68 /** Identifying info for the target for this operation. */ 69 public final SyncStorageEngine.EndPoint target; 70 public final int owningUid; 71 public final String owningPackage; 72 /** Why this sync was kicked off. {@link #REASON_NAMES} */ 73 public final int reason; 74 /** Where this sync was initiated. */ 75 public final int syncSource; 76 public final boolean allowParallelSyncs; 77 78 /** 79 * Sync extras. Note, DO NOT modify this bundle directly. When changing the content, always 80 * create a copy, update it, set it in this field. This is to avoid concurrent modifications 81 * when other threads are reading it. 82 */ 83 private volatile Bundle mImmutableExtras; 84 85 public final boolean isPeriodic; 86 /** jobId of the periodic SyncOperation that initiated this one */ 87 public final int sourcePeriodicId; 88 /** Operations are considered duplicates if keys are equal */ 89 public final String key; 90 91 /** Poll frequency of periodic sync in milliseconds */ 92 public final long periodMillis; 93 /** Flex time of periodic sync in milliseconds */ 94 public final long flexMillis; 95 /** Descriptive string key for this operation */ 96 public String wakeLockName; 97 /** 98 * Used when duplicate pending syncs are present. The one with the lowest expectedRuntime 99 * is kept, others are discarded. 100 */ 101 public long expectedRuntime; 102 103 /** Stores the number of times this sync operation failed and had to be retried. */ 104 int retries; 105 106 /** 107 * Indicates if a sync that was originally scheduled as an EJ is being re-scheduled as a 108 * regular job. Specifically, this will be {@code true} if a sync is being backed-off but 109 * {@link ContentResolver#SYNC_EXTRAS_IGNORE_BACKOFF} is not set. 110 */ 111 boolean scheduleEjAsRegularJob; 112 113 /** jobId of the JobScheduler job corresponding to this sync */ 114 public int jobId; 115 116 @SyncExemption 117 public int syncExemptionFlag; 118 SyncOperation(Account account, int userId, int owningUid, String owningPackage, int reason, int source, String provider, Bundle extras, boolean allowParallelSyncs, @SyncExemption int syncExemptionFlag)119 public SyncOperation(Account account, int userId, int owningUid, String owningPackage, 120 int reason, int source, String provider, Bundle extras, 121 boolean allowParallelSyncs, @SyncExemption int syncExemptionFlag) { 122 this(new SyncStorageEngine.EndPoint(account, provider, userId), owningUid, owningPackage, 123 reason, source, extras, allowParallelSyncs, syncExemptionFlag); 124 } 125 SyncOperation(SyncStorageEngine.EndPoint info, int owningUid, String owningPackage, int reason, int source, Bundle extras, boolean allowParallelSyncs, @SyncExemption int syncExemptionFlag)126 private SyncOperation(SyncStorageEngine.EndPoint info, int owningUid, String owningPackage, 127 int reason, int source, Bundle extras, boolean allowParallelSyncs, 128 @SyncExemption int syncExemptionFlag) { 129 this(info, owningUid, owningPackage, reason, source, extras, allowParallelSyncs, false, 130 NO_JOB_ID, 0, 0, syncExemptionFlag); 131 } 132 SyncOperation(SyncOperation op, long periodMillis, long flexMillis)133 public SyncOperation(SyncOperation op, long periodMillis, long flexMillis) { 134 this(op.target, op.owningUid, op.owningPackage, op.reason, op.syncSource, 135 op.mImmutableExtras, op.allowParallelSyncs, op.isPeriodic, op.sourcePeriodicId, 136 periodMillis, flexMillis, ContentResolver.SYNC_EXEMPTION_NONE); 137 } 138 SyncOperation(SyncStorageEngine.EndPoint info, int owningUid, String owningPackage, int reason, int source, Bundle extras, boolean allowParallelSyncs, boolean isPeriodic, int sourcePeriodicId, long periodMillis, long flexMillis, @SyncExemption int syncExemptionFlag)139 public SyncOperation(SyncStorageEngine.EndPoint info, int owningUid, String owningPackage, 140 int reason, int source, Bundle extras, 141 boolean allowParallelSyncs, 142 boolean isPeriodic, int sourcePeriodicId, long periodMillis, 143 long flexMillis, @SyncExemption int syncExemptionFlag) { 144 this.target = info; 145 this.owningUid = owningUid; 146 this.owningPackage = owningPackage; 147 this.reason = reason; 148 this.syncSource = source; 149 this.mImmutableExtras = new Bundle(extras); 150 this.allowParallelSyncs = allowParallelSyncs; 151 this.isPeriodic = isPeriodic; 152 this.sourcePeriodicId = sourcePeriodicId; 153 this.periodMillis = periodMillis; 154 this.flexMillis = flexMillis; 155 this.jobId = NO_JOB_ID; 156 this.key = toKey(); 157 this.syncExemptionFlag = syncExemptionFlag; 158 } 159 160 /* Get a one off sync operation instance from a periodic sync. */ createOneTimeSyncOperation()161 public SyncOperation createOneTimeSyncOperation() { 162 if (!isPeriodic) { 163 return null; 164 } 165 SyncOperation op = new SyncOperation(target, owningUid, owningPackage, reason, syncSource, 166 mImmutableExtras, allowParallelSyncs, false, jobId /* sourcePeriodicId */, 167 periodMillis, flexMillis, ContentResolver.SYNC_EXEMPTION_NONE); 168 return op; 169 } 170 SyncOperation(SyncOperation other)171 public SyncOperation(SyncOperation other) { 172 target = other.target; 173 owningUid = other.owningUid; 174 owningPackage = other.owningPackage; 175 reason = other.reason; 176 syncSource = other.syncSource; 177 allowParallelSyncs = other.allowParallelSyncs; 178 179 // Since we treat this field as immutable, it's okay to use a shallow copy here. 180 // No need to create a copy. 181 mImmutableExtras = other.mImmutableExtras; 182 wakeLockName = other.wakeLockName(); 183 isPeriodic = other.isPeriodic; 184 sourcePeriodicId = other.sourcePeriodicId; 185 periodMillis = other.periodMillis; 186 flexMillis = other.flexMillis; 187 this.key = other.key; 188 syncExemptionFlag = other.syncExemptionFlag; 189 } 190 191 /** 192 * All fields are stored in a corresponding key in the persistable bundle. 193 * 194 * {@link #mImmutableExtras} is a Bundle and can contain parcelable objects. 195 * But only the type Account 196 * is allowed {@link ContentResolver#validateSyncExtrasBundle(Bundle)} that can't be stored in 197 * a PersistableBundle. For every value of type Account with key 'key', we store a 198 * PersistableBundle containing account information at key 'ACCOUNT:key'. The Account object 199 * can be reconstructed using this. 200 * 201 * We put a flag with key 'SyncManagerJob', to identify while reconstructing a sync operation 202 * from a bundle whether the bundle actually contains information about a sync. 203 * @return A persistable bundle containing all information to re-construct the sync operation. 204 */ toJobInfoExtras()205 PersistableBundle toJobInfoExtras() { 206 // This will be passed as extras bundle to a JobScheduler job. 207 PersistableBundle jobInfoExtras = new PersistableBundle(); 208 209 PersistableBundle syncExtrasBundle = new PersistableBundle(); 210 211 final Bundle extras = mImmutableExtras; 212 for (String key : extras.keySet()) { 213 Object value = extras.get(key); 214 if (value instanceof Account) { 215 Account account = (Account) value; 216 PersistableBundle accountBundle = new PersistableBundle(); 217 accountBundle.putString("accountName", account.name); 218 accountBundle.putString("accountType", account.type); 219 // This is stored in jobInfoExtras so that we don't override a user specified 220 // sync extra with the same key. 221 jobInfoExtras.putPersistableBundle("ACCOUNT:" + key, accountBundle); 222 } else if (value instanceof Long) { 223 syncExtrasBundle.putLong(key, (Long) value); 224 } else if (value instanceof Integer) { 225 syncExtrasBundle.putInt(key, (Integer) value); 226 } else if (value instanceof Boolean) { 227 syncExtrasBundle.putBoolean(key, (Boolean) value); 228 } else if (value instanceof Float) { 229 syncExtrasBundle.putDouble(key, (double) (float) value); 230 } else if (value instanceof Double) { 231 syncExtrasBundle.putDouble(key, (Double) value); 232 } else if (value instanceof String) { 233 syncExtrasBundle.putString(key, (String) value); 234 } else if (value == null) { 235 syncExtrasBundle.putString(key, null); 236 } else { 237 Slog.e(TAG, "Unknown extra type."); 238 } 239 } 240 jobInfoExtras.putPersistableBundle("syncExtras", syncExtrasBundle); 241 242 jobInfoExtras.putBoolean("SyncManagerJob", true); 243 244 jobInfoExtras.putString("provider", target.provider); 245 jobInfoExtras.putString("accountName", target.account.name); 246 jobInfoExtras.putString("accountType", target.account.type); 247 jobInfoExtras.putInt("userId", target.userId); 248 jobInfoExtras.putInt("owningUid", owningUid); 249 jobInfoExtras.putString("owningPackage", owningPackage); 250 jobInfoExtras.putInt("reason", reason); 251 jobInfoExtras.putInt("source", syncSource); 252 jobInfoExtras.putBoolean("allowParallelSyncs", allowParallelSyncs); 253 jobInfoExtras.putInt("jobId", jobId); 254 jobInfoExtras.putBoolean("isPeriodic", isPeriodic); 255 jobInfoExtras.putInt("sourcePeriodicId", sourcePeriodicId); 256 jobInfoExtras.putLong("periodMillis", periodMillis); 257 jobInfoExtras.putLong("flexMillis", flexMillis); 258 jobInfoExtras.putLong("expectedRuntime", expectedRuntime); 259 jobInfoExtras.putInt("retries", retries); 260 jobInfoExtras.putInt("syncExemptionFlag", syncExemptionFlag); 261 jobInfoExtras.putBoolean("ejDowngradedToRegular", scheduleEjAsRegularJob); 262 return jobInfoExtras; 263 } 264 265 /** 266 * Reconstructs a sync operation from an extras Bundle. Returns null if the bundle doesn't 267 * contain a valid sync operation. 268 */ maybeCreateFromJobExtras(PersistableBundle jobExtras)269 static SyncOperation maybeCreateFromJobExtras(PersistableBundle jobExtras) { 270 if (jobExtras == null) { 271 return null; 272 } 273 String accountName, accountType; 274 String provider; 275 int userId, owningUid; 276 String owningPackage; 277 int reason, source; 278 int initiatedBy; 279 Bundle extras; 280 boolean allowParallelSyncs, isPeriodic; 281 long periodMillis, flexMillis; 282 int syncExemptionFlag; 283 284 if (!jobExtras.getBoolean("SyncManagerJob", false)) { 285 return null; 286 } 287 288 accountName = jobExtras.getString("accountName"); 289 accountType = jobExtras.getString("accountType"); 290 provider = jobExtras.getString("provider"); 291 userId = jobExtras.getInt("userId", Integer.MAX_VALUE); 292 owningUid = jobExtras.getInt("owningUid"); 293 owningPackage = jobExtras.getString("owningPackage"); 294 reason = jobExtras.getInt("reason", Integer.MAX_VALUE); 295 source = jobExtras.getInt("source", Integer.MAX_VALUE); 296 allowParallelSyncs = jobExtras.getBoolean("allowParallelSyncs", false); 297 isPeriodic = jobExtras.getBoolean("isPeriodic", false); 298 initiatedBy = jobExtras.getInt("sourcePeriodicId", NO_JOB_ID); 299 periodMillis = jobExtras.getLong("periodMillis"); 300 flexMillis = jobExtras.getLong("flexMillis"); 301 syncExemptionFlag = jobExtras.getInt("syncExemptionFlag", 302 ContentResolver.SYNC_EXEMPTION_NONE); 303 extras = new Bundle(); 304 305 PersistableBundle syncExtras = jobExtras.getPersistableBundle("syncExtras"); 306 if (syncExtras != null) { 307 extras.putAll(syncExtras); 308 } 309 310 for (String key: jobExtras.keySet()) { 311 if (key!= null && key.startsWith("ACCOUNT:")) { 312 String newKey = key.substring(8); // Strip off the 'ACCOUNT:' prefix. 313 PersistableBundle accountsBundle = jobExtras.getPersistableBundle(key); 314 Account account = new Account(accountsBundle.getString("accountName"), 315 accountsBundle.getString("accountType")); 316 extras.putParcelable(newKey, account); 317 } 318 } 319 320 Account account = new Account(accountName, accountType); 321 SyncStorageEngine.EndPoint target = 322 new SyncStorageEngine.EndPoint(account, provider, userId); 323 SyncOperation op = new SyncOperation(target, owningUid, owningPackage, reason, source, 324 extras, allowParallelSyncs, isPeriodic, initiatedBy, periodMillis, flexMillis, 325 syncExemptionFlag); 326 op.jobId = jobExtras.getInt("jobId"); 327 op.expectedRuntime = jobExtras.getLong("expectedRuntime"); 328 op.retries = jobExtras.getInt("retries"); 329 op.scheduleEjAsRegularJob = jobExtras.getBoolean("ejDowngradedToRegular"); 330 return op; 331 } 332 333 /** 334 * Determine whether if this sync operation is running, the provided operation would conflict 335 * with it. 336 * Parallel syncs allow multiple accounts to be synced at the same time. 337 */ isConflict(SyncOperation toRun)338 boolean isConflict(SyncOperation toRun) { 339 final SyncStorageEngine.EndPoint other = toRun.target; 340 return target.account.type.equals(other.account.type) 341 && target.provider.equals(other.provider) 342 && target.userId == other.userId 343 && (!allowParallelSyncs 344 || target.account.name.equals(other.account.name)); 345 } 346 isReasonPeriodic()347 boolean isReasonPeriodic() { 348 return reason == REASON_PERIODIC; 349 } 350 matchesPeriodicOperation(SyncOperation other)351 boolean matchesPeriodicOperation(SyncOperation other) { 352 return target.matchesSpec(other.target) 353 && SyncManager.syncExtrasEquals(mImmutableExtras, other.mImmutableExtras, true) 354 && periodMillis == other.periodMillis && flexMillis == other.flexMillis; 355 } 356 isDerivedFromFailedPeriodicSync()357 boolean isDerivedFromFailedPeriodicSync() { 358 return sourcePeriodicId != NO_JOB_ID; 359 } 360 getJobBias()361 int getJobBias() { 362 if (isInitialization()) { 363 return JobInfo.BIAS_SYNC_INITIALIZATION; 364 } else if (isExpedited()) { 365 return JobInfo.BIAS_SYNC_EXPEDITED; 366 } 367 return JobInfo.BIAS_DEFAULT; 368 } 369 toKey()370 private String toKey() { 371 final Bundle extras = mImmutableExtras; 372 StringBuilder sb = new StringBuilder(); 373 sb.append("provider: ").append(target.provider); 374 sb.append(" account {name=" + target.account.name 375 + ", user=" 376 + target.userId 377 + ", type=" 378 + target.account.type 379 + "}"); 380 sb.append(" isPeriodic: ").append(isPeriodic); 381 sb.append(" period: ").append(periodMillis); 382 sb.append(" flex: ").append(flexMillis); 383 sb.append(" extras: "); 384 extrasToStringBuilder(extras, sb); 385 return sb.toString(); 386 } 387 388 @Override toString()389 public String toString() { 390 return dump(null, true, null, false); 391 } 392 toSafeString()393 public String toSafeString() { 394 return dump(null, true, null, true); 395 } 396 dump(PackageManager pm, boolean shorter, SyncAdapterStateFetcher appStates, boolean logSafe)397 String dump(PackageManager pm, boolean shorter, SyncAdapterStateFetcher appStates, 398 boolean logSafe) { 399 final Bundle extras = mImmutableExtras; 400 StringBuilder sb = new StringBuilder(); 401 sb.append("JobId=").append(jobId) 402 .append(" ") 403 .append(logSafe ? "***" : target.account.name) 404 .append("/") 405 .append(target.account.type) 406 .append(" u") 407 .append(target.userId) 408 .append(" [") 409 .append(target.provider) 410 .append("] "); 411 sb.append(SyncStorageEngine.SOURCES[syncSource]); 412 if (expectedRuntime != 0) { 413 sb.append(" ExpectedIn="); 414 SyncManager.formatDurationHMS(sb, 415 (expectedRuntime - SystemClock.elapsedRealtime())); 416 } 417 if (extras.getBoolean(ContentResolver.SYNC_EXTRAS_EXPEDITED, false)) { 418 sb.append(" EXPEDITED"); 419 } 420 if (extras.getBoolean(ContentResolver.SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB, false)) { 421 sb.append(" EXPEDITED-JOB"); 422 if (scheduleEjAsRegularJob) { 423 sb.append("(scheduled-as-regular)"); 424 } 425 } 426 switch (syncExemptionFlag) { 427 case ContentResolver.SYNC_EXEMPTION_NONE: 428 break; 429 case ContentResolver.SYNC_EXEMPTION_PROMOTE_BUCKET: 430 sb.append(" STANDBY-EXEMPTED"); 431 break; 432 case ContentResolver.SYNC_EXEMPTION_PROMOTE_BUCKET_WITH_TEMP: 433 sb.append(" STANDBY-EXEMPTED(TOP)"); 434 break; 435 default: 436 sb.append(" ExemptionFlag=" + syncExemptionFlag); 437 break; 438 } 439 sb.append(" Reason="); 440 sb.append(reasonToString(pm, reason)); 441 if (isPeriodic) { 442 sb.append(" (period="); 443 SyncManager.formatDurationHMS(sb, periodMillis); 444 sb.append(" flex="); 445 SyncManager.formatDurationHMS(sb, flexMillis); 446 sb.append(")"); 447 } 448 if (retries > 0) { 449 sb.append(" Retries="); 450 sb.append(retries); 451 } 452 if (!shorter) { 453 sb.append(" Owner={"); 454 UserHandle.formatUid(sb, owningUid); 455 sb.append(" "); 456 sb.append(owningPackage); 457 if (appStates != null) { 458 sb.append(" ["); 459 sb.append(appStates.getStandbyBucket( 460 UserHandle.getUserId(owningUid), owningPackage)); 461 sb.append("]"); 462 463 if (appStates.isAppActive(owningUid)) { 464 sb.append(" [ACTIVE]"); 465 } 466 } 467 sb.append("}"); 468 if (!extras.keySet().isEmpty()) { 469 sb.append(" "); 470 extrasToStringBuilder(extras, sb); 471 } 472 } 473 return sb.toString(); 474 } 475 reasonToString(PackageManager pm, int reason)476 static String reasonToString(PackageManager pm, int reason) { 477 if (reason >= 0) { 478 if (pm != null) { 479 final String[] packages = pm.getPackagesForUid(reason); 480 if (packages != null && packages.length == 1) { 481 return packages[0]; 482 } 483 final String name = pm.getNameForUid(reason); 484 if (name != null) { 485 return name; 486 } 487 return String.valueOf(reason); 488 } else { 489 return String.valueOf(reason); 490 } 491 } else { 492 final int index = -reason - 1; 493 if (index >= REASON_NAMES.length) { 494 return String.valueOf(reason); 495 } else { 496 return REASON_NAMES[index]; 497 } 498 } 499 } 500 isInitialization()501 boolean isInitialization() { 502 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_INITIALIZE, false); 503 } 504 isExpedited()505 boolean isExpedited() { 506 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_EXPEDITED, false); 507 } 508 isUpload()509 boolean isUpload() { 510 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_UPLOAD, false); 511 } 512 513 /** 514 * Disable SYNC_EXTRAS_UPLOAD, so it will be a two-way (normal) sync. 515 */ enableTwoWaySync()516 void enableTwoWaySync() { 517 removeExtra(ContentResolver.SYNC_EXTRAS_UPLOAD); 518 } 519 hasIgnoreBackoff()520 boolean hasIgnoreBackoff() { 521 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF, false); 522 } 523 524 /** 525 * Disable SYNC_EXTRAS_IGNORE_BACKOFF. 526 * 527 * The SYNC_EXTRAS_IGNORE_BACKOFF only applies to the first attempt to sync a given 528 * request. Retries of the request will always honor the backoff, so clear the 529 * flag in case we retry this request. 530 */ enableBackoff()531 void enableBackoff() { 532 removeExtra(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF); 533 } 534 hasDoNotRetry()535 boolean hasDoNotRetry() { 536 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_DO_NOT_RETRY, false); 537 } 538 isNotAllowedOnMetered()539 boolean isNotAllowedOnMetered() { 540 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_DISALLOW_METERED, false); 541 } 542 isManual()543 boolean isManual() { 544 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, false); 545 } 546 isIgnoreSettings()547 boolean isIgnoreSettings() { 548 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_SETTINGS, false); 549 } 550 hasRequireCharging()551 boolean hasRequireCharging() { 552 return mImmutableExtras.getBoolean(ContentResolver.SYNC_EXTRAS_REQUIRE_CHARGING, false); 553 } 554 isScheduledAsExpeditedJob()555 boolean isScheduledAsExpeditedJob() { 556 return mImmutableExtras.getBoolean( 557 ContentResolver.SYNC_EXTRAS_SCHEDULE_AS_EXPEDITED_JOB, false); 558 } 559 isAppStandbyExempted()560 boolean isAppStandbyExempted() { 561 return syncExemptionFlag != ContentResolver.SYNC_EXEMPTION_NONE; 562 } 563 areExtrasEqual(Bundle other, boolean includeSyncSettings)564 boolean areExtrasEqual(Bundle other, boolean includeSyncSettings) { 565 return SyncManager.syncExtrasEquals(mImmutableExtras, other, includeSyncSettings); 566 } 567 extrasToStringBuilder(Bundle bundle, StringBuilder sb)568 static void extrasToStringBuilder(Bundle bundle, StringBuilder sb) { 569 if (bundle == null) { 570 sb.append("null"); 571 return; 572 } 573 sb.append("["); 574 for (String key : bundle.keySet()) { 575 sb.append(key).append("=").append(bundle.get(key)).append(" "); 576 } 577 sb.append("]"); 578 } 579 extrasToString(Bundle bundle)580 private static String extrasToString(Bundle bundle) { 581 final StringBuilder sb = new StringBuilder(); 582 extrasToStringBuilder(bundle, sb); 583 return sb.toString(); 584 } 585 wakeLockName()586 String wakeLockName() { 587 if (wakeLockName != null) { 588 return wakeLockName; 589 } 590 return (wakeLockName = target.provider 591 + "/" + target.account.type); 592 } 593 594 // TODO: Test this to make sure that casting to object doesn't lose the type info for EventLog. toEventLog(int event)595 public Object[] toEventLog(int event) { 596 Object[] logArray = new Object[4]; 597 logArray[1] = event; 598 logArray[2] = syncSource; 599 logArray[0] = target.provider; 600 logArray[3] = target.account.name.hashCode(); 601 return logArray; 602 } 603 604 /** 605 * Removes a sync extra. Note do not call it from multiple threads simultaneously. 606 */ removeExtra(String key)607 private void removeExtra(String key) { 608 final Bundle b = mImmutableExtras; 609 if (!b.containsKey(key)) { 610 return; 611 } 612 final Bundle clone = new Bundle(b); 613 clone.remove(key); 614 mImmutableExtras = clone; 615 } 616 getClonedExtras()617 public Bundle getClonedExtras() { 618 return new Bundle(mImmutableExtras); 619 } 620 getExtrasAsString()621 public String getExtrasAsString() { 622 return extrasToString(mImmutableExtras); 623 } 624 } 625