/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.exchange; import android.accounts.Account; import android.accounts.OperationCanceledException; import android.app.Service; import android.content.AbstractThreadedSyncAdapter; import android.content.ContentProviderClient; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.SyncResult; import android.database.Cursor; import android.os.Bundle; import android.os.IBinder; import android.provider.CalendarContract.Events; import android.util.Log; import com.android.emailcommon.provider.EmailContent; import com.android.emailcommon.provider.EmailContent.AccountColumns; import com.android.emailcommon.provider.EmailContent.MailboxColumns; import com.android.emailcommon.provider.Mailbox; public class CalendarSyncAdapterService extends Service { private static final String TAG = "EAS CalendarSyncAdapterService"; private static SyncAdapterImpl sSyncAdapter = null; private static final Object sSyncAdapterLock = new Object(); private static final String ACCOUNT_AND_TYPE_CALENDAR = MailboxColumns.ACCOUNT_KEY + "=? AND " + MailboxColumns.TYPE + '=' + Mailbox.TYPE_CALENDAR; private static final String DIRTY_IN_ACCOUNT = Events.DIRTY + "=1 AND " + Events.ACCOUNT_NAME + "=?"; private static final String[] ID_SYNC_KEY_PROJECTION = new String[] {MailboxColumns.ID, MailboxColumns.SYNC_KEY}; private static final int ID_SYNC_KEY_MAILBOX_ID = 0; private static final int ID_SYNC_KEY_SYNC_KEY = 1; public CalendarSyncAdapterService() { super(); } private static class SyncAdapterImpl extends AbstractThreadedSyncAdapter { private Context mContext; public SyncAdapterImpl(Context context) { super(context, true /* autoInitialize */); mContext = context; } @Override public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) { try { CalendarSyncAdapterService.performSync(mContext, account, extras, authority, provider, syncResult); } catch (OperationCanceledException e) { } } } @Override public void onCreate() { super.onCreate(); synchronized (sSyncAdapterLock) { if (sSyncAdapter == null) { sSyncAdapter = new SyncAdapterImpl(getApplicationContext()); } } } @Override public IBinder onBind(Intent intent) { return sSyncAdapter.getSyncAdapterBinder(); } /** * Partial integration with system SyncManager; we tell our EAS ExchangeService to start a * calendar sync when we get the signal from SyncManager. * The missing piece at this point is integration with the push/ping mechanism in EAS; this will * be put in place at a later time. */ private static void performSync(Context context, Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) throws OperationCanceledException { ContentResolver cr = context.getContentResolver(); boolean logging = Eas.USER_LOG; if (extras.getBoolean(ContentResolver.SYNC_EXTRAS_UPLOAD)) { Cursor c = cr.query(Events.CONTENT_URI, new String[] {Events._ID}, DIRTY_IN_ACCOUNT, new String[] {account.name}, null); try { if (!c.moveToFirst()) { if (logging) { Log.d(TAG, "No changes for " + account.name); } return; } } finally { c.close(); } } // Find the (EmailProvider) account associated with this email address Cursor accountCursor = cr.query(com.android.emailcommon.provider.Account.CONTENT_URI, EmailContent.ID_PROJECTION, AccountColumns.EMAIL_ADDRESS + "=?", new String[] {account.name}, null); try { if (accountCursor.moveToFirst()) { long accountId = accountCursor.getLong(0); // Now, find the calendar mailbox associated with the account Cursor mailboxCursor = cr.query(Mailbox.CONTENT_URI, ID_SYNC_KEY_PROJECTION, ACCOUNT_AND_TYPE_CALENDAR, new String[] {Long.toString(accountId)}, null); try { if (mailboxCursor.moveToFirst()) { if (logging) { Log.d(TAG, "Upload sync requested for " + account.name); } String syncKey = mailboxCursor.getString(ID_SYNC_KEY_SYNC_KEY); if ((syncKey == null) || (syncKey.equals("0"))) { if (logging) { Log.d(TAG, "Can't sync; mailbox in initial state"); } return; } // Ask for a sync from our sync manager ExchangeService.serviceRequest(mailboxCursor.getLong( ID_SYNC_KEY_MAILBOX_ID), ExchangeService.SYNC_UPSYNC); } } finally { mailboxCursor.close(); } } } finally { accountCursor.close(); } } }