/* * 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 android.content; import android.os.SystemClock; import com.google.android.collect.Maps; import android.util.Pair; import android.util.Log; import android.accounts.Account; import java.util.HashMap; import java.util.ArrayList; import java.util.Map; import java.util.Iterator; /** * * @hide */ public class SyncQueue { private static final String TAG = "SyncManager"; private SyncStorageEngine mSyncStorageEngine; // A Map of SyncOperations operationKey -> SyncOperation that is designed for // quick lookup of an enqueued SyncOperation. private final HashMap mOperationsMap = Maps.newHashMap(); public SyncQueue(SyncStorageEngine syncStorageEngine) { mSyncStorageEngine = syncStorageEngine; ArrayList ops = mSyncStorageEngine.getPendingOperations(); final int N = ops.size(); for (int i=0; i nextOperation() { SyncOperation best = null; long bestRunTime = 0; boolean bestIsInitial = false; for (SyncOperation op : mOperationsMap.values()) { final long opRunTime = getOpTime(op); final boolean opIsInitial = getIsInitial(op); if (isOpBetter(best, bestRunTime, bestIsInitial, op, opRunTime, opIsInitial)) { best = op; bestIsInitial = opIsInitial; bestRunTime = opRunTime; } } if (best == null) { return null; } return Pair.create(best, bestRunTime); } // VisibleForTesting long getOpTime(SyncOperation op) { long opRunTime = op.earliestRunTime; if (!op.extras.getBoolean(ContentResolver.SYNC_EXTRAS_IGNORE_BACKOFF, false)) { Pair backoff = mSyncStorageEngine.getBackoff(op.account, op.authority); long delayUntil = mSyncStorageEngine.getDelayUntilTime(op.account, op.authority); opRunTime = Math.max( Math.max(opRunTime, delayUntil), backoff != null ? backoff.first : 0); } return opRunTime; } // VisibleForTesting boolean getIsInitial(SyncOperation op) { // Initial op is defined as an op with an unknown syncable that is not a retry. // We know a sync is a retry if the intialization flag is set, since that will only // be set by the sync dispatching code, thus if it is set it must have already been // dispatched return !op.extras.getBoolean(ContentResolver.SYNC_EXTRAS_INITIALIZE, false) && mSyncStorageEngine.getIsSyncable(op.account, op.authority) < 0; } // return true if op is a better candidate than best. Rules: // if the "Initial" state differs, make the current the best if it is "Initial". // else, if the expedited state differs, pick the expedited unless it is backed off and the // non-expedited isn't // VisibleForTesting boolean isOpBetter( SyncOperation best, long bestRunTime, boolean bestIsInitial, SyncOperation op, long opRunTime, boolean opIsInitial) { boolean setBest = false; if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, "nextOperation: Processing op: " + op); } if (best == null) { if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " First op selected"); } setBest = true; } else if (bestIsInitial == opIsInitial) { if (best.expedited == op.expedited) { if (opRunTime < bestRunTime) { // if both have same level, earlier time wins if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " Same expedite level - new op selected"); } setBest = true; } } else { final long now = SystemClock.elapsedRealtime(); if (op.expedited) { if (opRunTime <= now || bestRunTime > now) { // if op is expedited, it wins unless op can't run yet and best can if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " New op is expedited and can run - new op selected"); } setBest = true; } else { if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " New op is expedited but can't run and best can"); } } } else { if (bestRunTime > now && opRunTime <= now) { // if best is expedited but can't run yet and op can run, op wins if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " New op is not expedited but can run - new op selected"); } setBest = true; } } } } else { if (opIsInitial) { if (Log.isLoggable(TAG, Log.VERBOSE)) { Log.v(TAG, " New op is init - new op selected"); } setBest = true; } } return setBest; } /** * Find and return the SyncOperation that should be run next and is ready to run. * @param now the current {@link android.os.SystemClock#elapsedRealtime()}, used to * decide if the sync operation is ready to run * @return the SyncOperation that should be run next and is ready to run. */ public Pair nextReadyToRun(long now) { Pair nextOpAndRunTime = nextOperation(); if (nextOpAndRunTime == null || nextOpAndRunTime.second > now) { return null; } return nextOpAndRunTime; } public void remove(Account account, String authority) { Iterator> entries = mOperationsMap.entrySet().iterator(); while (entries.hasNext()) { Map.Entry entry = entries.next(); SyncOperation syncOperation = entry.getValue(); if (account != null && !syncOperation.account.equals(account)) { continue; } if (authority != null && !syncOperation.authority.equals(authority)) { 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 void dump(StringBuilder sb) { sb.append("SyncQueue: ").append(mOperationsMap.size()).append(" operation(s)\n"); for (SyncOperation operation : mOperationsMap.values()) { sb.append(operation).append("\n"); } } }