fix bug that was stopping any sync service sync from from taking place. Change-Id: Ib3f46d43be630d4196fbf0ca32da03b6e7a6a5b9
275 lines
11 KiB
Java
275 lines
11 KiB
Java
/*
|
|
* Copyright (C) 2010 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.android.server.content;
|
|
|
|
import android.content.pm.PackageManager;
|
|
import android.content.SyncAdapterType;
|
|
import android.content.SyncAdaptersCache;
|
|
import android.content.pm.RegisteredServicesCache.ServiceInfo;
|
|
import android.os.Bundle;
|
|
import android.os.SystemClock;
|
|
import android.text.format.DateUtils;
|
|
import android.util.Log;
|
|
import android.util.Pair;
|
|
|
|
import com.google.android.collect.Maps;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Collection;
|
|
import java.util.HashMap;
|
|
import java.util.Iterator;
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* Queue of pending sync operations. Not inherently thread safe, external
|
|
* callers are responsible for locking.
|
|
*
|
|
* @hide
|
|
*/
|
|
public class SyncQueue {
|
|
private static final String TAG = "SyncManager";
|
|
private final SyncStorageEngine mSyncStorageEngine;
|
|
private final SyncAdaptersCache mSyncAdapters;
|
|
private final PackageManager mPackageManager;
|
|
|
|
// A Map of SyncOperations operationKey -> SyncOperation that is designed for
|
|
// quick lookup of an enqueued SyncOperation.
|
|
private final HashMap<String, SyncOperation> mOperationsMap = Maps.newHashMap();
|
|
|
|
public SyncQueue(PackageManager packageManager, SyncStorageEngine syncStorageEngine,
|
|
final SyncAdaptersCache syncAdapters) {
|
|
mPackageManager = packageManager;
|
|
mSyncStorageEngine = syncStorageEngine;
|
|
mSyncAdapters = syncAdapters;
|
|
}
|
|
|
|
public void addPendingOperations(int userId) {
|
|
for (SyncStorageEngine.PendingOperation op : mSyncStorageEngine.getPendingOperations()) {
|
|
final SyncStorageEngine.EndPoint info = op.target;
|
|
if (info.userId != userId) continue;
|
|
|
|
final Pair<Long, Long> backoff = mSyncStorageEngine.getBackoff(info);
|
|
SyncOperation operationToAdd;
|
|
if (info.target_provider) {
|
|
final ServiceInfo<SyncAdapterType> syncAdapterInfo = mSyncAdapters.getServiceInfo(
|
|
SyncAdapterType.newKey(info.provider, info.account.type), info.userId);
|
|
if (syncAdapterInfo == null) {
|
|
if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
|
Log.v(TAG, "Missing sync adapter info for authority " + op.target);
|
|
}
|
|
continue;
|
|
}
|
|
operationToAdd = new SyncOperation(
|
|
info.account, info.userId, op.reason, op.syncSource, info.provider,
|
|
op.extras,
|
|
0 /* delay */,
|
|
0 /* flex */,
|
|
backoff != null ? backoff.first : 0,
|
|
mSyncStorageEngine.getDelayUntilTime(info),
|
|
syncAdapterInfo.type.allowParallelSyncs());
|
|
operationToAdd.expedited = op.expedited;
|
|
operationToAdd.pendingOperation = op;
|
|
add(operationToAdd, op);
|
|
} else if (info.target_service) {
|
|
try {
|
|
mPackageManager.getServiceInfo(info.service, 0);
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
|
Log.w(TAG, "Missing sync service for authority " + op.target);
|
|
}
|
|
continue;
|
|
}
|
|
operationToAdd = new SyncOperation(
|
|
info.service, info.userId, op.reason, op.syncSource,
|
|
op.extras,
|
|
0 /* delay */,
|
|
0 /* flex */,
|
|
backoff != null ? backoff.first : 0,
|
|
mSyncStorageEngine.getDelayUntilTime(info));
|
|
operationToAdd.expedited = op.expedited;
|
|
operationToAdd.pendingOperation = op;
|
|
add(operationToAdd, op);
|
|
}
|
|
}
|
|
}
|
|
|
|
public boolean add(SyncOperation operation) {
|
|
return add(operation, null /* this is not coming from the database */);
|
|
}
|
|
|
|
/**
|
|
* Adds a SyncOperation to the queue and creates a PendingOperation object to track that sync.
|
|
* If an operation is added that already exists, the existing operation is updated if the newly
|
|
* added operation occurs before (or the interval overlaps).
|
|
*/
|
|
private boolean add(SyncOperation operation,
|
|
SyncStorageEngine.PendingOperation pop) {
|
|
// If an operation with the same key exists and this one should run sooner/overlaps,
|
|
// replace the run interval of the existing operation with this new one.
|
|
// Complications: what if the existing operation is expedited but the new operation has an
|
|
// earlier run time? Will not be a problem for periodic syncs (no expedited flag), and for
|
|
// one-off syncs we only change it if the new sync is sooner.
|
|
final String operationKey = operation.key;
|
|
final SyncOperation existingOperation = mOperationsMap.get(operationKey);
|
|
|
|
if (existingOperation != null) {
|
|
boolean changed = false;
|
|
if (operation.compareTo(existingOperation) <= 0 ) {
|
|
existingOperation.expedited = operation.expedited;
|
|
long newRunTime =
|
|
Math.min(existingOperation.latestRunTime, operation.latestRunTime);
|
|
// Take smaller runtime.
|
|
existingOperation.latestRunTime = newRunTime;
|
|
// Take newer flextime.
|
|
existingOperation.flexTime = operation.flexTime;
|
|
changed = true;
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
operation.pendingOperation = pop;
|
|
// Don't update the PendingOp if one already exists. This really is just a placeholder,
|
|
// no actual scheduling info is placed here.
|
|
if (operation.pendingOperation == null) {
|
|
pop = mSyncStorageEngine.insertIntoPending(operation);
|
|
if (pop == null) {
|
|
throw new IllegalStateException("error adding pending sync operation "
|
|
+ operation);
|
|
}
|
|
operation.pendingOperation = pop;
|
|
}
|
|
|
|
mOperationsMap.put(operationKey, operation);
|
|
return true;
|
|
}
|
|
|
|
public void removeUserLocked(int userId) {
|
|
ArrayList<SyncOperation> opsToRemove = new ArrayList<SyncOperation>();
|
|
for (SyncOperation op : mOperationsMap.values()) {
|
|
if (op.target.userId == userId) {
|
|
opsToRemove.add(op);
|
|
}
|
|
}
|
|
for (SyncOperation op : opsToRemove) {
|
|
remove(op);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove the specified operation if it is in the queue.
|
|
* @param operation the operation to remove
|
|
*/
|
|
public void remove(SyncOperation operation) {
|
|
boolean isLoggable = Log.isLoggable(TAG, Log.VERBOSE);
|
|
SyncOperation operationToRemove = mOperationsMap.remove(operation.key);
|
|
if (isLoggable) {
|
|
Log.v(TAG, "Attempting to remove: " + operation.key);
|
|
}
|
|
if (operationToRemove == null) {
|
|
if (isLoggable) {
|
|
Log.v(TAG, "Could not find: " + operation.key);
|
|
}
|
|
return;
|
|
}
|
|
if (!mSyncStorageEngine.deleteFromPending(operationToRemove.pendingOperation)) {
|
|
final String errorMessage = "unable to find pending row for " + operationToRemove;
|
|
Log.e(TAG, errorMessage, new IllegalStateException(errorMessage));
|
|
}
|
|
}
|
|
|
|
/** Reset backoffs for all operations in the queue. */
|
|
public void clearBackoffs() {
|
|
for (SyncOperation op : mOperationsMap.values()) {
|
|
op.backoff = 0L;
|
|
op.updateEffectiveRunTime();
|
|
}
|
|
}
|
|
|
|
public void onBackoffChanged(SyncStorageEngine.EndPoint target, long backoff) {
|
|
// For each op that matches the target of the changed op, update its
|
|
// backoff and effectiveStartTime
|
|
for (SyncOperation op : mOperationsMap.values()) {
|
|
if (op.target.matchesSpec(target)) {
|
|
op.backoff = backoff;
|
|
op.updateEffectiveRunTime();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void onDelayUntilTimeChanged(SyncStorageEngine.EndPoint target, long delayUntil) {
|
|
// for each op that matches the target info of the provided op, change the delay time.
|
|
for (SyncOperation op : mOperationsMap.values()) {
|
|
if (op.target.matchesSpec(target)) {
|
|
op.delayUntil = delayUntil;
|
|
op.updateEffectiveRunTime();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove all of the SyncOperations associated with a given target.
|
|
*
|
|
* @param info target object provided here can have null Account/provider. This is the case
|
|
* where you want to remove all ops associated with a provider (null Account) or all ops
|
|
* associated with an account (null provider).
|
|
* @param extras option bundle to include to further specify which operation to remove. If this
|
|
* bundle contains sync settings flags, they are ignored.
|
|
*/
|
|
public void remove(final SyncStorageEngine.EndPoint info, Bundle extras) {
|
|
Iterator<Map.Entry<String, SyncOperation>> entries = mOperationsMap.entrySet().iterator();
|
|
while (entries.hasNext()) {
|
|
Map.Entry<String, SyncOperation> entry = entries.next();
|
|
SyncOperation syncOperation = entry.getValue();
|
|
final SyncStorageEngine.EndPoint opInfo = syncOperation.target;
|
|
if (!opInfo.matchesSpec(info)) {
|
|
continue;
|
|
}
|
|
if (extras != null
|
|
&& !SyncManager.syncExtrasEquals(
|
|
syncOperation.extras,
|
|
extras,
|
|
false /* no config flags*/)) {
|
|
continue;
|
|
}
|
|
entries.remove();
|
|
if (!mSyncStorageEngine.deleteFromPending(syncOperation.pendingOperation)) {
|
|
final String errorMessage = "unable to find pending row for " + syncOperation;
|
|
Log.e(TAG, errorMessage, new IllegalStateException(errorMessage));
|
|
}
|
|
}
|
|
}
|
|
|
|
public Collection<SyncOperation> getOperations() {
|
|
return mOperationsMap.values();
|
|
}
|
|
|
|
public void dump(StringBuilder sb) {
|
|
final long now = SystemClock.elapsedRealtime();
|
|
sb.append("SyncQueue: ").append(mOperationsMap.size()).append(" operation(s)\n");
|
|
for (SyncOperation operation : mOperationsMap.values()) {
|
|
sb.append(" ");
|
|
if (operation.effectiveRunTime <= now) {
|
|
sb.append("READY");
|
|
} else {
|
|
sb.append(DateUtils.formatElapsedTime((operation.effectiveRunTime - now) / 1000));
|
|
}
|
|
sb.append(" - ");
|
|
sb.append(operation.dump(mPackageManager, false)).append("\n");
|
|
}
|
|
}
|
|
}
|