Home | History | Annotate | Download | only in exchange
      1 /*
      2  * Copyright (C) 2008-2009 Marc Blank
      3  * Licensed to The Android Open Source Project.
      4  *
      5  * Licensed under the Apache License, Version 2.0 (the "License");
      6  * you may not use this file except in compliance with the License.
      7  * You may obtain a copy of the License at
      8  *
      9  *      http://www.apache.org/licenses/LICENSE-2.0
     10  *
     11  * Unless required by applicable law or agreed to in writing, software
     12  * distributed under the License is distributed on an "AS IS" BASIS,
     13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14  * See the License for the specific language governing permissions and
     15  * limitations under the License.
     16  */
     17 
     18 package com.android.exchange;
     19 
     20 import android.app.AlarmManager;
     21 import android.app.PendingIntent;
     22 import android.app.Service;
     23 import android.content.BroadcastReceiver;
     24 import android.content.ContentResolver;
     25 import android.content.ContentUris;
     26 import android.content.ContentValues;
     27 import android.content.Context;
     28 import android.content.Intent;
     29 import android.content.IntentFilter;
     30 import android.database.ContentObserver;
     31 import android.database.Cursor;
     32 import android.net.ConnectivityManager;
     33 import android.net.NetworkInfo;
     34 import android.net.NetworkInfo.State;
     35 import android.net.Uri;
     36 import android.os.Bundle;
     37 import android.os.Debug;
     38 import android.os.Handler;
     39 import android.os.IBinder;
     40 import android.os.PowerManager;
     41 import android.os.PowerManager.WakeLock;
     42 import android.os.Process;
     43 import android.os.RemoteCallbackList;
     44 import android.os.RemoteException;
     45 import android.provider.CalendarContract;
     46 import android.provider.CalendarContract.Calendars;
     47 import android.provider.CalendarContract.Events;
     48 import android.provider.ContactsContract;
     49 import android.util.Log;
     50 
     51 import com.android.emailcommon.Api;
     52 import com.android.emailcommon.TempDirectory;
     53 import com.android.emailcommon.provider.Account;
     54 import com.android.emailcommon.provider.EmailContent;
     55 import com.android.emailcommon.provider.EmailContent.Attachment;
     56 import com.android.emailcommon.provider.EmailContent.HostAuthColumns;
     57 import com.android.emailcommon.provider.EmailContent.MailboxColumns;
     58 import com.android.emailcommon.provider.EmailContent.Message;
     59 import com.android.emailcommon.provider.EmailContent.SyncColumns;
     60 import com.android.emailcommon.provider.HostAuth;
     61 import com.android.emailcommon.provider.Mailbox;
     62 import com.android.emailcommon.provider.Policy;
     63 import com.android.emailcommon.provider.ProviderUnavailableException;
     64 import com.android.emailcommon.service.AccountServiceProxy;
     65 import com.android.emailcommon.service.EmailServiceProxy;
     66 import com.android.emailcommon.service.EmailServiceStatus;
     67 import com.android.emailcommon.service.IEmailService;
     68 import com.android.emailcommon.service.IEmailServiceCallback;
     69 import com.android.emailcommon.service.PolicyServiceProxy;
     70 import com.android.emailcommon.service.SearchParams;
     71 import com.android.emailcommon.utility.EmailAsyncTask;
     72 import com.android.emailcommon.utility.EmailClientConnectionManager;
     73 import com.android.emailcommon.utility.Utility;
     74 import com.android.exchange.adapter.CalendarSyncAdapter;
     75 import com.android.exchange.adapter.ContactsSyncAdapter;
     76 import com.android.exchange.adapter.Search;
     77 import com.android.exchange.provider.MailboxUtilities;
     78 import com.android.exchange.utility.FileLogger;
     79 
     80 import org.apache.http.conn.params.ConnManagerPNames;
     81 import org.apache.http.conn.params.ConnPerRoute;
     82 import org.apache.http.conn.routing.HttpRoute;
     83 import org.apache.http.params.BasicHttpParams;
     84 import org.apache.http.params.HttpParams;
     85 
     86 import java.io.IOException;
     87 import java.util.ArrayList;
     88 import java.util.HashMap;
     89 import java.util.List;
     90 import java.util.concurrent.ConcurrentHashMap;
     91 
     92 /**
     93  * The ExchangeService handles all aspects of starting, maintaining, and stopping the various sync
     94  * adapters used by Exchange.  However, it is capable of handing any kind of email sync, and it
     95  * would be appropriate to use for IMAP push, when that functionality is added to the Email
     96  * application.
     97  *
     98  * The Email application communicates with EAS sync adapters via ExchangeService's binder interface,
     99  * which exposes UI-related functionality to the application (see the definitions below)
    100  *
    101  * ExchangeService uses ContentObservers to detect changes to accounts, mailboxes, and messages in
    102  * order to maintain proper 2-way syncing of data.  (More documentation to follow)
    103  *
    104  */
    105 public class ExchangeService extends Service implements Runnable {
    106 
    107     private static final String TAG = "ExchangeService";
    108 
    109     // The ExchangeService's mailbox "id"
    110     public static final int EXTRA_MAILBOX_ID = -1;
    111     public static final int EXCHANGE_SERVICE_MAILBOX_ID = 0;
    112 
    113     private static final int SECONDS = 1000;
    114     private static final int MINUTES = 60*SECONDS;
    115     private static final int ONE_DAY_MINUTES = 1440;
    116 
    117     private static final int EXCHANGE_SERVICE_HEARTBEAT_TIME = 15*MINUTES;
    118     private static final int CONNECTIVITY_WAIT_TIME = 10*MINUTES;
    119 
    120     // Sync hold constants for services with transient errors
    121     private static final int HOLD_DELAY_MAXIMUM = 4*MINUTES;
    122 
    123     // Reason codes when ExchangeService.kick is called (mainly for debugging)
    124     // UI has changed data, requiring an upsync of changes
    125     public static final int SYNC_UPSYNC = 0;
    126     // A scheduled sync (when not using push)
    127     public static final int SYNC_SCHEDULED = 1;
    128     // Mailbox was marked push
    129     public static final int SYNC_PUSH = 2;
    130     // A ping (EAS push signal) was received
    131     public static final int SYNC_PING = 3;
    132     // Misc.
    133     public static final int SYNC_KICK = 4;
    134     // A part request (attachment load, for now) was sent to ExchangeService
    135     public static final int SYNC_SERVICE_PART_REQUEST = 5;
    136 
    137     // Requests >= SYNC_CALLBACK_START generate callbacks to the UI
    138     public static final int SYNC_CALLBACK_START = 6;
    139     // startSync was requested of ExchangeService (other than due to user request)
    140     public static final int SYNC_SERVICE_START_SYNC = SYNC_CALLBACK_START + 0;
    141     // startSync was requested of ExchangeService (due to user request)
    142     public static final int SYNC_UI_REQUEST = SYNC_CALLBACK_START + 1;
    143 
    144     private static final String WHERE_PUSH_OR_PING_NOT_ACCOUNT_MAILBOX =
    145         MailboxColumns.ACCOUNT_KEY + "=? and " + MailboxColumns.TYPE + "!=" +
    146         Mailbox.TYPE_EAS_ACCOUNT_MAILBOX + " and " + MailboxColumns.SYNC_INTERVAL +
    147         " IN (" + Mailbox.CHECK_INTERVAL_PING + ',' + Mailbox.CHECK_INTERVAL_PUSH + ')';
    148     protected static final String WHERE_IN_ACCOUNT_AND_PUSHABLE =
    149         MailboxColumns.ACCOUNT_KEY + "=? and type in (" + Mailbox.TYPE_INBOX + ','
    150         + Mailbox.TYPE_EAS_ACCOUNT_MAILBOX + ',' + Mailbox.TYPE_CONTACTS + ','
    151         + Mailbox.TYPE_CALENDAR + ')';
    152     protected static final String WHERE_IN_ACCOUNT_AND_TYPE_INBOX =
    153         MailboxColumns.ACCOUNT_KEY + "=? and type = " + Mailbox.TYPE_INBOX ;
    154     private static final String WHERE_MAILBOX_KEY = Message.MAILBOX_KEY + "=?";
    155     private static final String WHERE_PROTOCOL_EAS = HostAuthColumns.PROTOCOL + "=\"" +
    156         AbstractSyncService.EAS_PROTOCOL + "\"";
    157     private static final String WHERE_NOT_INTERVAL_NEVER_AND_ACCOUNT_KEY_IN =
    158         "(" + MailboxColumns.TYPE + '=' + Mailbox.TYPE_OUTBOX
    159         + " or " + MailboxColumns.SYNC_INTERVAL + "!=" + Mailbox.CHECK_INTERVAL_NEVER + ')'
    160         + " and " + MailboxColumns.ACCOUNT_KEY + " in (";
    161     private static final String ACCOUNT_KEY_IN = MailboxColumns.ACCOUNT_KEY + " in (";
    162     private static final String WHERE_CALENDAR_ID = Events.CALENDAR_ID + "=?";
    163 
    164     // Offsets into the syncStatus data for EAS that indicate type, exit status, and change count
    165     // The format is S<type_char>:<exit_char>:<change_count>
    166     public static final int STATUS_TYPE_CHAR = 1;
    167     public static final int STATUS_EXIT_CHAR = 3;
    168     public static final int STATUS_CHANGE_COUNT_OFFSET = 5;
    169 
    170     // Ready for ping
    171     public static final int PING_STATUS_OK = 0;
    172     // Service already running (can't ping)
    173     public static final int PING_STATUS_RUNNING = 1;
    174     // Service waiting after I/O error (can't ping)
    175     public static final int PING_STATUS_WAITING = 2;
    176     // Service had a fatal error; can't run
    177     public static final int PING_STATUS_UNABLE = 3;
    178 
    179     private static final int MAX_CLIENT_CONNECTION_MANAGER_SHUTDOWNS = 1;
    180 
    181     // We synchronize on this for all actions affecting the service and error maps
    182     private static final Object sSyncLock = new Object();
    183     // All threads can use this lock to wait for connectivity
    184     public static final Object sConnectivityLock = new Object();
    185     public static boolean sConnectivityHold = false;
    186 
    187     // Keeps track of running services (by mailbox id)
    188     private final HashMap<Long, AbstractSyncService> mServiceMap =
    189         new HashMap<Long, AbstractSyncService>();
    190     // Keeps track of services whose last sync ended with an error (by mailbox id)
    191     /*package*/ ConcurrentHashMap<Long, SyncError> mSyncErrorMap =
    192         new ConcurrentHashMap<Long, SyncError>();
    193     // Keeps track of which services require a wake lock (by mailbox id)
    194     private final HashMap<Long, Boolean> mWakeLocks = new HashMap<Long, Boolean>();
    195     // Keeps track of PendingIntents for mailbox alarms (by mailbox id)
    196     private final HashMap<Long, PendingIntent> mPendingIntents = new HashMap<Long, PendingIntent>();
    197     // The actual WakeLock obtained by ExchangeService
    198     private WakeLock mWakeLock = null;
    199     // Keep our cached list of active Accounts here
    200     public final AccountList mAccountList = new AccountList();
    201 
    202     // Observers that we use to look for changed mail-related data
    203     private final Handler mHandler = new Handler();
    204     private AccountObserver mAccountObserver;
    205     private MailboxObserver mMailboxObserver;
    206     private SyncedMessageObserver mSyncedMessageObserver;
    207 
    208     // Concurrent because CalendarSyncAdapter can modify the map during a wipe
    209     private final ConcurrentHashMap<Long, CalendarObserver> mCalendarObservers =
    210         new ConcurrentHashMap<Long, CalendarObserver>();
    211 
    212     private ContentResolver mResolver;
    213 
    214     // The singleton ExchangeService object, with its thread and stop flag
    215     protected static ExchangeService INSTANCE;
    216     private static Thread sServiceThread = null;
    217     // Cached unique device id
    218     private static String sDeviceId = null;
    219     // ConnectionManager that all EAS threads can use
    220     private static EmailClientConnectionManager sClientConnectionManager = null;
    221     // Count of ClientConnectionManager shutdowns
    222     private static volatile int sClientConnectionManagerShutdownCount = 0;
    223 
    224     private static volatile boolean sStartingUp = false;
    225     private static volatile boolean sStop = false;
    226 
    227     // The reason for ExchangeService's next wakeup call
    228     private String mNextWaitReason;
    229     // Whether we have an unsatisfied "kick" pending
    230     private boolean mKicked = false;
    231 
    232     // Receiver of connectivity broadcasts
    233     private ConnectivityReceiver mConnectivityReceiver = null;
    234     private ConnectivityReceiver mBackgroundDataSettingReceiver = null;
    235     private volatile boolean mBackgroundData = true;
    236     // The most current NetworkInfo (from ConnectivityManager)
    237     private NetworkInfo mNetworkInfo;
    238 
    239     // Callbacks as set up via setCallback
    240     private final RemoteCallbackList<IEmailServiceCallback> mCallbackList =
    241         new RemoteCallbackList<IEmailServiceCallback>();
    242 
    243     private interface ServiceCallbackWrapper {
    244         public void call(IEmailServiceCallback cb) throws RemoteException;
    245     }
    246 
    247     /**
    248      * Proxy that can be used by various sync adapters to tie into ExchangeService's callback system
    249      * Used this way:  ExchangeService.callback().callbackMethod(args...);
    250      * The proxy wraps checking for existence of a ExchangeService instance
    251      * Failures of these callbacks can be safely ignored.
    252      */
    253     static private final IEmailServiceCallback.Stub sCallbackProxy =
    254         new IEmailServiceCallback.Stub() {
    255 
    256         /**
    257          * Broadcast a callback to the everyone that's registered
    258          *
    259          * @param wrapper the ServiceCallbackWrapper used in the broadcast
    260          */
    261         private synchronized void broadcastCallback(ServiceCallbackWrapper wrapper) {
    262             RemoteCallbackList<IEmailServiceCallback> callbackList =
    263                 (INSTANCE == null) ? null: INSTANCE.mCallbackList;
    264             if (callbackList != null) {
    265                 // Call everyone on our callback list
    266                 int count = callbackList.beginBroadcast();
    267                 try {
    268                     for (int i = 0; i < count; i++) {
    269                         try {
    270                             wrapper.call(callbackList.getBroadcastItem(i));
    271                         } catch (RemoteException e) {
    272                             // Safe to ignore
    273                         } catch (RuntimeException e) {
    274                             // We don't want an exception in one call to prevent other calls, so
    275                             // we'll just log this and continue
    276                             Log.e(TAG, "Caught RuntimeException in broadcast", e);
    277                         }
    278                     }
    279                 } finally {
    280                     // No matter what, we need to finish the broadcast
    281                     callbackList.finishBroadcast();
    282                 }
    283             }
    284         }
    285 
    286         public void loadAttachmentStatus(final long messageId, final long attachmentId,
    287                 final int status, final int progress) {
    288             broadcastCallback(new ServiceCallbackWrapper() {
    289                 @Override
    290                 public void call(IEmailServiceCallback cb) throws RemoteException {
    291                     cb.loadAttachmentStatus(messageId, attachmentId, status, progress);
    292                 }
    293             });
    294         }
    295 
    296         public void sendMessageStatus(final long messageId, final String subject, final int status,
    297                 final int progress) {
    298             broadcastCallback(new ServiceCallbackWrapper() {
    299                 @Override
    300                 public void call(IEmailServiceCallback cb) throws RemoteException {
    301                     cb.sendMessageStatus(messageId, subject, status, progress);
    302                 }
    303             });
    304         }
    305 
    306         public void syncMailboxListStatus(final long accountId, final int status,
    307                 final int progress) {
    308             broadcastCallback(new ServiceCallbackWrapper() {
    309                 @Override
    310                 public void call(IEmailServiceCallback cb) throws RemoteException {
    311                     cb.syncMailboxListStatus(accountId, status, progress);
    312                 }
    313             });
    314         }
    315 
    316         public void syncMailboxStatus(final long mailboxId, final int status,
    317                 final int progress) {
    318             broadcastCallback(new ServiceCallbackWrapper() {
    319                 @Override
    320                 public void call(IEmailServiceCallback cb) throws RemoteException {
    321                     cb.syncMailboxStatus(mailboxId, status, progress);
    322                 }
    323             });
    324         }
    325     };
    326 
    327     /**
    328      * Create our EmailService implementation here.
    329      */
    330     private final IEmailService.Stub mBinder = new IEmailService.Stub() {
    331 
    332         public int getApiLevel() {
    333             return Api.LEVEL;
    334         }
    335 
    336         public Bundle validate(HostAuth hostAuth) throws RemoteException {
    337             return AbstractSyncService.validate(EasSyncService.class,
    338                     hostAuth, ExchangeService.this);
    339         }
    340 
    341         public Bundle autoDiscover(String userName, String password) throws RemoteException {
    342             return new EasSyncService().tryAutodiscover(userName, password);
    343         }
    344 
    345         public void startSync(long mailboxId, boolean userRequest) throws RemoteException {
    346             ExchangeService exchangeService = INSTANCE;
    347             if (exchangeService == null) return;
    348             checkExchangeServiceServiceRunning();
    349             Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
    350             if (m == null) return;
    351             Account acct = Account.restoreAccountWithId(exchangeService, m.mAccountKey);
    352             if (acct == null) return;
    353             // If this is a user request and we're being held, release the hold; this allows us to
    354             // try again (the hold might have been specific to this account and released already)
    355             if (userRequest) {
    356                 if (onSyncDisabledHold(acct)) {
    357                     releaseSyncHolds(exchangeService, AbstractSyncService.EXIT_ACCESS_DENIED, acct);
    358                     log("User requested sync of account in sync disabled hold; releasing");
    359                 } else if (onSecurityHold(acct)) {
    360                     releaseSyncHolds(exchangeService, AbstractSyncService.EXIT_SECURITY_FAILURE,
    361                             acct);
    362                     log("User requested sync of account in security hold; releasing");
    363                 }
    364                 if (sConnectivityHold) {
    365                     try {
    366                         // UI is expecting the callbacks....
    367                         sCallbackProxy.syncMailboxStatus(mailboxId, EmailServiceStatus.IN_PROGRESS,
    368                                 0);
    369                         sCallbackProxy.syncMailboxStatus(mailboxId,
    370                                 EmailServiceStatus.CONNECTION_ERROR, 0);
    371                     } catch (RemoteException ignore) {
    372                     }
    373                     return;
    374                 }
    375             }
    376             if (m.mType == Mailbox.TYPE_OUTBOX) {
    377                 // We're using SERVER_ID to indicate an error condition (it has no other use for
    378                 // sent mail)  Upon request to sync the Outbox, we clear this so that all messages
    379                 // are candidates for sending.
    380                 ContentValues cv = new ContentValues();
    381                 cv.put(SyncColumns.SERVER_ID, 0);
    382                 exchangeService.getContentResolver().update(Message.CONTENT_URI,
    383                     cv, WHERE_MAILBOX_KEY, new String[] {Long.toString(mailboxId)});
    384                 // Clear the error state; the Outbox sync will be started from checkMailboxes
    385                 exchangeService.mSyncErrorMap.remove(mailboxId);
    386                 kick("start outbox");
    387                 // Outbox can't be synced in EAS
    388                 return;
    389             } else if (!isSyncable(m)) {
    390                 try {
    391                     // UI may be expecting the callbacks, so send them
    392                     sCallbackProxy.syncMailboxStatus(mailboxId, EmailServiceStatus.IN_PROGRESS, 0);
    393                     sCallbackProxy.syncMailboxStatus(mailboxId, EmailServiceStatus.SUCCESS, 0);
    394                 } catch (RemoteException ignore) {
    395                     // We tried
    396                 }
    397                 return;
    398             }
    399             startManualSync(mailboxId, userRequest ? ExchangeService.SYNC_UI_REQUEST :
    400                 ExchangeService.SYNC_SERVICE_START_SYNC, null);
    401         }
    402 
    403         public void stopSync(long mailboxId) throws RemoteException {
    404             stopManualSync(mailboxId);
    405         }
    406 
    407         public void loadAttachment(long attachmentId, boolean background) throws RemoteException {
    408             Attachment att = Attachment.restoreAttachmentWithId(ExchangeService.this, attachmentId);
    409             log("loadAttachment " + attachmentId + ": " + att.mFileName);
    410             sendMessageRequest(new PartRequest(att, null, null));
    411         }
    412 
    413         public void updateFolderList(long accountId) throws RemoteException {
    414             reloadFolderList(ExchangeService.this, accountId, false);
    415         }
    416 
    417         public void hostChanged(long accountId) throws RemoteException {
    418             ExchangeService exchangeService = INSTANCE;
    419             if (exchangeService == null) return;
    420             ConcurrentHashMap<Long, SyncError> syncErrorMap = exchangeService.mSyncErrorMap;
    421             // Go through the various error mailboxes
    422             for (long mailboxId: syncErrorMap.keySet()) {
    423                 SyncError error = syncErrorMap.get(mailboxId);
    424                 // If it's a login failure, look a little harder
    425                 Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
    426                 // If it's for the account whose host has changed, clear the error
    427                 // If the mailbox is no longer around, remove the entry in the map
    428                 if (m == null) {
    429                     syncErrorMap.remove(mailboxId);
    430                 } else if (error != null && m.mAccountKey == accountId) {
    431                     error.fatal = false;
    432                     error.holdEndTime = 0;
    433                 }
    434             }
    435             // Stop any running syncs
    436             exchangeService.stopAccountSyncs(accountId, true);
    437             // Kick ExchangeService
    438             kick("host changed");
    439         }
    440 
    441         public void setLogging(int flags) throws RemoteException {
    442             Eas.setUserDebug(flags);
    443         }
    444 
    445         public void sendMeetingResponse(long messageId, int response) throws RemoteException {
    446             sendMessageRequest(new MeetingResponseRequest(messageId, response));
    447         }
    448 
    449         public void loadMore(long messageId) throws RemoteException {
    450         }
    451 
    452         // The following three methods are not implemented in this version
    453         public boolean createFolder(long accountId, String name) throws RemoteException {
    454             return false;
    455         }
    456 
    457         public boolean deleteFolder(long accountId, String name) throws RemoteException {
    458             return false;
    459         }
    460 
    461         public boolean renameFolder(long accountId, String oldName, String newName)
    462                 throws RemoteException {
    463             return false;
    464         }
    465 
    466         public void setCallback(IEmailServiceCallback cb) throws RemoteException {
    467             mCallbackList.register(cb);
    468         }
    469 
    470         /**
    471          * Delete PIM (calendar, contacts) data for the specified account
    472          *
    473          * @param accountId the account whose data should be deleted
    474          * @throws RemoteException
    475          */
    476         public void deleteAccountPIMData(long accountId) throws RemoteException {
    477             // Stop any running syncs
    478             ExchangeService.stopAccountSyncs(accountId);
    479             // Delete the data
    480             ExchangeService.deleteAccountPIMData(accountId);
    481         }
    482 
    483         public int searchMessages(long accountId, SearchParams searchParams, long destMailboxId) {
    484             ExchangeService exchangeService = INSTANCE;
    485             if (exchangeService == null) return 0;
    486             return Search.searchMessages(exchangeService, accountId, searchParams,
    487                     destMailboxId);
    488         }
    489     };
    490 
    491     /**
    492      * Return a list of all Accounts in EmailProvider.  Because the result of this call may be used
    493      * in account reconciliation, an exception is thrown if the result cannot be guaranteed accurate
    494      * @param context the caller's context
    495      * @param accounts a list that Accounts will be added into
    496      * @return the list of Accounts
    497      * @throws ProviderUnavailableException if the list of Accounts cannot be guaranteed valid
    498      */
    499     private static AccountList collectEasAccounts(Context context, AccountList accounts) {
    500         ContentResolver resolver = context.getContentResolver();
    501         Cursor c = resolver.query(Account.CONTENT_URI, Account.CONTENT_PROJECTION, null, null,
    502                 null);
    503         // We must throw here; callers might use the information we provide for reconciliation, etc.
    504         if (c == null) throw new ProviderUnavailableException();
    505         try {
    506             ContentValues cv = new ContentValues();
    507             while (c.moveToNext()) {
    508                 long hostAuthId = c.getLong(Account.CONTENT_HOST_AUTH_KEY_RECV_COLUMN);
    509                 if (hostAuthId > 0) {
    510                     HostAuth ha = HostAuth.restoreHostAuthWithId(context, hostAuthId);
    511                     if (ha != null && ha.mProtocol.equals("eas")) {
    512                         Account account = new Account();
    513                         account.restore(c);
    514                         // Cache the HostAuth
    515                         account.mHostAuthRecv = ha;
    516                         accounts.add(account);
    517                         // Fixup flags for inbox (should accept moved mail)
    518                         Mailbox inbox = Mailbox.restoreMailboxOfType(context, account.mId,
    519                                 Mailbox.TYPE_INBOX);
    520                         if (inbox != null &&
    521                                 ((inbox.mFlags & Mailbox.FLAG_ACCEPTS_MOVED_MAIL) == 0)) {
    522                             cv.put(MailboxColumns.FLAGS,
    523                                     inbox.mFlags | Mailbox.FLAG_ACCEPTS_MOVED_MAIL);
    524                             resolver.update(
    525                                     ContentUris.withAppendedId(Mailbox.CONTENT_URI, inbox.mId), cv,
    526                                     null, null);
    527                         }
    528                     }
    529                 }
    530             }
    531         } finally {
    532             c.close();
    533         }
    534         return accounts;
    535     }
    536 
    537     static class AccountList extends ArrayList<Account> {
    538         private static final long serialVersionUID = 1L;
    539 
    540         @Override
    541         public boolean add(Account account) {
    542             // Cache the account manager account
    543             account.mAmAccount = new android.accounts.Account(account.mEmailAddress,
    544                     Eas.EXCHANGE_ACCOUNT_MANAGER_TYPE);
    545             super.add(account);
    546             return true;
    547         }
    548 
    549         public boolean contains(long id) {
    550             for (Account account : this) {
    551                 if (account.mId == id) {
    552                     return true;
    553                 }
    554             }
    555             return false;
    556         }
    557 
    558         public Account getById(long id) {
    559             for (Account account : this) {
    560                 if (account.mId == id) {
    561                     return account;
    562                 }
    563             }
    564             return null;
    565         }
    566 
    567         public Account getByName(String accountName) {
    568             for (Account account : this) {
    569                 if (account.mEmailAddress.equalsIgnoreCase(accountName)) {
    570                     return account;
    571                 }
    572             }
    573             return null;
    574         }
    575     }
    576 
    577     public static void deleteAccountPIMData(long accountId) {
    578         ExchangeService exchangeService = INSTANCE;
    579         if (exchangeService == null) return;
    580         Mailbox mailbox =
    581             Mailbox.restoreMailboxOfType(exchangeService, accountId, Mailbox.TYPE_CONTACTS);
    582         if (mailbox != null) {
    583             EasSyncService service = new EasSyncService(exchangeService, mailbox);
    584             ContactsSyncAdapter adapter = new ContactsSyncAdapter(service);
    585             adapter.wipe();
    586         }
    587         mailbox =
    588             Mailbox.restoreMailboxOfType(exchangeService, accountId, Mailbox.TYPE_CALENDAR);
    589         if (mailbox != null) {
    590             EasSyncService service = new EasSyncService(exchangeService, mailbox);
    591             CalendarSyncAdapter adapter = new CalendarSyncAdapter(service);
    592             adapter.wipe();
    593         }
    594     }
    595 
    596     private boolean onSecurityHold(Account account) {
    597         return (account.mFlags & Account.FLAGS_SECURITY_HOLD) != 0;
    598     }
    599 
    600     private boolean onSyncDisabledHold(Account account) {
    601         return (account.mFlags & Account.FLAGS_SYNC_DISABLED) != 0;
    602     }
    603 
    604     class AccountObserver extends ContentObserver {
    605         String mSyncableEasMailboxSelector = null;
    606         String mEasAccountSelector = null;
    607 
    608         // Runs when ExchangeService first starts
    609         public AccountObserver(Handler handler) {
    610             super(handler);
    611             // At startup, we want to see what EAS accounts exist and cache them
    612             // TODO: Move database work out of UI thread
    613             Context context = getContext();
    614             synchronized (mAccountList) {
    615                 try {
    616                     collectEasAccounts(context, mAccountList);
    617                 } catch (ProviderUnavailableException e) {
    618                     // Just leave if EmailProvider is unavailable
    619                     return;
    620                 }
    621                 // Create an account mailbox for any account without one
    622                 for (Account account : mAccountList) {
    623                     int cnt = Mailbox.count(context, Mailbox.CONTENT_URI, "accountKey="
    624                             + account.mId, null);
    625                     if (cnt == 0) {
    626                         // This case handles a newly created account
    627                         addAccountMailbox(account.mId);
    628                     }
    629                 }
    630             }
    631             // Run through accounts and update account hold information
    632             Utility.runAsync(new Runnable() {
    633                 @Override
    634                 public void run() {
    635                     synchronized (mAccountList) {
    636                         for (Account account : mAccountList) {
    637                             if ((account.mFlags & Account.FLAGS_SECURITY_HOLD) != 0) {
    638                                 // If we're in a security hold, and our policies are active, release
    639                                 // the hold; otherwise, ping PolicyService that this account's
    640                                 // policies are required
    641                                 if (PolicyServiceProxy.isActive(ExchangeService.this, null)) {
    642                                     PolicyServiceProxy.setAccountHoldFlag(ExchangeService.this,
    643                                             account, false);
    644                                     log("isActive true; release hold for " + account.mDisplayName);
    645                                 } else {
    646                                     PolicyServiceProxy.policiesRequired(ExchangeService.this,
    647                                             account.mId);
    648                                 }
    649                             }
    650                         }
    651                     }
    652                 }});
    653         }
    654 
    655         /**
    656          * Returns a String suitable for appending to a where clause that selects for all syncable
    657          * mailboxes in all eas accounts
    658          * @return a complex selection string that is not to be cached
    659          */
    660         public String getSyncableEasMailboxWhere() {
    661             if (mSyncableEasMailboxSelector == null) {
    662                 StringBuilder sb = new StringBuilder(WHERE_NOT_INTERVAL_NEVER_AND_ACCOUNT_KEY_IN);
    663                 boolean first = true;
    664                 synchronized (mAccountList) {
    665                     for (Account account : mAccountList) {
    666                         if (!first) {
    667                             sb.append(',');
    668                         } else {
    669                             first = false;
    670                         }
    671                         sb.append(account.mId);
    672                     }
    673                 }
    674                 sb.append(')');
    675                 mSyncableEasMailboxSelector = sb.toString();
    676             }
    677             return mSyncableEasMailboxSelector;
    678         }
    679 
    680         /**
    681          * Returns a String suitable for appending to a where clause that selects for all eas
    682          * accounts.
    683          * @return a String in the form "accountKey in (a, b, c...)" that is not to be cached
    684          */
    685         public String getAccountKeyWhere() {
    686             if (mEasAccountSelector == null) {
    687                 StringBuilder sb = new StringBuilder(ACCOUNT_KEY_IN);
    688                 boolean first = true;
    689                 synchronized (mAccountList) {
    690                     for (Account account : mAccountList) {
    691                         if (!first) {
    692                             sb.append(',');
    693                         } else {
    694                             first = false;
    695                         }
    696                         sb.append(account.mId);
    697                     }
    698                 }
    699                 sb.append(')');
    700                 mEasAccountSelector = sb.toString();
    701             }
    702             return mEasAccountSelector;
    703         }
    704 
    705         private void onAccountChanged() {
    706             try {
    707                 maybeStartExchangeServiceThread();
    708                 Context context = getContext();
    709 
    710                 // A change to the list requires us to scan for deletions (stop running syncs)
    711                 // At startup, we want to see what accounts exist and cache them
    712                 AccountList currentAccounts = new AccountList();
    713                 try {
    714                     collectEasAccounts(context, currentAccounts);
    715                 } catch (ProviderUnavailableException e) {
    716                     // Just leave if EmailProvider is unavailable
    717                     return;
    718                 }
    719                 synchronized (mAccountList) {
    720                     for (Account account : mAccountList) {
    721                         boolean accountIncomplete =
    722                             (account.mFlags & Account.FLAGS_INCOMPLETE) != 0;
    723                         // If the current list doesn't include this account and the account wasn't
    724                         // incomplete, then this is a deletion
    725                         if (!currentAccounts.contains(account.mId) && !accountIncomplete) {
    726                             // The implication is that the account has been deleted; let's find out
    727                             alwaysLog("Observer found deleted account: " + account.mDisplayName);
    728                             // Run the reconciler (the reconciliation itself runs in the Email app)
    729                             runAccountReconcilerSync(ExchangeService.this);
    730                             // See if the account is still around
    731                             Account deletedAccount =
    732                                 Account.restoreAccountWithId(context, account.mId);
    733                             if (deletedAccount != null) {
    734                                 // It is; add it to our account list
    735                                 alwaysLog("Account still in provider: " + account.mDisplayName);
    736                                 currentAccounts.add(account);
    737                             } else {
    738                                 // It isn't; stop syncs and clear our selectors
    739                                 alwaysLog("Account deletion confirmed: " + account.mDisplayName);
    740                                 stopAccountSyncs(account.mId, true);
    741                                 mSyncableEasMailboxSelector = null;
    742                                 mEasAccountSelector = null;
    743                             }
    744                         } else {
    745                             // Get the newest version of this account
    746                             Account updatedAccount =
    747                                 Account.restoreAccountWithId(context, account.mId);
    748                             if (updatedAccount == null) continue;
    749                             if (account.mSyncInterval != updatedAccount.mSyncInterval
    750                                     || account.mSyncLookback != updatedAccount.mSyncLookback) {
    751                                 // Set the inbox interval to the interval of the Account
    752                                 // This setting should NOT affect other boxes
    753                                 ContentValues cv = new ContentValues();
    754                                 cv.put(MailboxColumns.SYNC_INTERVAL, updatedAccount.mSyncInterval);
    755                                 getContentResolver().update(Mailbox.CONTENT_URI, cv,
    756                                         WHERE_IN_ACCOUNT_AND_TYPE_INBOX, new String[] {
    757                                         Long.toString(account.mId)
    758                                 });
    759                                 // Stop all current syncs; the appropriate ones will restart
    760                                 log("Account " + account.mDisplayName + " changed; stop syncs");
    761                                 stopAccountSyncs(account.mId, true);
    762                             }
    763 
    764                             // See if this account is no longer on security hold
    765                             if (onSecurityHold(account) && !onSecurityHold(updatedAccount)) {
    766                                 releaseSyncHolds(ExchangeService.this,
    767                                         AbstractSyncService.EXIT_SECURITY_FAILURE, account);
    768                             }
    769 
    770                             // Put current values into our cached account
    771                             account.mSyncInterval = updatedAccount.mSyncInterval;
    772                             account.mSyncLookback = updatedAccount.mSyncLookback;
    773                             account.mFlags = updatedAccount.mFlags;
    774                         }
    775                     }
    776                     // Look for new accounts
    777                     for (Account account : currentAccounts) {
    778                         if (!mAccountList.contains(account.mId)) {
    779                             // Don't forget to cache the HostAuth
    780                             HostAuth ha = HostAuth.restoreHostAuthWithId(getContext(),
    781                                     account.mHostAuthKeyRecv);
    782                             if (ha == null) continue;
    783                             account.mHostAuthRecv = ha;
    784                             // This is an addition; create our magic hidden mailbox...
    785                             log("Account observer found new account: " + account.mDisplayName);
    786                             addAccountMailbox(account.mId);
    787                             mAccountList.add(account);
    788                             mSyncableEasMailboxSelector = null;
    789                             mEasAccountSelector = null;
    790                         }
    791                     }
    792                     // Finally, make sure our account list is up to date
    793                     mAccountList.clear();
    794                     mAccountList.addAll(currentAccounts);
    795                 }
    796 
    797                 // See if there's anything to do...
    798                 kick("account changed");
    799             } catch (ProviderUnavailableException e) {
    800                 alwaysLog("Observer failed; provider unavailable");
    801             }
    802         }
    803 
    804         @Override
    805         public void onChange(boolean selfChange) {
    806             new Thread(new Runnable() {
    807                public void run() {
    808                    onAccountChanged();
    809                 }}, "Account Observer").start();
    810         }
    811 
    812         private void addAccountMailbox(long acctId) {
    813             Account acct = Account.restoreAccountWithId(getContext(), acctId);
    814             Mailbox main = new Mailbox();
    815             main.mDisplayName = Eas.ACCOUNT_MAILBOX_PREFIX;
    816             main.mServerId = Eas.ACCOUNT_MAILBOX_PREFIX + System.nanoTime();
    817             main.mAccountKey = acct.mId;
    818             main.mType = Mailbox.TYPE_EAS_ACCOUNT_MAILBOX;
    819             main.mSyncInterval = Mailbox.CHECK_INTERVAL_PUSH;
    820             main.mFlagVisible = false;
    821             main.save(getContext());
    822             log("Initializing account: " + acct.mDisplayName);
    823         }
    824 
    825     }
    826 
    827     /**
    828      * Register a specific Calendar's data observer; we need to recognize when the SYNC_EVENTS
    829      * column has changed (when sync has turned off or on)
    830      * @param account the Account whose Calendar we're observing
    831      */
    832     private void registerCalendarObserver(Account account) {
    833         // Get a new observer
    834         CalendarObserver observer = new CalendarObserver(mHandler, account);
    835         if (observer.mCalendarId != 0) {
    836             // If we find the Calendar (and we'd better) register it and store it in the map
    837             mCalendarObservers.put(account.mId, observer);
    838             mResolver.registerContentObserver(
    839                     ContentUris.withAppendedId(Calendars.CONTENT_URI, observer.mCalendarId), false,
    840                     observer);
    841         }
    842     }
    843 
    844     /**
    845      * Unregister all CalendarObserver's
    846      */
    847     static public void unregisterCalendarObservers() {
    848         ExchangeService exchangeService = INSTANCE;
    849         if (exchangeService == null) return;
    850         ContentResolver resolver = exchangeService.mResolver;
    851         for (CalendarObserver observer: exchangeService.mCalendarObservers.values()) {
    852             resolver.unregisterContentObserver(observer);
    853         }
    854         exchangeService.mCalendarObservers.clear();
    855     }
    856 
    857     /**
    858      * Return the syncable state of an account's calendar, as determined by the sync_events column
    859      * of our Calendar (from CalendarProvider2)
    860      * Note that the current state of sync_events is cached in our CalendarObserver
    861      * @param accountId the id of the account whose calendar we are checking
    862      * @return whether or not syncing of events is enabled
    863      */
    864     private boolean isCalendarEnabled(long accountId) {
    865         CalendarObserver observer = mCalendarObservers.get(accountId);
    866         if (observer != null) {
    867             return (observer.mSyncEvents == 1);
    868         }
    869         // If there's no observer, there's no Calendar in CalendarProvider2, so we return true
    870         // to allow Calendar creation
    871         return true;
    872     }
    873 
    874     private class CalendarObserver extends ContentObserver {
    875         long mAccountId;
    876         long mCalendarId;
    877         long mSyncEvents;
    878         String mAccountName;
    879 
    880         public CalendarObserver(Handler handler, Account account) {
    881             super(handler);
    882             mAccountId = account.mId;
    883             mAccountName = account.mEmailAddress;
    884 
    885             // Find the Calendar for this account
    886             Cursor c = mResolver.query(Calendars.CONTENT_URI,
    887                     new String[] {Calendars._ID, Calendars.SYNC_EVENTS},
    888                     CalendarSyncAdapter.CALENDAR_SELECTION,
    889                     new String[] {account.mEmailAddress, Eas.EXCHANGE_ACCOUNT_MANAGER_TYPE},
    890                     null);
    891             if (c != null) {
    892                 // Save its id and its sync events status
    893                 try {
    894                     if (c.moveToFirst()) {
    895                         mCalendarId = c.getLong(0);
    896                         mSyncEvents = c.getLong(1);
    897                     }
    898                 } finally {
    899                     c.close();
    900                 }
    901             }
    902         }
    903 
    904         @Override
    905         public synchronized void onChange(boolean selfChange) {
    906             // See if the user has changed syncing of our calendar
    907             if (!selfChange) {
    908                 new Thread(new Runnable() {
    909                     public void run() {
    910                         try {
    911                             Cursor c = mResolver.query(Calendars.CONTENT_URI,
    912                                     new String[] {Calendars.SYNC_EVENTS}, Calendars._ID + "=?",
    913                                     new String[] {Long.toString(mCalendarId)}, null);
    914                             if (c == null) return;
    915                             // Get its sync events; if it's changed, we've got work to do
    916                             try {
    917                                 if (c.moveToFirst()) {
    918                                     long newSyncEvents = c.getLong(0);
    919                                     if (newSyncEvents != mSyncEvents) {
    920                                         log("_sync_events changed for calendar in " + mAccountName);
    921                                         Mailbox mailbox = Mailbox.restoreMailboxOfType(INSTANCE,
    922                                                 mAccountId, Mailbox.TYPE_CALENDAR);
    923                                         // Sanity check for mailbox deletion
    924                                         if (mailbox == null) return;
    925                                         ContentValues cv = new ContentValues();
    926                                         if (newSyncEvents == 0) {
    927                                             // When sync is disabled, we're supposed to delete
    928                                             // all events in the calendar
    929                                             log("Deleting events and setting syncKey to 0 for " +
    930                                                     mAccountName);
    931                                             // First, stop any sync that's ongoing
    932                                             stopManualSync(mailbox.mId);
    933                                             // Set the syncKey to 0 (reset)
    934                                             EasSyncService service =
    935                                                 new EasSyncService(INSTANCE, mailbox);
    936                                             CalendarSyncAdapter adapter =
    937                                                 new CalendarSyncAdapter(service);
    938                                             try {
    939                                                 adapter.setSyncKey("0", false);
    940                                             } catch (IOException e) {
    941                                                 // The provider can't be reached; nothing to be done
    942                                             }
    943                                             // Reset the sync key locally and stop syncing
    944                                             cv.put(Mailbox.SYNC_KEY, "0");
    945                                             cv.put(Mailbox.SYNC_INTERVAL,
    946                                                     Mailbox.CHECK_INTERVAL_NEVER);
    947                                             mResolver.update(ContentUris.withAppendedId(
    948                                                     Mailbox.CONTENT_URI, mailbox.mId), cv, null,
    949                                                     null);
    950                                             // Delete all events using the sync adapter
    951                                             // parameter so that the deletion is only local
    952                                             Uri eventsAsSyncAdapter =
    953                                                 CalendarSyncAdapter.asSyncAdapter(
    954                                                     Events.CONTENT_URI,
    955                                                     mAccountName,
    956                                                     Eas.EXCHANGE_ACCOUNT_MANAGER_TYPE);
    957                                             mResolver.delete(eventsAsSyncAdapter, WHERE_CALENDAR_ID,
    958                                                     new String[] {Long.toString(mCalendarId)});
    959                                         } else {
    960                                             // Make this a push mailbox and kick; this will start
    961                                             // a resync of the Calendar; the account mailbox will
    962                                             // ping on this during the next cycle of the ping loop
    963                                             cv.put(Mailbox.SYNC_INTERVAL,
    964                                                     Mailbox.CHECK_INTERVAL_PUSH);
    965                                             mResolver.update(ContentUris.withAppendedId(
    966                                                     Mailbox.CONTENT_URI, mailbox.mId), cv, null,
    967                                                     null);
    968                                             kick("calendar sync changed");
    969                                         }
    970 
    971                                         // Save away the new value
    972                                         mSyncEvents = newSyncEvents;
    973                                     }
    974                                 }
    975                             } finally {
    976                                 c.close();
    977                             }
    978                         } catch (ProviderUnavailableException e) {
    979                             Log.w(TAG, "Observer failed; provider unavailable");
    980                         }
    981                     }}, "Calendar Observer").start();
    982             }
    983         }
    984     }
    985 
    986     private class MailboxObserver extends ContentObserver {
    987         public MailboxObserver(Handler handler) {
    988             super(handler);
    989         }
    990 
    991         @Override
    992         public void onChange(boolean selfChange) {
    993             // See if there's anything to do...
    994             if (!selfChange) {
    995                 kick("mailbox changed");
    996             }
    997         }
    998     }
    999 
   1000     private class SyncedMessageObserver extends ContentObserver {
   1001         Intent syncAlarmIntent = new Intent(INSTANCE, EmailSyncAlarmReceiver.class);
   1002         PendingIntent syncAlarmPendingIntent =
   1003             PendingIntent.getBroadcast(INSTANCE, 0, syncAlarmIntent, 0);
   1004         AlarmManager alarmManager = (AlarmManager)INSTANCE.getSystemService(Context.ALARM_SERVICE);
   1005 
   1006         public SyncedMessageObserver(Handler handler) {
   1007             super(handler);
   1008         }
   1009 
   1010         @Override
   1011         public void onChange(boolean selfChange) {
   1012             alarmManager.set(AlarmManager.RTC_WAKEUP,
   1013                     System.currentTimeMillis() + 10*SECONDS, syncAlarmPendingIntent);
   1014         }
   1015     }
   1016 
   1017     static public IEmailServiceCallback callback() {
   1018         return sCallbackProxy;
   1019     }
   1020 
   1021     static public Account getAccountById(long accountId) {
   1022         ExchangeService exchangeService = INSTANCE;
   1023         if (exchangeService != null) {
   1024             AccountList accountList = exchangeService.mAccountList;
   1025             synchronized (accountList) {
   1026                 return accountList.getById(accountId);
   1027             }
   1028         }
   1029         return null;
   1030     }
   1031 
   1032     static public Account getAccountByName(String accountName) {
   1033         ExchangeService exchangeService = INSTANCE;
   1034         if (exchangeService != null) {
   1035             AccountList accountList = exchangeService.mAccountList;
   1036             synchronized (accountList) {
   1037                 return accountList.getByName(accountName);
   1038             }
   1039         }
   1040         return null;
   1041     }
   1042 
   1043     static public String getEasAccountSelector() {
   1044         ExchangeService exchangeService = INSTANCE;
   1045         if (exchangeService != null && exchangeService.mAccountObserver != null) {
   1046             return exchangeService.mAccountObserver.getAccountKeyWhere();
   1047         }
   1048         return null;
   1049     }
   1050 
   1051     public class SyncStatus {
   1052         static public final int NOT_RUNNING = 0;
   1053         static public final int DIED = 1;
   1054         static public final int SYNC = 2;
   1055         static public final int IDLE = 3;
   1056     }
   1057 
   1058     /*package*/ class SyncError {
   1059         int reason;
   1060         boolean fatal = false;
   1061         long holdDelay = 15*SECONDS;
   1062         long holdEndTime = System.currentTimeMillis() + holdDelay;
   1063 
   1064         SyncError(int _reason, boolean _fatal) {
   1065             reason = _reason;
   1066             fatal = _fatal;
   1067         }
   1068 
   1069         /**
   1070          * We double the holdDelay from 15 seconds through 4 mins
   1071          */
   1072         void escalate() {
   1073             if (holdDelay < HOLD_DELAY_MAXIMUM) {
   1074                 holdDelay *= 2;
   1075             }
   1076             holdEndTime = System.currentTimeMillis() + holdDelay;
   1077         }
   1078     }
   1079 
   1080     private void logSyncHolds() {
   1081         if (Eas.USER_LOG) {
   1082             log("Sync holds:");
   1083             long time = System.currentTimeMillis();
   1084             for (long mailboxId : mSyncErrorMap.keySet()) {
   1085                 Mailbox m = Mailbox.restoreMailboxWithId(this, mailboxId);
   1086                 if (m == null) {
   1087                     log("Mailbox " + mailboxId + " no longer exists");
   1088                 } else {
   1089                     SyncError error = mSyncErrorMap.get(mailboxId);
   1090                     if (error != null) {
   1091                         log("Mailbox " + m.mDisplayName + ", error = " + error.reason
   1092                                 + ", fatal = " + error.fatal);
   1093                         if (error.holdEndTime > 0) {
   1094                             log("Hold ends in " + ((error.holdEndTime - time) / 1000) + "s");
   1095                         }
   1096                     }
   1097                 }
   1098             }
   1099         }
   1100     }
   1101 
   1102     /**
   1103      * Release security holds for the specified account
   1104      * @param account the account whose Mailboxes should be released from security hold
   1105      */
   1106     static public void releaseSecurityHold(Account account) {
   1107         ExchangeService exchangeService = INSTANCE;
   1108         if (exchangeService != null) {
   1109             exchangeService.releaseSyncHolds(INSTANCE, AbstractSyncService.EXIT_SECURITY_FAILURE,
   1110                     account);
   1111         }
   1112     }
   1113 
   1114     /**
   1115      * Release a specific type of hold (the reason) for the specified Account; if the account
   1116      * is null, mailboxes from all accounts with the specified hold will be released
   1117      * @param reason the reason for the SyncError (AbstractSyncService.EXIT_XXX)
   1118      * @param account an Account whose mailboxes should be released (or all if null)
   1119      * @return whether or not any mailboxes were released
   1120      */
   1121     /*package*/ boolean releaseSyncHolds(Context context, int reason, Account account) {
   1122         boolean holdWasReleased = releaseSyncHoldsImpl(context, reason, account);
   1123         kick("security release");
   1124         return holdWasReleased;
   1125     }
   1126 
   1127     private boolean releaseSyncHoldsImpl(Context context, int reason, Account account) {
   1128         boolean holdWasReleased = false;
   1129         for (long mailboxId: mSyncErrorMap.keySet()) {
   1130             if (account != null) {
   1131                 Mailbox m = Mailbox.restoreMailboxWithId(context, mailboxId);
   1132                 if (m == null) {
   1133                     mSyncErrorMap.remove(mailboxId);
   1134                 } else if (m.mAccountKey != account.mId) {
   1135                     continue;
   1136                 }
   1137             }
   1138             SyncError error = mSyncErrorMap.get(mailboxId);
   1139             if (error != null && error.reason == reason) {
   1140                 mSyncErrorMap.remove(mailboxId);
   1141                 holdWasReleased = true;
   1142             }
   1143         }
   1144         return holdWasReleased;
   1145     }
   1146 
   1147     /**
   1148      * Reconcile Exchange accounts with AccountManager (asynchronous)
   1149      * @param context the caller's Context
   1150      */
   1151     public static void reconcileAccounts(final Context context) {
   1152         Utility.runAsync(new Runnable() {
   1153             @Override
   1154             public void run() {
   1155                 ExchangeService exchangeService = INSTANCE;
   1156                 if (exchangeService != null) {
   1157                     exchangeService.runAccountReconcilerSync(context);
   1158                 }
   1159             }});
   1160     }
   1161 
   1162     /**
   1163      * Blocking call to the account reconciler
   1164      */
   1165     public static void runAccountReconcilerSync(Context context) {
   1166         alwaysLog("Reconciling accounts...");
   1167         new AccountServiceProxy(context).reconcileAccounts(
   1168                 HostAuth.SCHEME_EAS, Eas.EXCHANGE_ACCOUNT_MANAGER_TYPE);
   1169     }
   1170 
   1171     public static void log(String str) {
   1172         log(TAG, str);
   1173     }
   1174 
   1175     public static void log(String tag, String str) {
   1176         if (Eas.USER_LOG) {
   1177             Log.d(tag, str);
   1178             if (Eas.FILE_LOG) {
   1179                 FileLogger.log(tag, str);
   1180             }
   1181         }
   1182     }
   1183 
   1184     public static void alwaysLog(String str) {
   1185         if (!Eas.USER_LOG) {
   1186             Log.d(TAG, str);
   1187         } else {
   1188             log(str);
   1189         }
   1190     }
   1191 
   1192     /**
   1193      * EAS requires a unique device id, so that sync is possible from a variety of different
   1194      * devices (e.g. the syncKey is specific to a device)  If we're on an emulator or some other
   1195      * device that doesn't provide one, we can create it as "device".
   1196      * This would work on a real device as well, but it would be better to use the "real" id if
   1197      * it's available
   1198      */
   1199     static public String getDeviceId(Context context) throws IOException {
   1200         if (sDeviceId == null) {
   1201             sDeviceId = new AccountServiceProxy(context).getDeviceId();
   1202             alwaysLog("Received deviceId from Email app: " + sDeviceId);
   1203         }
   1204         return sDeviceId;
   1205     }
   1206 
   1207     @Override
   1208     public IBinder onBind(Intent arg0) {
   1209         return mBinder;
   1210     }
   1211 
   1212     static public ConnPerRoute sConnPerRoute = new ConnPerRoute() {
   1213         public int getMaxForRoute(HttpRoute route) {
   1214             return 8;
   1215         }
   1216     };
   1217 
   1218     static public synchronized EmailClientConnectionManager getClientConnectionManager() {
   1219         if (sClientConnectionManager == null) {
   1220             // After two tries, kill the process.  Most likely, this will happen in the background
   1221             // The service will restart itself after about 5 seconds
   1222             if (sClientConnectionManagerShutdownCount > MAX_CLIENT_CONNECTION_MANAGER_SHUTDOWNS) {
   1223                 alwaysLog("Shutting down process to unblock threads");
   1224                 Process.killProcess(Process.myPid());
   1225             }
   1226             HttpParams params = new BasicHttpParams();
   1227             params.setIntParameter(ConnManagerPNames.MAX_TOTAL_CONNECTIONS, 25);
   1228             params.setParameter(ConnManagerPNames.MAX_CONNECTIONS_PER_ROUTE, sConnPerRoute);
   1229             sClientConnectionManager = EmailClientConnectionManager.newInstance(params);
   1230         }
   1231         // Null is a valid return result if we get an exception
   1232         return sClientConnectionManager;
   1233     }
   1234 
   1235     static private synchronized void shutdownConnectionManager() {
   1236         if (sClientConnectionManager != null) {
   1237             log("Shutting down ClientConnectionManager");
   1238             sClientConnectionManager.shutdown();
   1239             sClientConnectionManagerShutdownCount++;
   1240             sClientConnectionManager = null;
   1241         }
   1242     }
   1243 
   1244     public static void stopAccountSyncs(long acctId) {
   1245         ExchangeService exchangeService = INSTANCE;
   1246         if (exchangeService != null) {
   1247             exchangeService.stopAccountSyncs(acctId, true);
   1248         }
   1249     }
   1250 
   1251     private void stopAccountSyncs(long acctId, boolean includeAccountMailbox) {
   1252         synchronized (sSyncLock) {
   1253             List<Long> deletedBoxes = new ArrayList<Long>();
   1254             for (Long mid : mServiceMap.keySet()) {
   1255                 Mailbox box = Mailbox.restoreMailboxWithId(this, mid);
   1256                 if (box != null) {
   1257                     if (box.mAccountKey == acctId) {
   1258                         if (!includeAccountMailbox &&
   1259                                 box.mType == Mailbox.TYPE_EAS_ACCOUNT_MAILBOX) {
   1260                             AbstractSyncService svc = mServiceMap.get(mid);
   1261                             if (svc != null) {
   1262                                 svc.stop();
   1263                             }
   1264                             continue;
   1265                         }
   1266                         AbstractSyncService svc = mServiceMap.get(mid);
   1267                         if (svc != null) {
   1268                             svc.stop();
   1269                             Thread t = svc.mThread;
   1270                             if (t != null) {
   1271                                 t.interrupt();
   1272                             }
   1273                         }
   1274                         deletedBoxes.add(mid);
   1275                     }
   1276                 }
   1277             }
   1278             for (Long mid : deletedBoxes) {
   1279                 releaseMailbox(mid);
   1280             }
   1281         }
   1282     }
   1283 
   1284     static private void reloadFolderListFailed(long accountId) {
   1285         try {
   1286             callback().syncMailboxListStatus(accountId,
   1287                     EmailServiceStatus.ACCOUNT_UNINITIALIZED, 0);
   1288         } catch (RemoteException e1) {
   1289             // Don't care if this fails
   1290         }
   1291     }
   1292 
   1293     static public void reloadFolderList(Context context, long accountId, boolean force) {
   1294         ExchangeService exchangeService = INSTANCE;
   1295         if (exchangeService == null) return;
   1296         Cursor c = context.getContentResolver().query(Mailbox.CONTENT_URI,
   1297                 Mailbox.CONTENT_PROJECTION, MailboxColumns.ACCOUNT_KEY + "=? AND " +
   1298                 MailboxColumns.TYPE + "=?",
   1299                 new String[] {Long.toString(accountId),
   1300                     Long.toString(Mailbox.TYPE_EAS_ACCOUNT_MAILBOX)}, null);
   1301         try {
   1302             if (c.moveToFirst()) {
   1303                 synchronized(sSyncLock) {
   1304                     Mailbox mailbox = new Mailbox();
   1305                     mailbox.restore(c);
   1306                     Account acct = Account.restoreAccountWithId(context, accountId);
   1307                     if (acct == null) {
   1308                         reloadFolderListFailed(accountId);
   1309                         return;
   1310                     }
   1311                     String syncKey = acct.mSyncKey;
   1312                     // No need to reload the list if we don't have one
   1313                     if (!force && (syncKey == null || syncKey.equals("0"))) {
   1314                         reloadFolderListFailed(accountId);
   1315                         return;
   1316                     }
   1317 
   1318                     // Change all ping/push boxes to push/hold
   1319                     ContentValues cv = new ContentValues();
   1320                     cv.put(Mailbox.SYNC_INTERVAL, Mailbox.CHECK_INTERVAL_PUSH_HOLD);
   1321                     context.getContentResolver().update(Mailbox.CONTENT_URI, cv,
   1322                             WHERE_PUSH_OR_PING_NOT_ACCOUNT_MAILBOX,
   1323                             new String[] {Long.toString(accountId)});
   1324                     log("Set push/ping boxes to push/hold");
   1325 
   1326                     long id = mailbox.mId;
   1327                     AbstractSyncService svc = exchangeService.mServiceMap.get(id);
   1328                     // Tell the service we're done
   1329                     if (svc != null) {
   1330                         synchronized (svc.getSynchronizer()) {
   1331                             svc.stop();
   1332                             // Interrupt the thread so that it can stop
   1333                             Thread thread = svc.mThread;
   1334                             if (thread != null) {
   1335                                 thread.setName(thread.getName() + " (Stopped)");
   1336                                 thread.interrupt();
   1337                             }
   1338                         }
   1339                         // Abandon the service
   1340                         exchangeService.releaseMailbox(id);
   1341                         // And have it start naturally
   1342                         kick("reload folder list");
   1343                     }
   1344                 }
   1345             }
   1346         } finally {
   1347             c.close();
   1348         }
   1349     }
   1350 
   1351     /**
   1352      * Informs ExchangeService that an account has a new folder list; as a result, any existing
   1353      * folder might have become invalid.  Therefore, we act as if the account has been deleted, and
   1354      * then we reinitialize it.
   1355      *
   1356      * @param acctId
   1357      */
   1358     static public void stopNonAccountMailboxSyncsForAccount(long acctId) {
   1359         ExchangeService exchangeService = INSTANCE;
   1360         if (exchangeService != null) {
   1361             exchangeService.stopAccountSyncs(acctId, false);
   1362             kick("reload folder list");
   1363         }
   1364     }
   1365 
   1366     private void acquireWakeLock(long id) {
   1367         synchronized (mWakeLocks) {
   1368             Boolean lock = mWakeLocks.get(id);
   1369             if (lock == null) {
   1370                 if (mWakeLock == null) {
   1371                     PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);
   1372                     mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "MAIL_SERVICE");
   1373                     mWakeLock.acquire();
   1374                     //log("+WAKE LOCK ACQUIRED");
   1375                 }
   1376                 mWakeLocks.put(id, true);
   1377              }
   1378         }
   1379     }
   1380 
   1381     private void releaseWakeLock(long id) {
   1382         synchronized (mWakeLocks) {
   1383             Boolean lock = mWakeLocks.get(id);
   1384             if (lock != null) {
   1385                 mWakeLocks.remove(id);
   1386                 if (mWakeLocks.isEmpty()) {
   1387                     if (mWakeLock != null) {
   1388                         mWakeLock.release();
   1389                     }
   1390                     mWakeLock = null;
   1391                     //log("+WAKE LOCK RELEASED");
   1392                 } else {
   1393                 }
   1394             }
   1395         }
   1396     }
   1397 
   1398     static public String alarmOwner(long id) {
   1399         if (id == EXTRA_MAILBOX_ID) {
   1400             return "ExchangeService";
   1401         } else {
   1402             String name = Long.toString(id);
   1403             if (Eas.USER_LOG && INSTANCE != null) {
   1404                 Mailbox m = Mailbox.restoreMailboxWithId(INSTANCE, id);
   1405                 if (m != null) {
   1406                     name = m.mDisplayName + '(' + m.mAccountKey + ')';
   1407                 }
   1408             }
   1409             return "Mailbox " + name;
   1410         }
   1411     }
   1412 
   1413     private void clearAlarm(long id) {
   1414         synchronized (mPendingIntents) {
   1415             PendingIntent pi = mPendingIntents.get(id);
   1416             if (pi != null) {
   1417                 AlarmManager alarmManager = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
   1418                 alarmManager.cancel(pi);
   1419                 //log("+Alarm cleared for " + alarmOwner(id));
   1420                 mPendingIntents.remove(id);
   1421             }
   1422         }
   1423     }
   1424 
   1425     private void setAlarm(long id, long millis) {
   1426         synchronized (mPendingIntents) {
   1427             PendingIntent pi = mPendingIntents.get(id);
   1428             if (pi == null) {
   1429                 Intent i = new Intent(this, MailboxAlarmReceiver.class);
   1430                 i.putExtra("mailbox", id);
   1431                 i.setData(Uri.parse("Box" + id));
   1432                 pi = PendingIntent.getBroadcast(this, 0, i, 0);
   1433                 mPendingIntents.put(id, pi);
   1434 
   1435                 AlarmManager alarmManager = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
   1436                 alarmManager.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + millis, pi);
   1437                 //log("+Alarm set for " + alarmOwner(id) + ", " + millis/1000 + "s");
   1438             }
   1439         }
   1440     }
   1441 
   1442     private void clearAlarms() {
   1443         AlarmManager alarmManager = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
   1444         synchronized (mPendingIntents) {
   1445             for (PendingIntent pi : mPendingIntents.values()) {
   1446                 alarmManager.cancel(pi);
   1447             }
   1448             mPendingIntents.clear();
   1449         }
   1450     }
   1451 
   1452     static public void runAwake(long id) {
   1453         ExchangeService exchangeService = INSTANCE;
   1454         if (exchangeService != null) {
   1455             exchangeService.acquireWakeLock(id);
   1456             exchangeService.clearAlarm(id);
   1457         }
   1458     }
   1459 
   1460     static public void runAsleep(long id, long millis) {
   1461         ExchangeService exchangeService = INSTANCE;
   1462         if (exchangeService != null) {
   1463             exchangeService.setAlarm(id, millis);
   1464             exchangeService.releaseWakeLock(id);
   1465         }
   1466     }
   1467 
   1468     static public void clearWatchdogAlarm(long id) {
   1469         ExchangeService exchangeService = INSTANCE;
   1470         if (exchangeService != null) {
   1471             exchangeService.clearAlarm(id);
   1472         }
   1473     }
   1474 
   1475     static public void setWatchdogAlarm(long id, long millis) {
   1476         ExchangeService exchangeService = INSTANCE;
   1477         if (exchangeService != null) {
   1478             exchangeService.setAlarm(id, millis);
   1479         }
   1480     }
   1481 
   1482     static public void alert(Context context, final long id) {
   1483         final ExchangeService exchangeService = INSTANCE;
   1484         checkExchangeServiceServiceRunning();
   1485         if (id < 0) {
   1486             log("ExchangeService alert");
   1487             kick("ping ExchangeService");
   1488         } else if (exchangeService == null) {
   1489             context.startService(new Intent(context, ExchangeService.class));
   1490         } else {
   1491             final AbstractSyncService service = exchangeService.mServiceMap.get(id);
   1492             if (service != null) {
   1493                 // Handle alerts in a background thread, as we are typically called from a
   1494                 // broadcast receiver, and are therefore running in the UI thread
   1495                 String threadName = "ExchangeService Alert: ";
   1496                 if (service.mMailbox != null) {
   1497                     threadName += service.mMailbox.mDisplayName;
   1498                 }
   1499                 new Thread(new Runnable() {
   1500                    public void run() {
   1501                        Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, id);
   1502                        if (m != null) {
   1503                            // We ignore drafts completely (doesn't sync).  Changes in Outbox are
   1504                            // handled in the checkMailboxes loop, so we can ignore these pings.
   1505                            if (Eas.USER_LOG) {
   1506                                Log.d(TAG, "Alert for mailbox " + id + " (" + m.mDisplayName + ")");
   1507                            }
   1508                            if (m.mType == Mailbox.TYPE_DRAFTS || m.mType == Mailbox.TYPE_OUTBOX) {
   1509                                String[] args = new String[] {Long.toString(m.mId)};
   1510                                ContentResolver resolver = INSTANCE.mResolver;
   1511                                resolver.delete(Message.DELETED_CONTENT_URI, WHERE_MAILBOX_KEY,
   1512                                        args);
   1513                                resolver.delete(Message.UPDATED_CONTENT_URI, WHERE_MAILBOX_KEY,
   1514                                        args);
   1515                                return;
   1516                            }
   1517                            service.mAccount = Account.restoreAccountWithId(INSTANCE, m.mAccountKey);
   1518                            service.mMailbox = m;
   1519                            // Send the alarm to the sync service
   1520                            if (!service.alarm()) {
   1521                                // A false return means that we were forced to interrupt the thread
   1522                                // In this case, we release the mailbox so that we can start another
   1523                                // thread to do the work
   1524                                log("Alarm failed; releasing mailbox");
   1525                                synchronized(sSyncLock) {
   1526                                    exchangeService.releaseMailbox(id);
   1527                                }
   1528                                // Shutdown the connection manager; this should close all of our
   1529                                // sockets and generate IOExceptions all around.
   1530                                ExchangeService.shutdownConnectionManager();
   1531                            }
   1532                        }
   1533                     }}, threadName).start();
   1534             }
   1535         }
   1536     }
   1537 
   1538     public class ConnectivityReceiver extends BroadcastReceiver {
   1539         @Override
   1540         public void onReceive(Context context, Intent intent) {
   1541             if (intent.getAction().equals(ConnectivityManager.CONNECTIVITY_ACTION)) {
   1542                 Bundle b = intent.getExtras();
   1543                 if (b != null) {
   1544                     NetworkInfo a = (NetworkInfo)b.get(ConnectivityManager.EXTRA_NETWORK_INFO);
   1545                     String info = "Connectivity alert for " + a.getTypeName();
   1546                     State state = a.getState();
   1547                     if (state == State.CONNECTED) {
   1548                         info += " CONNECTED";
   1549                         log(info);
   1550                         synchronized (sConnectivityLock) {
   1551                             sConnectivityLock.notifyAll();
   1552                         }
   1553                         kick("connected");
   1554                     } else if (state == State.DISCONNECTED) {
   1555                         info += " DISCONNECTED";
   1556                         log(info);
   1557                         kick("disconnected");
   1558                     }
   1559                 }
   1560             } else if (intent.getAction().equals(
   1561                     ConnectivityManager.ACTION_BACKGROUND_DATA_SETTING_CHANGED)) {
   1562                 ConnectivityManager cm =
   1563                         (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
   1564                 mBackgroundData = cm.getBackgroundDataSetting();
   1565                 // If background data is now on, we want to kick ExchangeService
   1566                 if (mBackgroundData) {
   1567                     kick("background data on");
   1568                     log("Background data on; restart syncs");
   1569                 // Otherwise, stop all syncs
   1570                 } else {
   1571                     log("Background data off: stop all syncs");
   1572                     EmailAsyncTask.runAsyncParallel(new Runnable() {
   1573                         @Override
   1574                         public void run() {
   1575                             synchronized (mAccountList) {
   1576                                 for (Account account : mAccountList)
   1577                                     ExchangeService.stopAccountSyncs(account.mId);
   1578                             }
   1579                         }});
   1580                 }
   1581             }
   1582         }
   1583     }
   1584 
   1585     /**
   1586      * Starts a service thread and enters it into the service map
   1587      * This is the point of instantiation of all sync threads
   1588      * @param service the service to start
   1589      * @param m the Mailbox on which the service will operate
   1590      */
   1591     private void startServiceThread(AbstractSyncService service, Mailbox m) {
   1592         if (m == null) return;
   1593         synchronized (sSyncLock) {
   1594             String mailboxName = m.mDisplayName;
   1595             String accountName = service.mAccount.mDisplayName;
   1596             Thread thread = new Thread(service, mailboxName + "[" + accountName + "]");
   1597             log("Starting thread for " + mailboxName + " in account " + accountName);
   1598             thread.start();
   1599             mServiceMap.put(m.mId, service);
   1600             runAwake(m.mId);
   1601             if ((m.mServerId != null) && !m.mServerId.startsWith(Eas.ACCOUNT_MAILBOX_PREFIX)) {
   1602                 stopPing(m.mAccountKey);
   1603             }
   1604         }
   1605     }
   1606 
   1607     /**
   1608      * Stop any ping in progress for the given account
   1609      * @param accountId
   1610      */
   1611     private void stopPing(long accountId) {
   1612         // Go through our active mailboxes looking for the right one
   1613         synchronized (sSyncLock) {
   1614             for (long mailboxId: mServiceMap.keySet()) {
   1615                 Mailbox m = Mailbox.restoreMailboxWithId(this, mailboxId);
   1616                 if (m != null) {
   1617                     String serverId = m.mServerId;
   1618                     if (m.mAccountKey == accountId && serverId != null &&
   1619                             serverId.startsWith(Eas.ACCOUNT_MAILBOX_PREFIX)) {
   1620                         // Here's our account mailbox; reset him (stopping pings)
   1621                         AbstractSyncService svc = mServiceMap.get(mailboxId);
   1622                         svc.reset();
   1623                     }
   1624                 }
   1625             }
   1626         }
   1627     }
   1628 
   1629     private void requestSync(Mailbox m, int reason, Request req) {
   1630         // Don't sync if there's no connectivity
   1631         if (sConnectivityHold || (m == null) || sStop) {
   1632             if (reason >= SYNC_CALLBACK_START) {
   1633                 try {
   1634                     sCallbackProxy.syncMailboxStatus(m.mId, EmailServiceStatus.CONNECTION_ERROR, 0);
   1635                 } catch (RemoteException e) {
   1636                     // We tried...
   1637                 }
   1638             }
   1639             return;
   1640         }
   1641         synchronized (sSyncLock) {
   1642             Account acct = Account.restoreAccountWithId(this, m.mAccountKey);
   1643             if (acct != null) {
   1644                 // Always make sure there's not a running instance of this service
   1645                 AbstractSyncService service = mServiceMap.get(m.mId);
   1646                 if (service == null) {
   1647                     service = new EasSyncService(this, m);
   1648                     if (!((EasSyncService)service).mIsValid) return;
   1649                     service.mSyncReason = reason;
   1650                     if (req != null) {
   1651                         service.addRequest(req);
   1652                     }
   1653                     startServiceThread(service, m);
   1654                 }
   1655             }
   1656         }
   1657     }
   1658 
   1659     private void stopServiceThreads() {
   1660         synchronized (sSyncLock) {
   1661             ArrayList<Long> toStop = new ArrayList<Long>();
   1662 
   1663             // Keep track of which services to stop
   1664             for (Long mailboxId : mServiceMap.keySet()) {
   1665                 toStop.add(mailboxId);
   1666             }
   1667 
   1668             // Shut down all of those running services
   1669             for (Long mailboxId : toStop) {
   1670                 AbstractSyncService svc = mServiceMap.get(mailboxId);
   1671                 if (svc != null) {
   1672                     log("Stopping " + svc.mAccount.mDisplayName + '/' + svc.mMailbox.mDisplayName);
   1673                     svc.stop();
   1674                     if (svc.mThread != null) {
   1675                         svc.mThread.interrupt();
   1676                     }
   1677                 }
   1678                 releaseWakeLock(mailboxId);
   1679             }
   1680         }
   1681     }
   1682 
   1683     private void waitForConnectivity() {
   1684         boolean waiting = false;
   1685         ConnectivityManager cm =
   1686             (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
   1687         while (!sStop) {
   1688             NetworkInfo info = cm.getActiveNetworkInfo();
   1689             if (info != null) {
   1690                 mNetworkInfo = info;
   1691                 // We're done if there's an active network
   1692                 if (waiting) {
   1693                     // If we've been waiting, release any I/O error holds
   1694                     releaseSyncHolds(this, AbstractSyncService.EXIT_IO_ERROR, null);
   1695                     // And log what's still being held
   1696                     logSyncHolds();
   1697                 }
   1698                 return;
   1699             } else {
   1700                 // If this is our first time through the loop, shut down running service threads
   1701                 if (!waiting) {
   1702                     waiting = true;
   1703                     stopServiceThreads();
   1704                 }
   1705                 // Wait until a network is connected (or 10 mins), but let the device sleep
   1706                 // We'll set an alarm just in case we don't get notified (bugs happen)
   1707                 synchronized (sConnectivityLock) {
   1708                     runAsleep(EXTRA_MAILBOX_ID, CONNECTIVITY_WAIT_TIME+5*SECONDS);
   1709                     try {
   1710                         log("Connectivity lock...");
   1711                         sConnectivityHold = true;
   1712                         sConnectivityLock.wait(CONNECTIVITY_WAIT_TIME);
   1713                         log("Connectivity lock released...");
   1714                     } catch (InterruptedException e) {
   1715                         // This is fine; we just go around the loop again
   1716                     } finally {
   1717                         sConnectivityHold = false;
   1718                     }
   1719                     runAwake(EXTRA_MAILBOX_ID);
   1720                 }
   1721             }
   1722         }
   1723     }
   1724 
   1725     /**
   1726      * Note that there are two ways the EAS ExchangeService service can be created:
   1727      *
   1728      * 1) as a background service instantiated via startService (which happens on boot, when the
   1729      * first EAS account is created, etc), in which case the service thread is spun up, mailboxes
   1730      * sync, etc. and
   1731      * 2) to execute an RPC call from the UI, in which case the background service will already be
   1732      * running most of the time (unless we're creating a first EAS account)
   1733      *
   1734      * If the running background service detects that there are no EAS accounts (on boot, if none
   1735      * were created, or afterward if the last remaining EAS account is deleted), it will call
   1736      * stopSelf() to terminate operation.
   1737      *
   1738      * The goal is to ensure that the background service is running at all times when there is at
   1739      * least one EAS account in existence
   1740      *
   1741      * Because there are edge cases in which our process can crash (typically, this has been seen
   1742      * in UI crashes, ANR's, etc.), it's possible for the UI to start up again without the
   1743      * background service having been started.  We explicitly try to start the service in Welcome
   1744      * (to handle the case of the app having been reloaded).  We also start the service on any
   1745      * startSync call (if it isn't already running)
   1746      */
   1747     @Override
   1748     public void onCreate() {
   1749         Utility.runAsync(new Runnable() {
   1750             @Override
   1751             public void run() {
   1752                 // Quick checks first, before getting the lock
   1753                 if (sStartingUp) return;
   1754                 synchronized (sSyncLock) {
   1755                     alwaysLog("!!! EAS ExchangeService, onCreate");
   1756                     // Try to start up properly; we might be coming back from a crash that the Email
   1757                     // application isn't aware of.
   1758                     startService(new Intent(EmailServiceProxy.EXCHANGE_INTENT));
   1759                     if (sStop) {
   1760                         return;
   1761                     }
   1762                 }
   1763             }});
   1764     }
   1765 
   1766     @Override
   1767     public int onStartCommand(Intent intent, int flags, int startId) {
   1768         alwaysLog("!!! EAS ExchangeService, onStartCommand, startingUp = " + sStartingUp +
   1769                 ", running = " + (INSTANCE != null));
   1770         if (!sStartingUp && INSTANCE == null) {
   1771             sStartingUp = true;
   1772             Utility.runAsync(new Runnable() {
   1773                 @Override
   1774                 public void run() {
   1775                     try {
   1776                         synchronized (sSyncLock) {
   1777                             // ExchangeService cannot start unless we can connect to AccountService
   1778                             if (!new AccountServiceProxy(ExchangeService.this).test()) {
   1779                                 alwaysLog("!!! Email application not found; stopping self");
   1780                                 stopSelf();
   1781                             }
   1782                             if (sDeviceId == null) {
   1783                                 try {
   1784                                     String deviceId = getDeviceId(ExchangeService.this);
   1785                                     if (deviceId != null) {
   1786                                         sDeviceId = deviceId;
   1787                                     }
   1788                                 } catch (IOException e) {
   1789                                 }
   1790                                 if (sDeviceId == null) {
   1791                                     alwaysLog("!!! deviceId unknown; stopping self and retrying");
   1792                                     stopSelf();
   1793                                     // Try to restart ourselves in a few seconds
   1794                                     Utility.runAsync(new Runnable() {
   1795                                         @Override
   1796                                         public void run() {
   1797                                             try {
   1798                                                 Thread.sleep(5000);
   1799                                             } catch (InterruptedException e) {
   1800                                             }
   1801                                             startService(new Intent(
   1802                                                     EmailServiceProxy.EXCHANGE_INTENT));
   1803                                         }});
   1804                                     return;
   1805                                 }
   1806                             }
   1807                             // Run the reconciler and clean up mismatched accounts - if we weren't
   1808                             // running when accounts were deleted, it won't have been called.
   1809                             runAccountReconcilerSync(ExchangeService.this);
   1810                             // Update other services depending on final account configuration
   1811                             maybeStartExchangeServiceThread();
   1812                             if (sServiceThread == null) {
   1813                                 log("!!! EAS ExchangeService, stopping self");
   1814                                 stopSelf();
   1815                             } else if (sStop) {
   1816                                 // If we were trying to stop, attempt a restart in 5 secs
   1817                                 setAlarm(EXCHANGE_SERVICE_MAILBOX_ID, 5*SECONDS);
   1818                             }
   1819                         }
   1820                     } finally {
   1821                         sStartingUp = false;
   1822                     }
   1823                 }});
   1824         }
   1825         return Service.START_STICKY;
   1826     }
   1827 
   1828     @Override
   1829     public void onDestroy() {
   1830         log("!!! EAS ExchangeService, onDestroy");
   1831         // Handle shutting down off the UI thread
   1832         Utility.runAsync(new Runnable() {
   1833             @Override
   1834             public void run() {
   1835                 // Quick checks first, before getting the lock
   1836                 if (INSTANCE == null || sServiceThread == null) return;
   1837                 synchronized(sSyncLock) {
   1838                     // Stop the sync manager thread and return
   1839                     if (sServiceThread != null) {
   1840                         sStop = true;
   1841                         sServiceThread.interrupt();
   1842                     }
   1843                 }
   1844             }});
   1845     }
   1846 
   1847     void maybeStartExchangeServiceThread() {
   1848         // Start our thread...
   1849         // See if there are any EAS accounts; otherwise, just go away
   1850         if (sServiceThread == null || !sServiceThread.isAlive()) {
   1851             if (EmailContent.count(this, HostAuth.CONTENT_URI, WHERE_PROTOCOL_EAS, null) > 0) {
   1852                 log(sServiceThread == null ? "Starting thread..." : "Restarting thread...");
   1853                 sServiceThread = new Thread(this, "ExchangeService");
   1854                 INSTANCE = this;
   1855                 sServiceThread.start();
   1856             }
   1857         }
   1858     }
   1859 
   1860     /**
   1861      * Start up the ExchangeService service if it's not already running
   1862      * This is a stopgap for cases in which ExchangeService died (due to a crash somewhere in
   1863      * com.android.email) and hasn't been restarted. See the comment for onCreate for details
   1864      */
   1865     static void checkExchangeServiceServiceRunning() {
   1866         ExchangeService exchangeService = INSTANCE;
   1867         if (exchangeService == null) return;
   1868         if (sServiceThread == null) {
   1869             log("!!! checkExchangeServiceServiceRunning; starting service...");
   1870             exchangeService.startService(new Intent(exchangeService, ExchangeService.class));
   1871         }
   1872     }
   1873 
   1874     public void run() {
   1875         sStop = false;
   1876         alwaysLog("ExchangeService thread running");
   1877         // If we're really debugging, turn on all logging
   1878         if (Eas.DEBUG) {
   1879             Eas.USER_LOG = true;
   1880             Eas.PARSER_LOG = true;
   1881             Eas.FILE_LOG = true;
   1882         }
   1883 
   1884         TempDirectory.setTempDirectory(this);
   1885 
   1886         // If we need to wait for the debugger, do so
   1887         if (Eas.WAIT_DEBUG) {
   1888             Debug.waitForDebugger();
   1889         }
   1890 
   1891         // Synchronize here to prevent a shutdown from happening while we initialize our observers
   1892         // and receivers
   1893         synchronized (sSyncLock) {
   1894             if (INSTANCE != null) {
   1895                 mResolver = getContentResolver();
   1896 
   1897                 // Set up our observers; we need them to know when to start/stop various syncs based
   1898                 // on the insert/delete/update of mailboxes and accounts
   1899                 // We also observe synced messages to trigger upsyncs at the appropriate time
   1900                 mAccountObserver = new AccountObserver(mHandler);
   1901                 mResolver.registerContentObserver(Account.NOTIFIER_URI, true, mAccountObserver);
   1902                 mMailboxObserver = new MailboxObserver(mHandler);
   1903                 mResolver.registerContentObserver(Mailbox.CONTENT_URI, false, mMailboxObserver);
   1904                 mSyncedMessageObserver = new SyncedMessageObserver(mHandler);
   1905                 mResolver.registerContentObserver(Message.SYNCED_CONTENT_URI, true,
   1906                         mSyncedMessageObserver);
   1907 
   1908                 // Set up receivers for connectivity and background data setting
   1909                 mConnectivityReceiver = new ConnectivityReceiver();
   1910                 registerReceiver(mConnectivityReceiver, new IntentFilter(
   1911                         ConnectivityManager.CONNECTIVITY_ACTION));
   1912 
   1913                 mBackgroundDataSettingReceiver = new ConnectivityReceiver();
   1914                 registerReceiver(mBackgroundDataSettingReceiver, new IntentFilter(
   1915                         ConnectivityManager.ACTION_BACKGROUND_DATA_SETTING_CHANGED));
   1916                 // Save away the current background data setting; we'll keep track of it with the
   1917                 // receiver we just registered
   1918                 ConnectivityManager cm = (ConnectivityManager)getSystemService(
   1919                         Context.CONNECTIVITY_SERVICE);
   1920                 mBackgroundData = cm.getBackgroundDataSetting();
   1921 
   1922                 // Do any required work to clean up our Mailboxes (this serves to upgrade
   1923                 // mailboxes that existed prior to EmailProvider database version 17)
   1924                 MailboxUtilities.fixupUninitializedParentKeys(this, getEasAccountSelector());
   1925             }
   1926         }
   1927 
   1928         try {
   1929             // Loop indefinitely until we're shut down
   1930             while (!sStop) {
   1931                 runAwake(EXTRA_MAILBOX_ID);
   1932                 waitForConnectivity();
   1933                 mNextWaitReason = null;
   1934                 long nextWait = checkMailboxes();
   1935                 try {
   1936                     synchronized (this) {
   1937                         if (!mKicked) {
   1938                             if (nextWait < 0) {
   1939                                 log("Negative wait? Setting to 1s");
   1940                                 nextWait = 1*SECONDS;
   1941                             }
   1942                             if (nextWait > 10*SECONDS) {
   1943                                 if (mNextWaitReason != null) {
   1944                                     log("Next awake " + nextWait / 1000 + "s: " + mNextWaitReason);
   1945                                 }
   1946                                 runAsleep(EXTRA_MAILBOX_ID, nextWait + (3*SECONDS));
   1947                             }
   1948                             wait(nextWait);
   1949                         }
   1950                     }
   1951                 } catch (InterruptedException e) {
   1952                     // Needs to be caught, but causes no problem
   1953                     log("ExchangeService interrupted");
   1954                 } finally {
   1955                     synchronized (this) {
   1956                         if (mKicked) {
   1957                             //log("Wait deferred due to kick");
   1958                             mKicked = false;
   1959                         }
   1960                     }
   1961                 }
   1962             }
   1963             log("Shutdown requested");
   1964         } catch (ProviderUnavailableException pue) {
   1965             // Shutdown cleanly in this case
   1966             // NOTE: Sync adapters will also crash with this error, but that is already handled
   1967             // in the adapters themselves, i.e. they return cleanly via done().  When the Email
   1968             // process starts running again, the Exchange process will be started again in due
   1969             // course, assuming there is at least one existing EAS account.
   1970             Log.e(TAG, "EmailProvider unavailable; shutting down");
   1971             // Ask for our service to be restarted; this should kick-start the Email process as well
   1972             startService(new Intent(this, ExchangeService.class));
   1973         } catch (RuntimeException e) {
   1974             // Crash; this is a completely unexpected runtime error
   1975             Log.e(TAG, "RuntimeException in ExchangeService", e);
   1976             throw e;
   1977         } finally {
   1978             shutdown();
   1979         }
   1980     }
   1981 
   1982     private void shutdown() {
   1983         synchronized (sSyncLock) {
   1984             // If INSTANCE is null, we've already been shut down
   1985             if (INSTANCE != null) {
   1986                 log("ExchangeService shutting down...");
   1987 
   1988                 // Stop our running syncs
   1989                 stopServiceThreads();
   1990 
   1991                 // Stop receivers
   1992                 if (mConnectivityReceiver != null) {
   1993                     unregisterReceiver(mConnectivityReceiver);
   1994                 }
   1995                 if (mBackgroundDataSettingReceiver != null) {
   1996                     unregisterReceiver(mBackgroundDataSettingReceiver);
   1997                 }
   1998 
   1999                 // Unregister observers
   2000                 ContentResolver resolver = getContentResolver();
   2001                 if (mSyncedMessageObserver != null) {
   2002                     resolver.unregisterContentObserver(mSyncedMessageObserver);
   2003                     mSyncedMessageObserver = null;
   2004                 }
   2005                 if (mAccountObserver != null) {
   2006                     resolver.unregisterContentObserver(mAccountObserver);
   2007                     mAccountObserver = null;
   2008                 }
   2009                 if (mMailboxObserver != null) {
   2010                     resolver.unregisterContentObserver(mMailboxObserver);
   2011                     mMailboxObserver = null;
   2012                 }
   2013                 unregisterCalendarObservers();
   2014 
   2015                 // Clear pending alarms and associated Intents
   2016                 clearAlarms();
   2017 
   2018                 // Release our wake lock, if we have one
   2019                 synchronized (mWakeLocks) {
   2020                     if (mWakeLock != null) {
   2021                         mWakeLock.release();
   2022                         mWakeLock = null;
   2023                     }
   2024                 }
   2025 
   2026                 INSTANCE = null;
   2027                 sServiceThread = null;
   2028                 sStop = false;
   2029                 log("Goodbye");
   2030             }
   2031         }
   2032     }
   2033 
   2034     /**
   2035      * Release a mailbox from the service map and release its wake lock.
   2036      * NOTE: This method MUST be called while holding sSyncLock!
   2037      *
   2038      * @param mailboxId the id of the mailbox to be released
   2039      */
   2040     private void releaseMailbox(long mailboxId) {
   2041         mServiceMap.remove(mailboxId);
   2042         releaseWakeLock(mailboxId);
   2043     }
   2044 
   2045     /**
   2046      * Check whether an Outbox (referenced by a Cursor) has any messages that can be sent
   2047      * @param c the cursor to an Outbox
   2048      * @return true if there is mail to be sent
   2049      */
   2050     private boolean hasSendableMessages(Cursor outboxCursor) {
   2051         Cursor c = mResolver.query(Message.CONTENT_URI, Message.ID_COLUMN_PROJECTION,
   2052                 EasOutboxService.MAILBOX_KEY_AND_NOT_SEND_FAILED,
   2053                 new String[] {Long.toString(outboxCursor.getLong(Mailbox.CONTENT_ID_COLUMN))},
   2054                 null);
   2055         try {
   2056             while (c.moveToNext()) {
   2057                 if (!Utility.hasUnloadedAttachments(this, c.getLong(Message.CONTENT_ID_COLUMN))) {
   2058                     return true;
   2059                 }
   2060             }
   2061         } finally {
   2062             if (c != null) {
   2063                 c.close();
   2064             }
   2065         }
   2066         return false;
   2067     }
   2068 
   2069     /**
   2070      * Determine whether the account is allowed to sync automatically, as opposed to manually, based
   2071      * on whether the "require manual sync when roaming" policy is in force and applicable
   2072      * @param account the account
   2073      * @return whether or not the account can sync automatically
   2074      */
   2075     /*package*/ static boolean canAutoSync(Account account) {
   2076         ExchangeService exchangeService = INSTANCE;
   2077         if (exchangeService == null) {
   2078             return false;
   2079         }
   2080         NetworkInfo networkInfo = exchangeService.mNetworkInfo;
   2081 
   2082         // Enforce manual sync only while roaming here
   2083         long policyKey = account.mPolicyKey;
   2084         // Quick exit from this check
   2085         if ((policyKey != 0) && (networkInfo != null) &&
   2086                 (ConnectivityManager.isNetworkTypeMobile(networkInfo.getType()))) {
   2087             // We'll cache the Policy data here
   2088             Policy policy = account.mPolicy;
   2089             if (policy == null) {
   2090                 policy = Policy.restorePolicyWithId(INSTANCE, policyKey);
   2091                 account.mPolicy = policy;
   2092             }
   2093             if (policy != null && policy.mRequireManualSyncWhenRoaming && networkInfo.isRoaming()) {
   2094                 return false;
   2095             }
   2096         }
   2097         return true;
   2098     }
   2099 
   2100     /**
   2101      * Convenience method to determine whether Email sync is enabled for a given account
   2102      * @param account the Account in question
   2103      * @return whether Email sync is enabled
   2104      */
   2105     private boolean canSyncEmail(android.accounts.Account account) {
   2106         return ContentResolver.getSyncAutomatically(account, EmailContent.AUTHORITY);
   2107     }
   2108 
   2109     /**
   2110      * Determine whether a mailbox of a given type in a given account can be synced automatically
   2111      * by ExchangeService.  This is an increasingly complex determination, taking into account
   2112      * security policies and user settings (both within the Email application and in the Settings
   2113      * application)
   2114      *
   2115      * @param account the Account that the mailbox is in
   2116      * @param type the type of the Mailbox
   2117      * @return whether or not to start a sync
   2118      */
   2119     private boolean isMailboxSyncable(Account account, int type) {
   2120         // This 'if' statement performs checks to see whether or not a mailbox is a
   2121         // candidate for syncing based on policies, user settings, & other restrictions
   2122         if (type == Mailbox.TYPE_CONTACTS || type == Mailbox.TYPE_CALENDAR) {
   2123             // Contacts/Calendar obey this setting from ContentResolver
   2124             if (!ContentResolver.getMasterSyncAutomatically()) {
   2125                 return false;
   2126             }
   2127             // Get the right authority for the mailbox
   2128             String authority;
   2129             if (type == Mailbox.TYPE_CONTACTS) {
   2130                 authority = ContactsContract.AUTHORITY;
   2131             } else {
   2132                 authority = CalendarContract.AUTHORITY;
   2133                 if (!mCalendarObservers.containsKey(account.mId)){
   2134                     // Make sure we have an observer for this Calendar, as
   2135                     // we need to be able to detect sync state changes, sigh
   2136                     registerCalendarObserver(account);
   2137                 }
   2138             }
   2139             // See if "sync automatically" is set; if not, punt
   2140             if (!ContentResolver.getSyncAutomatically(account.mAmAccount, authority)) {
   2141                 return false;
   2142             // See if the calendar is enabled from the Calendar app UI; if not, punt
   2143             } else if ((type == Mailbox.TYPE_CALENDAR) && !isCalendarEnabled(account.mId)) {
   2144                 return false;
   2145             }
   2146         // Never automatically sync trash
   2147         } else if (type == Mailbox.TYPE_TRASH) {
   2148             return false;
   2149         // For non-outbox mail, we do three checks:
   2150         // 1) are we restricted by policy (i.e. manual sync only),
   2151         // 2) has the user checked the "Sync Email" box in Account Settings, and
   2152         // 3) does the user have the master "background data" box checked in Settings
   2153         } else if (type != Mailbox.TYPE_OUTBOX &&
   2154                 (!canAutoSync(account) || !canSyncEmail(account.mAmAccount) ||
   2155                         !mBackgroundData)) {
   2156             return false;
   2157         }
   2158         return true;
   2159     }
   2160 
   2161     private long checkMailboxes () {
   2162         // First, see if any running mailboxes have been deleted
   2163         ArrayList<Long> deletedMailboxes = new ArrayList<Long>();
   2164         synchronized (sSyncLock) {
   2165             for (long mailboxId: mServiceMap.keySet()) {
   2166                 Mailbox m = Mailbox.restoreMailboxWithId(this, mailboxId);
   2167                 if (m == null) {
   2168                     deletedMailboxes.add(mailboxId);
   2169                 }
   2170             }
   2171             // If so, stop them or remove them from the map
   2172             for (Long mailboxId: deletedMailboxes) {
   2173                 AbstractSyncService svc = mServiceMap.get(mailboxId);
   2174                 if (svc == null || svc.mThread == null) {
   2175                     releaseMailbox(mailboxId);
   2176                     continue;
   2177                 } else {
   2178                     boolean alive = svc.mThread.isAlive();
   2179                     log("Deleted mailbox: " + svc.mMailboxName);
   2180                     if (alive) {
   2181                         stopManualSync(mailboxId);
   2182                     } else {
   2183                         log("Removing from serviceMap");
   2184                         releaseMailbox(mailboxId);
   2185                     }
   2186                 }
   2187             }
   2188         }
   2189 
   2190         long nextWait = EXCHANGE_SERVICE_HEARTBEAT_TIME;
   2191         long now = System.currentTimeMillis();
   2192 
   2193         // Start up threads that need it; use a query which finds eas mailboxes where the
   2194         // the sync interval is not "never".  This is the set of mailboxes that we control
   2195         if (mAccountObserver == null) {
   2196             log("mAccountObserver null; service died??");
   2197             return nextWait;
   2198         }
   2199 
   2200         Cursor c = getContentResolver().query(Mailbox.CONTENT_URI, Mailbox.CONTENT_PROJECTION,
   2201                 mAccountObserver.getSyncableEasMailboxWhere(), null, null);
   2202         if (c == null) throw new ProviderUnavailableException();
   2203         try {
   2204             while (c.moveToNext()) {
   2205                 long mailboxId = c.getLong(Mailbox.CONTENT_ID_COLUMN);
   2206                 AbstractSyncService service = null;
   2207                 synchronized (sSyncLock) {
   2208                     service = mServiceMap.get(mailboxId);
   2209                 }
   2210                 if (service == null) {
   2211                     // Get the cached account
   2212                     Account account = getAccountById(c.getInt(Mailbox.CONTENT_ACCOUNT_KEY_COLUMN));
   2213                     if (account == null) continue;
   2214 
   2215                     // We handle a few types of mailboxes specially
   2216                     int mailboxType = c.getInt(Mailbox.CONTENT_TYPE_COLUMN);
   2217                     if (!isMailboxSyncable(account, mailboxType)) {
   2218                         continue;
   2219                     }
   2220 
   2221                     // Check whether we're in a hold (temporary or permanent)
   2222                     SyncError syncError = mSyncErrorMap.get(mailboxId);
   2223                     if (syncError != null) {
   2224                         // Nothing we can do about fatal errors
   2225                         if (syncError.fatal) continue;
   2226                         if (now < syncError.holdEndTime) {
   2227                             // If release time is earlier than next wait time,
   2228                             // move next wait time up to the release time
   2229                             if (syncError.holdEndTime < now + nextWait) {
   2230                                 nextWait = syncError.holdEndTime - now;
   2231                                 mNextWaitReason = "Release hold";
   2232                             }
   2233                             continue;
   2234                         } else {
   2235                             // Keep the error around, but clear the end time
   2236                             syncError.holdEndTime = 0;
   2237                         }
   2238                     }
   2239 
   2240                     // Otherwise, we use the sync interval
   2241                     long syncInterval = c.getInt(Mailbox.CONTENT_SYNC_INTERVAL_COLUMN);
   2242                     if (syncInterval == Mailbox.CHECK_INTERVAL_PUSH) {
   2243                         Mailbox m = EmailContent.getContent(c, Mailbox.class);
   2244                         requestSync(m, SYNC_PUSH, null);
   2245                     } else if (mailboxType == Mailbox.TYPE_OUTBOX) {
   2246                         if (hasSendableMessages(c)) {
   2247                             Mailbox m = EmailContent.getContent(c, Mailbox.class);
   2248                             startServiceThread(new EasOutboxService(this, m), m);
   2249                         }
   2250                     } else if (syncInterval > 0 && syncInterval <= ONE_DAY_MINUTES) {
   2251                         long lastSync = c.getLong(Mailbox.CONTENT_SYNC_TIME_COLUMN);
   2252                         long sinceLastSync = now - lastSync;
   2253                         long toNextSync = syncInterval*MINUTES - sinceLastSync;
   2254                         String name = c.getString(Mailbox.CONTENT_DISPLAY_NAME_COLUMN);
   2255                         if (toNextSync <= 0) {
   2256                             Mailbox m = EmailContent.getContent(c, Mailbox.class);
   2257                             requestSync(m, SYNC_SCHEDULED, null);
   2258                         } else if (toNextSync < nextWait) {
   2259                             nextWait = toNextSync;
   2260                             if (Eas.USER_LOG) {
   2261                                 log("Next sync for " + name + " in " + nextWait/1000 + "s");
   2262                             }
   2263                             mNextWaitReason = "Scheduled sync, " + name;
   2264                         } else if (Eas.USER_LOG) {
   2265                             log("Next sync for " + name + " in " + toNextSync/1000 + "s");
   2266                         }
   2267                     }
   2268                 } else {
   2269                     Thread thread = service.mThread;
   2270                     // Look for threads that have died and remove them from the map
   2271                     if (thread != null && !thread.isAlive()) {
   2272                         if (Eas.USER_LOG) {
   2273                             log("Dead thread, mailbox released: " +
   2274                                     c.getString(Mailbox.CONTENT_DISPLAY_NAME_COLUMN));
   2275                         }
   2276                         releaseMailbox(mailboxId);
   2277                         // Restart this if necessary
   2278                         if (nextWait > 3*SECONDS) {
   2279                             nextWait = 3*SECONDS;
   2280                             mNextWaitReason = "Clean up dead thread(s)";
   2281                         }
   2282                     } else {
   2283                         long requestTime = service.mRequestTime;
   2284                         if (requestTime > 0) {
   2285                             long timeToRequest = requestTime - now;
   2286                             if (timeToRequest <= 0) {
   2287                                 service.mRequestTime = 0;
   2288                                 service.alarm();
   2289                             } else if (requestTime > 0 && timeToRequest < nextWait) {
   2290                                 if (timeToRequest < 11*MINUTES) {
   2291                                     nextWait = timeToRequest < 250 ? 250 : timeToRequest;
   2292                                     mNextWaitReason = "Sync data change";
   2293                                 } else {
   2294                                     log("Illegal timeToRequest: " + timeToRequest);
   2295                                 }
   2296                             }
   2297                         }
   2298                     }
   2299                 }
   2300             }
   2301         } finally {
   2302             c.close();
   2303         }
   2304         return nextWait;
   2305     }
   2306 
   2307     static public void serviceRequest(long mailboxId, int reason) {
   2308         serviceRequest(mailboxId, 5*SECONDS, reason);
   2309     }
   2310 
   2311     /**
   2312      * Return a boolean indicating whether the mailbox can be synced
   2313      * @param m the mailbox
   2314      * @return whether or not the mailbox can be synced
   2315      */
   2316     public static boolean isSyncable(Mailbox m) {
   2317         return m.loadsFromServer(HostAuth.SCHEME_EAS);
   2318     }
   2319 
   2320     static public void serviceRequest(long mailboxId, long ms, int reason) {
   2321         ExchangeService exchangeService = INSTANCE;
   2322         if (exchangeService == null) return;
   2323         Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
   2324         if (m == null || !isSyncable(m)) return;
   2325         try {
   2326             AbstractSyncService service = exchangeService.mServiceMap.get(mailboxId);
   2327             if (service != null) {
   2328                 service.mRequestTime = System.currentTimeMillis() + ms;
   2329                 kick("service request");
   2330             } else {
   2331                 startManualSync(mailboxId, reason, null);
   2332             }
   2333         } catch (Exception e) {
   2334             e.printStackTrace();
   2335         }
   2336     }
   2337 
   2338     static public void serviceRequestImmediate(long mailboxId) {
   2339         ExchangeService exchangeService = INSTANCE;
   2340         if (exchangeService == null) return;
   2341         AbstractSyncService service = exchangeService.mServiceMap.get(mailboxId);
   2342         if (service != null) {
   2343             service.mRequestTime = System.currentTimeMillis();
   2344             Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
   2345             if (m != null) {
   2346                 service.mAccount = Account.restoreAccountWithId(exchangeService, m.mAccountKey);
   2347                 service.mMailbox = m;
   2348                 kick("service request immediate");
   2349             }
   2350         }
   2351     }
   2352 
   2353     static public void sendMessageRequest(Request req) {
   2354         ExchangeService exchangeService = INSTANCE;
   2355         if (exchangeService == null) return;
   2356         Message msg = Message.restoreMessageWithId(exchangeService, req.mMessageId);
   2357         if (msg == null) {
   2358             return;
   2359         }
   2360         long mailboxId = msg.mMailboxKey;
   2361         AbstractSyncService service = exchangeService.mServiceMap.get(mailboxId);
   2362 
   2363         if (service == null) {
   2364             startManualSync(mailboxId, SYNC_SERVICE_PART_REQUEST, req);
   2365             kick("part request");
   2366         } else {
   2367             service.addRequest(req);
   2368         }
   2369     }
   2370 
   2371     /**
   2372      * Determine whether a given Mailbox can be synced, i.e. is not already syncing and is not in
   2373      * an error state
   2374      *
   2375      * @param mailboxId
   2376      * @return whether or not the Mailbox is available for syncing (i.e. is a valid push target)
   2377      */
   2378     static public int pingStatus(long mailboxId) {
   2379         ExchangeService exchangeService = INSTANCE;
   2380         if (exchangeService == null) return PING_STATUS_OK;
   2381         // Already syncing...
   2382         if (exchangeService.mServiceMap.get(mailboxId) != null) {
   2383             return PING_STATUS_RUNNING;
   2384         }
   2385         // No errors or a transient error, don't ping...
   2386         SyncError error = exchangeService.mSyncErrorMap.get(mailboxId);
   2387         if (error != null) {
   2388             if (error.fatal) {
   2389                 return PING_STATUS_UNABLE;
   2390             } else if (error.holdEndTime > 0) {
   2391                 return PING_STATUS_WAITING;
   2392             }
   2393         }
   2394         return PING_STATUS_OK;
   2395     }
   2396 
   2397     static public void startManualSync(long mailboxId, int reason, Request req) {
   2398         ExchangeService exchangeService = INSTANCE;
   2399         if (exchangeService == null) return;
   2400         synchronized (sSyncLock) {
   2401             AbstractSyncService svc = exchangeService.mServiceMap.get(mailboxId);
   2402             if (svc == null) {
   2403                 exchangeService.mSyncErrorMap.remove(mailboxId);
   2404                 Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
   2405                 if (m != null) {
   2406                     log("Starting sync for " + m.mDisplayName);
   2407                     exchangeService.requestSync(m, reason, req);
   2408                 }
   2409             } else {
   2410                 // If this is a ui request, set the sync reason for the service
   2411                 if (reason >= SYNC_CALLBACK_START) {
   2412                     svc.mSyncReason = reason;
   2413                 }
   2414             }
   2415         }
   2416     }
   2417 
   2418     // DO NOT CALL THIS IN A LOOP ON THE SERVICEMAP
   2419     static public void stopManualSync(long mailboxId) {
   2420         ExchangeService exchangeService = INSTANCE;
   2421         if (exchangeService == null) return;
   2422         synchronized (sSyncLock) {
   2423             AbstractSyncService svc = exchangeService.mServiceMap.get(mailboxId);
   2424             if (svc != null) {
   2425                 log("Stopping sync for " + svc.mMailboxName);
   2426                 svc.stop();
   2427                 svc.mThread.interrupt();
   2428                 exchangeService.releaseWakeLock(mailboxId);
   2429             }
   2430         }
   2431     }
   2432 
   2433     /**
   2434      * Wake up ExchangeService to check for mailboxes needing service
   2435      */
   2436     static public void kick(String reason) {
   2437        ExchangeService exchangeService = INSTANCE;
   2438        if (exchangeService != null) {
   2439             synchronized (exchangeService) {
   2440                 //INSTANCE.log("Kick: " + reason);
   2441                 exchangeService.mKicked = true;
   2442                 exchangeService.notify();
   2443             }
   2444         }
   2445         if (sConnectivityLock != null) {
   2446             synchronized (sConnectivityLock) {
   2447                 sConnectivityLock.notify();
   2448             }
   2449         }
   2450     }
   2451 
   2452     static public void accountUpdated(long acctId) {
   2453         ExchangeService exchangeService = INSTANCE;
   2454         if (exchangeService == null) return;
   2455         synchronized (sSyncLock) {
   2456             for (AbstractSyncService svc : exchangeService.mServiceMap.values()) {
   2457                 if (svc.mAccount.mId == acctId) {
   2458                     svc.mAccount = Account.restoreAccountWithId(exchangeService, acctId);
   2459                 }
   2460             }
   2461         }
   2462     }
   2463 
   2464     /**
   2465      * Tell ExchangeService to remove the mailbox from the map of mailboxes with sync errors
   2466      * @param mailboxId the id of the mailbox
   2467      */
   2468     static public void removeFromSyncErrorMap(long mailboxId) {
   2469         ExchangeService exchangeService = INSTANCE;
   2470         if (exchangeService != null) {
   2471             exchangeService.mSyncErrorMap.remove(mailboxId);
   2472         }
   2473     }
   2474 
   2475     private boolean isRunningInServiceThread(long mailboxId) {
   2476         AbstractSyncService syncService = mServiceMap.get(mailboxId);
   2477         Thread thisThread = Thread.currentThread();
   2478         return syncService != null && syncService.mThread != null &&
   2479             thisThread == syncService.mThread;
   2480     }
   2481 
   2482     /**
   2483      * Sent by services indicating that their thread is finished; action depends on the exitStatus
   2484      * of the service.
   2485      *
   2486      * @param svc the service that is finished
   2487      */
   2488     static public void done(AbstractSyncService svc) {
   2489         ExchangeService exchangeService = INSTANCE;
   2490         if (exchangeService == null) return;
   2491         synchronized(sSyncLock) {
   2492             long mailboxId = svc.mMailboxId;
   2493             // If we're no longer the syncing thread for the mailbox, just return
   2494             if (!exchangeService.isRunningInServiceThread(mailboxId)) {
   2495                 return;
   2496             }
   2497             exchangeService.releaseMailbox(mailboxId);
   2498 
   2499             ConcurrentHashMap<Long, SyncError> errorMap = exchangeService.mSyncErrorMap;
   2500             SyncError syncError = errorMap.get(mailboxId);
   2501 
   2502             int exitStatus = svc.mExitStatus;
   2503             Mailbox m = Mailbox.restoreMailboxWithId(exchangeService, mailboxId);
   2504             if (m == null) return;
   2505 
   2506             if (exitStatus != AbstractSyncService.EXIT_LOGIN_FAILURE) {
   2507                 long accountId = m.mAccountKey;
   2508                 Account account = Account.restoreAccountWithId(exchangeService, accountId);
   2509                 if (account == null) return;
   2510                 if (exchangeService.releaseSyncHolds(exchangeService,
   2511                         AbstractSyncService.EXIT_LOGIN_FAILURE, account)) {
   2512                     new AccountServiceProxy(exchangeService).notifyLoginSucceeded(accountId);
   2513                 }
   2514             }
   2515 
   2516             switch (exitStatus) {
   2517                 case AbstractSyncService.EXIT_DONE:
   2518                     if (svc.hasPendingRequests()) {
   2519                         // TODO Handle this case
   2520                     }
   2521                     errorMap.remove(mailboxId);
   2522                     // If we've had a successful sync, clear the shutdown count
   2523                     synchronized (ExchangeService.class) {
   2524                         sClientConnectionManagerShutdownCount = 0;
   2525                     }
   2526                     break;
   2527                 // I/O errors get retried at increasing intervals
   2528                 case AbstractSyncService.EXIT_IO_ERROR:
   2529                     if (syncError != null) {
   2530                         syncError.escalate();
   2531                         log(m.mDisplayName + " held for " + syncError.holdDelay + "ms");
   2532                     } else {
   2533                         errorMap.put(mailboxId, exchangeService.new SyncError(exitStatus, false));
   2534                         log(m.mDisplayName + " added to syncErrorMap, hold for 15s");
   2535                     }
   2536                     break;
   2537                 // These errors are not retried automatically
   2538                 case AbstractSyncService.EXIT_LOGIN_FAILURE:
   2539                     new AccountServiceProxy(exchangeService).notifyLoginFailed(m.mAccountKey);
   2540                     // Fall through
   2541                 case AbstractSyncService.EXIT_SECURITY_FAILURE:
   2542                 case AbstractSyncService.EXIT_ACCESS_DENIED:
   2543                 case AbstractSyncService.EXIT_EXCEPTION:
   2544                     errorMap.put(mailboxId, exchangeService.new SyncError(exitStatus, true));
   2545                     break;
   2546             }
   2547             kick("sync completed");
   2548         }
   2549     }
   2550 
   2551     /**
   2552      * Given the status string from a Mailbox, return the type code for the last sync
   2553      * @param status the syncStatus column of a Mailbox
   2554      * @return
   2555      */
   2556     static public int getStatusType(String status) {
   2557         if (status == null) {
   2558             return -1;
   2559         } else {
   2560             return status.charAt(STATUS_TYPE_CHAR) - '0';
   2561         }
   2562     }
   2563 
   2564     /**
   2565      * Given the status string from a Mailbox, return the change count for the last sync
   2566      * The change count is the number of adds + deletes + changes in the last sync
   2567      * @param status the syncStatus column of a Mailbox
   2568      * @return
   2569      */
   2570     static public int getStatusChangeCount(String status) {
   2571         try {
   2572             String s = status.substring(STATUS_CHANGE_COUNT_OFFSET);
   2573             return Integer.parseInt(s);
   2574         } catch (RuntimeException e) {
   2575             return -1;
   2576         }
   2577     }
   2578 
   2579     static public Context getContext() {
   2580         return INSTANCE;
   2581     }
   2582 }
   2583