2010-02-02 19:17:48 +00:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2011-06-27 19:12:41 +00:00
|
|
|
package com.android.email.service;
|
2010-02-02 19:17:48 +00:00
|
|
|
|
2012-08-21 22:21:40 +00:00
|
|
|
import android.accounts.AccountManager;
|
|
|
|
import android.accounts.AccountManagerFuture;
|
|
|
|
import android.accounts.AuthenticatorException;
|
|
|
|
import android.accounts.OperationCanceledException;
|
2012-04-25 17:26:46 +00:00
|
|
|
import android.app.Service;
|
2012-08-23 05:25:42 +00:00
|
|
|
import android.content.ComponentName;
|
|
|
|
import android.content.ContentProviderClient;
|
2012-08-21 22:21:40 +00:00
|
|
|
import android.content.ContentResolver;
|
|
|
|
import android.content.ContentUris;
|
|
|
|
import android.content.ContentValues;
|
2010-02-02 19:17:48 +00:00
|
|
|
import android.content.Context;
|
|
|
|
import android.content.Intent;
|
2012-08-23 05:25:42 +00:00
|
|
|
import android.content.pm.PackageManager;
|
2012-06-28 17:40:46 +00:00
|
|
|
import android.content.res.Resources;
|
|
|
|
import android.content.res.TypedArray;
|
|
|
|
import android.content.res.XmlResourceParser;
|
2012-08-21 22:21:40 +00:00
|
|
|
import android.database.Cursor;
|
|
|
|
import android.net.Uri;
|
2012-08-23 05:25:42 +00:00
|
|
|
import android.os.AsyncTask;
|
2012-04-25 17:26:46 +00:00
|
|
|
import android.os.Bundle;
|
|
|
|
import android.os.IBinder;
|
|
|
|
import android.os.RemoteException;
|
2012-08-21 22:21:40 +00:00
|
|
|
import android.provider.CalendarContract;
|
2012-08-23 05:25:42 +00:00
|
|
|
import android.provider.CalendarContract.Calendars;
|
|
|
|
import android.provider.CalendarContract.SyncState;
|
2012-08-21 22:21:40 +00:00
|
|
|
import android.provider.ContactsContract;
|
2012-08-23 05:25:42 +00:00
|
|
|
import android.provider.SyncStateContract;
|
2012-03-02 18:31:11 +00:00
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
import com.android.email.R;
|
2012-04-25 17:26:46 +00:00
|
|
|
import com.android.emailcommon.Api;
|
2012-06-28 17:40:46 +00:00
|
|
|
import com.android.emailcommon.Logging;
|
|
|
|
import com.android.emailcommon.provider.Account;
|
2012-08-21 22:21:40 +00:00
|
|
|
import com.android.emailcommon.provider.EmailContent;
|
|
|
|
import com.android.emailcommon.provider.EmailContent.AccountColumns;
|
2012-08-23 05:25:42 +00:00
|
|
|
import com.android.emailcommon.provider.HostAuth;
|
2011-06-27 19:12:41 +00:00
|
|
|
import com.android.emailcommon.service.EmailServiceProxy;
|
|
|
|
import com.android.emailcommon.service.IEmailService;
|
|
|
|
import com.android.emailcommon.service.IEmailServiceCallback;
|
2012-04-25 17:26:46 +00:00
|
|
|
import com.android.emailcommon.service.SearchParams;
|
2012-06-28 17:40:46 +00:00
|
|
|
import com.android.emailcommon.service.SyncWindow;
|
2013-05-26 04:32:32 +00:00
|
|
|
import com.android.mail.utils.LogUtils;
|
2012-06-28 17:40:46 +00:00
|
|
|
|
2013-03-05 17:45:10 +00:00
|
|
|
import com.google.common.collect.Lists;
|
|
|
|
import com.google.common.collect.Maps;
|
2012-06-28 17:40:46 +00:00
|
|
|
import org.xmlpull.v1.XmlPullParserException;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
2013-03-05 17:45:10 +00:00
|
|
|
import java.util.Map;
|
2011-06-27 19:12:41 +00:00
|
|
|
|
2010-02-02 19:17:48 +00:00
|
|
|
/**
|
2011-06-27 19:12:41 +00:00
|
|
|
* Utility functions for EmailService support.
|
2010-02-02 19:17:48 +00:00
|
|
|
*/
|
2011-06-27 19:12:41 +00:00
|
|
|
public class EmailServiceUtils {
|
2013-03-05 17:45:10 +00:00
|
|
|
private static final ArrayList<EmailServiceInfo> sServiceList = Lists.newArrayList();
|
|
|
|
private static final Map<String, EmailServiceInfo> sServiceMap = Maps.newHashMap();
|
2012-06-28 17:40:46 +00:00
|
|
|
|
2010-02-02 19:17:48 +00:00
|
|
|
/**
|
2012-06-28 17:40:46 +00:00
|
|
|
* Starts an EmailService by protocol
|
2010-02-02 19:17:48 +00:00
|
|
|
*/
|
2012-06-28 17:40:46 +00:00
|
|
|
public static void startService(Context context, String protocol) {
|
|
|
|
EmailServiceInfo info = getServiceInfo(context, protocol);
|
|
|
|
if (info != null && info.intentAction != null) {
|
|
|
|
context.startService(new Intent(info.intentAction));
|
|
|
|
}
|
2010-02-02 19:17:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-06-28 17:40:46 +00:00
|
|
|
* Starts all remote services
|
2010-02-02 19:17:48 +00:00
|
|
|
*/
|
2012-06-28 17:40:46 +00:00
|
|
|
public static void startRemoteServices(Context context) {
|
|
|
|
for (EmailServiceInfo info: getServiceInfoList(context)) {
|
|
|
|
if (info.intentAction != null) {
|
|
|
|
context.startService(new Intent(info.intentAction));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether or not remote services are present on device
|
|
|
|
*/
|
|
|
|
public static boolean areRemoteServicesInstalled(Context context) {
|
|
|
|
for (EmailServiceInfo info: getServiceInfoList(context)) {
|
|
|
|
if (info.intentAction != null) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Starts all remote services
|
|
|
|
*/
|
|
|
|
public static void setRemoteServicesLogging(Context context, int debugBits) {
|
|
|
|
for (EmailServiceInfo info: getServiceInfoList(context)) {
|
|
|
|
if (info.intentAction != null) {
|
|
|
|
EmailServiceProxy service =
|
|
|
|
EmailServiceUtils.getService(context, null, info.protocol);
|
|
|
|
if (service != null) {
|
|
|
|
try {
|
|
|
|
service.setLogging(debugBits);
|
|
|
|
} catch (RemoteException e) {
|
|
|
|
// Move along, nothing to see
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-13 02:56:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2011-06-27 19:12:41 +00:00
|
|
|
* Determine if the EmailService is available
|
2011-02-13 02:56:09 +00:00
|
|
|
*/
|
2012-06-28 17:40:46 +00:00
|
|
|
public static boolean isServiceAvailable(Context context, String protocol) {
|
|
|
|
EmailServiceInfo info = getServiceInfo(context, protocol);
|
|
|
|
if (info == null) return false;
|
|
|
|
if (info.klass != null) return true;
|
|
|
|
return new EmailServiceProxy(context, info.intentAction, null).test();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For a given account id, return a service proxy if applicable, or null.
|
|
|
|
*
|
|
|
|
* @param accountId the message of interest
|
|
|
|
* @result service proxy, or null if n/a
|
|
|
|
*/
|
|
|
|
public static EmailServiceProxy getServiceForAccount(Context context,
|
|
|
|
IEmailServiceCallback callback, long accountId) {
|
|
|
|
return getService(context, callback, Account.getProtocol(context, accountId));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Holder of service information (currently just name and class/intent); if there is a class
|
|
|
|
* member, this is a (local, i.e. same process) service; otherwise, this is a remote service
|
|
|
|
*/
|
|
|
|
public static class EmailServiceInfo {
|
|
|
|
public String protocol;
|
|
|
|
public String name;
|
|
|
|
public String accountType;
|
|
|
|
Class<? extends Service> klass;
|
|
|
|
String intentAction;
|
|
|
|
public int port;
|
|
|
|
public int portSsl;
|
|
|
|
public boolean defaultSsl;
|
|
|
|
public boolean offerTls;
|
|
|
|
public boolean offerCerts;
|
|
|
|
public boolean usesSmtp;
|
|
|
|
public boolean offerLocalDeletes;
|
|
|
|
public int defaultLocalDeletes;
|
|
|
|
public boolean offerPrefix;
|
|
|
|
public boolean usesAutodiscover;
|
|
|
|
public boolean offerLookback;
|
|
|
|
public int defaultLookback;
|
|
|
|
public boolean syncChanges;
|
|
|
|
public boolean syncContacts;
|
|
|
|
public boolean syncCalendar;
|
|
|
|
public boolean offerAttachmentPreload;
|
|
|
|
public CharSequence[] syncIntervalStrings;
|
|
|
|
public CharSequence[] syncIntervals;
|
|
|
|
public int defaultSyncInterval;
|
2012-08-16 21:11:11 +00:00
|
|
|
public String inferPrefix;
|
2012-08-23 05:25:42 +00:00
|
|
|
public boolean requiresAccountUpdate;
|
2012-08-24 02:40:47 +00:00
|
|
|
public boolean offerLoadMore;
|
2012-09-08 20:08:50 +00:00
|
|
|
public boolean requiresSetup;
|
2012-09-20 20:34:13 +00:00
|
|
|
public boolean hide;
|
2012-06-28 17:40:46 +00:00
|
|
|
|
2012-09-20 20:34:13 +00:00
|
|
|
@Override
|
2012-06-28 17:40:46 +00:00
|
|
|
public String toString() {
|
|
|
|
StringBuilder sb = new StringBuilder("Protocol: ");
|
|
|
|
sb.append(protocol);
|
|
|
|
sb.append(", ");
|
|
|
|
sb.append(klass != null ? "Local" : "Remote");
|
2012-09-08 01:31:04 +00:00
|
|
|
sb.append(" , Account Type: ");
|
|
|
|
sb.append(accountType);
|
2012-06-28 17:40:46 +00:00
|
|
|
return sb.toString();
|
|
|
|
}
|
2010-02-02 23:06:52 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
public static EmailServiceProxy getService(Context context, IEmailServiceCallback callback,
|
|
|
|
String protocol) {
|
2012-08-16 17:13:00 +00:00
|
|
|
EmailServiceInfo info = null;
|
2012-06-28 17:40:46 +00:00
|
|
|
// Handle the degenerate case here (account might have been deleted)
|
2012-08-16 17:13:00 +00:00
|
|
|
if (protocol != null) {
|
|
|
|
info = getServiceInfo(context, protocol);
|
|
|
|
}
|
|
|
|
if (info == null) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Returning NullService for " + protocol);
|
2012-06-28 17:40:46 +00:00
|
|
|
return new EmailServiceProxy(context, NullService.class, null);
|
2012-08-23 05:25:42 +00:00
|
|
|
} else {
|
|
|
|
return getServiceFromInfo(context, callback, info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static EmailServiceProxy getServiceFromInfo(Context context,
|
|
|
|
IEmailServiceCallback callback, EmailServiceInfo info) {
|
|
|
|
if (info.klass != null) {
|
2012-06-28 17:40:46 +00:00
|
|
|
return new EmailServiceProxy(context, info.klass, callback);
|
|
|
|
} else {
|
|
|
|
return new EmailServiceProxy(context, info.intentAction, callback);
|
|
|
|
}
|
2011-06-27 19:12:41 +00:00
|
|
|
}
|
|
|
|
|
2012-09-08 20:08:50 +00:00
|
|
|
public static EmailServiceInfo getServiceInfoForAccount(Context context, long accountId) {
|
|
|
|
String protocol = Account.getProtocol(context, accountId);
|
|
|
|
return getServiceInfo(context, protocol);
|
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
public static EmailServiceInfo getServiceInfo(Context context, String protocol) {
|
|
|
|
if (sServiceList.isEmpty()) {
|
|
|
|
findServices(context);
|
|
|
|
}
|
2013-03-05 17:45:10 +00:00
|
|
|
return sServiceMap.get(protocol);
|
2011-06-27 19:12:41 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
public static List<EmailServiceInfo> getServiceInfoList(Context context) {
|
2012-07-20 17:51:33 +00:00
|
|
|
synchronized(sServiceList) {
|
|
|
|
if (sServiceList.isEmpty()) {
|
|
|
|
findServices(context);
|
|
|
|
}
|
|
|
|
return sServiceList;
|
2012-06-28 17:40:46 +00:00
|
|
|
}
|
2010-02-24 19:51:59 +00:00
|
|
|
}
|
|
|
|
|
2012-08-21 22:21:40 +00:00
|
|
|
private static void finishAccountManagerBlocker(AccountManagerFuture<?> future) {
|
|
|
|
try {
|
|
|
|
// Note: All of the potential errors are simply logged
|
|
|
|
// here, as there is nothing to actually do about them.
|
|
|
|
future.getResult();
|
|
|
|
} catch (OperationCanceledException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, e.toString());
|
2012-08-21 22:21:40 +00:00
|
|
|
} catch (AuthenticatorException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, e.toString());
|
2012-08-21 22:21:40 +00:00
|
|
|
} catch (IOException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, e.toString());
|
2012-08-21 22:21:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
private static class UpdateAccountManagerTask extends AsyncTask<Void, Void, Void> {
|
|
|
|
private final Context mContext;
|
|
|
|
private final android.accounts.Account mAccount;
|
|
|
|
private final EmailServiceInfo mOldInfo;
|
|
|
|
private final EmailServiceInfo mNewInfo;
|
|
|
|
|
|
|
|
public UpdateAccountManagerTask(Context context, android.accounts.Account amAccount,
|
|
|
|
EmailServiceInfo oldInfo, EmailServiceInfo newInfo) {
|
|
|
|
super();
|
|
|
|
mContext = context;
|
|
|
|
mAccount = amAccount;
|
|
|
|
mOldInfo = oldInfo;
|
|
|
|
mNewInfo = newInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... params) {
|
|
|
|
updateAccountManagerType(mContext, mAccount, mOldInfo, mNewInfo);
|
|
|
|
return null;
|
|
|
|
}
|
2012-08-21 22:21:40 +00:00
|
|
|
}
|
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
private static class DisableComponentsTask extends AsyncTask<Void, Void, Void> {
|
|
|
|
private final Context mContext;
|
|
|
|
|
|
|
|
public DisableComponentsTask(Context context) {
|
|
|
|
super();
|
|
|
|
mContext = context;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... params) {
|
|
|
|
disableComponent(mContext, LegacyEmailAuthenticatorService.class);
|
|
|
|
disableComponent(mContext, LegacyEasAuthenticatorService.class);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void updateAccountManagerType(Context context,
|
|
|
|
android.accounts.Account amAccount, EmailServiceInfo oldInfo,
|
|
|
|
EmailServiceInfo newInfo) {
|
2013-03-05 17:45:10 +00:00
|
|
|
final ContentResolver resolver = context.getContentResolver();
|
|
|
|
final Cursor c = resolver.query(Account.CONTENT_URI, Account.CONTENT_PROJECTION,
|
2012-08-21 22:21:40 +00:00
|
|
|
AccountColumns.EMAIL_ADDRESS + "=?", new String[] { amAccount.name }, null);
|
|
|
|
// That's odd, isn't it?
|
|
|
|
if (c == null) return;
|
|
|
|
try {
|
|
|
|
if (c.moveToNext()) {
|
|
|
|
// Get the EmailProvider Account/HostAuth
|
2013-03-05 17:45:10 +00:00
|
|
|
final Account account = new Account();
|
2012-08-21 22:21:40 +00:00
|
|
|
account.restore(c);
|
2013-03-05 17:45:10 +00:00
|
|
|
final HostAuth hostAuth =
|
2012-08-21 22:21:40 +00:00
|
|
|
HostAuth.restoreHostAuthWithId(context, account.mHostAuthKeyRecv);
|
2013-03-05 17:45:10 +00:00
|
|
|
if (hostAuth == null) {
|
|
|
|
return;
|
|
|
|
}
|
2012-08-21 22:21:40 +00:00
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
// Make sure this email address is using the expected protocol; our query to
|
|
|
|
// AccountManager doesn't know which protocol was being used (com.android.email
|
|
|
|
// was used for both pop3 and imap
|
|
|
|
if (!hostAuth.mProtocol.equals(oldInfo.protocol)) {
|
|
|
|
return;
|
|
|
|
}
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Converting " + amAccount.name + " to "
|
|
|
|
+ newInfo.protocol);
|
2012-08-23 05:25:42 +00:00
|
|
|
|
2013-03-05 17:45:10 +00:00
|
|
|
final ContentValues accountValues = new ContentValues();
|
2012-08-21 22:21:40 +00:00
|
|
|
int oldFlags = account.mFlags;
|
|
|
|
|
|
|
|
// Mark the provider account incomplete so it can't get reconciled away
|
|
|
|
account.mFlags |= Account.FLAGS_INCOMPLETE;
|
|
|
|
accountValues.put(AccountColumns.FLAGS, account.mFlags);
|
2013-03-05 17:45:10 +00:00
|
|
|
final Uri accountUri = ContentUris.withAppendedId(Account.CONTENT_URI, account.mId);
|
2012-08-21 22:21:40 +00:00
|
|
|
resolver.update(accountUri, accountValues, null, null);
|
|
|
|
|
|
|
|
// Change the HostAuth to reference the new protocol; this has to be done before
|
|
|
|
// trying to create the AccountManager account (below)
|
2013-03-05 17:45:10 +00:00
|
|
|
final ContentValues hostValues = new ContentValues();
|
2012-08-23 05:25:42 +00:00
|
|
|
hostValues.put(HostAuth.PROTOCOL, newInfo.protocol);
|
2012-08-21 22:21:40 +00:00
|
|
|
resolver.update(ContentUris.withAppendedId(HostAuth.CONTENT_URI, hostAuth.mId),
|
|
|
|
hostValues, null, null);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Updated HostAuths");
|
2012-08-21 22:21:40 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
// Get current settings for the existing AccountManager account
|
|
|
|
boolean email = ContentResolver.getSyncAutomatically(amAccount,
|
|
|
|
EmailContent.AUTHORITY);
|
2012-08-23 05:25:42 +00:00
|
|
|
if (!email) {
|
|
|
|
// Try our old provider name
|
|
|
|
email = ContentResolver.getSyncAutomatically(amAccount,
|
|
|
|
"com.android.email.provider");
|
|
|
|
}
|
2013-03-05 17:45:10 +00:00
|
|
|
final boolean contacts = ContentResolver.getSyncAutomatically(amAccount,
|
2012-08-21 22:21:40 +00:00
|
|
|
ContactsContract.AUTHORITY);
|
2013-03-05 17:45:10 +00:00
|
|
|
final boolean calendar = ContentResolver.getSyncAutomatically(amAccount,
|
2012-08-21 22:21:40 +00:00
|
|
|
CalendarContract.AUTHORITY);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Email: " + email + ", Contacts: " + contacts + ","
|
|
|
|
+ " Calendar: " + calendar);
|
2012-08-23 05:25:42 +00:00
|
|
|
|
|
|
|
// Get sync keys for calendar/contacts
|
2013-03-05 17:45:10 +00:00
|
|
|
final String amName = amAccount.name;
|
|
|
|
final String oldType = amAccount.type;
|
2012-08-23 05:25:42 +00:00
|
|
|
ContentProviderClient client = context.getContentResolver()
|
|
|
|
.acquireContentProviderClient(CalendarContract.CONTENT_URI);
|
|
|
|
byte[] calendarSyncKey = null;
|
|
|
|
try {
|
|
|
|
calendarSyncKey = SyncStateContract.Helpers.get(client,
|
|
|
|
asCalendarSyncAdapter(SyncState.CONTENT_URI, amName, oldType),
|
|
|
|
new android.accounts.Account(amName, oldType));
|
|
|
|
} catch (RemoteException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Get calendar key FAILED");
|
2012-08-23 05:25:42 +00:00
|
|
|
} finally {
|
|
|
|
client.release();
|
|
|
|
}
|
|
|
|
client = context.getContentResolver()
|
|
|
|
.acquireContentProviderClient(ContactsContract.AUTHORITY_URI);
|
|
|
|
byte[] contactsSyncKey = null;
|
|
|
|
try {
|
|
|
|
contactsSyncKey = SyncStateContract.Helpers.get(client,
|
|
|
|
ContactsContract.SyncState.CONTENT_URI,
|
|
|
|
new android.accounts.Account(amName, oldType));
|
|
|
|
} catch (RemoteException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Get contacts key FAILED");
|
2012-08-23 05:25:42 +00:00
|
|
|
} finally {
|
|
|
|
client.release();
|
|
|
|
}
|
|
|
|
if (calendarSyncKey != null) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Got calendar key: "
|
|
|
|
+ new String(calendarSyncKey));
|
2012-08-23 05:25:42 +00:00
|
|
|
}
|
|
|
|
if (contactsSyncKey != null) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Got contacts key: "
|
|
|
|
+ new String(contactsSyncKey));
|
2012-08-23 05:25:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set up a new AccountManager account with new type and old settings
|
|
|
|
AccountManagerFuture<?> amFuture = MailService.setupAccountManagerAccount(
|
|
|
|
context, account, email, calendar, contacts, null);
|
|
|
|
finishAccountManagerBlocker(amFuture);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Created new AccountManager account");
|
2012-08-21 22:21:40 +00:00
|
|
|
|
|
|
|
// Delete the AccountManager account
|
2012-08-23 05:25:42 +00:00
|
|
|
amFuture = AccountManager.get(context)
|
2012-08-21 22:21:40 +00:00
|
|
|
.removeAccount(amAccount, null, null);
|
|
|
|
finishAccountManagerBlocker(amFuture);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Deleted old AccountManager account");
|
2012-08-21 22:21:40 +00:00
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
// Restore sync keys for contacts/calendar
|
|
|
|
if (calendarSyncKey != null && calendarSyncKey.length != 0) {
|
|
|
|
client = context.getContentResolver()
|
|
|
|
.acquireContentProviderClient(CalendarContract.CONTENT_URI);
|
|
|
|
try {
|
|
|
|
SyncStateContract.Helpers.set(client,
|
|
|
|
asCalendarSyncAdapter(SyncState.CONTENT_URI, amName,
|
|
|
|
newInfo.accountType),
|
|
|
|
new android.accounts.Account(amName, newInfo.accountType),
|
|
|
|
calendarSyncKey);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Set calendar key...");
|
2012-08-23 05:25:42 +00:00
|
|
|
} catch (RemoteException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Set calendar key FAILED");
|
2012-08-23 05:25:42 +00:00
|
|
|
} finally {
|
|
|
|
client.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (contactsSyncKey != null && contactsSyncKey.length != 0) {
|
|
|
|
client = context.getContentResolver()
|
|
|
|
.acquireContentProviderClient(ContactsContract.AUTHORITY_URI);
|
|
|
|
try {
|
|
|
|
SyncStateContract.Helpers.set(client,
|
|
|
|
ContactsContract.SyncState.CONTENT_URI,
|
|
|
|
new android.accounts.Account(amName, newInfo.accountType),
|
|
|
|
contactsSyncKey);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Set contacts key...");
|
2012-08-23 05:25:42 +00:00
|
|
|
} catch (RemoteException e) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Set contacts key FAILED");
|
2012-08-23 05:25:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldInfo.requiresAccountUpdate) {
|
|
|
|
EmailServiceProxy service =
|
|
|
|
EmailServiceUtils.getServiceFromInfo(context, null, newInfo);
|
|
|
|
try {
|
|
|
|
service.serviceUpdated(amAccount.name);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Updated account settings");
|
2012-08-23 05:25:42 +00:00
|
|
|
} catch (RemoteException e) {
|
|
|
|
// Old settings won't hurt anyone
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// That's all folks!
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Account update completed.");
|
2012-08-21 22:21:40 +00:00
|
|
|
} finally {
|
|
|
|
// Clear the incomplete flag on the provider account
|
|
|
|
accountValues.put(AccountColumns.FLAGS, oldFlags);
|
|
|
|
resolver.update(accountUri, accountValues, null, null);
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "[Incomplete flag cleared]");
|
2012-08-21 22:21:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
c.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
private static void disableComponent(Context context, Class<?> klass) {
|
2013-05-26 04:32:32 +00:00
|
|
|
LogUtils.w(Logging.LOG_TAG, "Disabling legacy authenticator " + klass.getSimpleName());
|
2012-08-23 05:25:42 +00:00
|
|
|
final ComponentName c = new ComponentName(context, klass);
|
|
|
|
context.getPackageManager().setComponentEnabledSetting(c,
|
|
|
|
PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
|
|
|
|
PackageManager.DONT_KILL_APP);
|
|
|
|
}
|
|
|
|
|
2010-02-02 23:06:52 +00:00
|
|
|
/**
|
2012-06-28 17:40:46 +00:00
|
|
|
* Parse services.xml file to find our available email services
|
2010-02-02 23:06:52 +00:00
|
|
|
*/
|
2012-06-28 17:40:46 +00:00
|
|
|
@SuppressWarnings("unchecked")
|
2012-08-21 22:21:40 +00:00
|
|
|
private static synchronized void findServices(Context context) {
|
2012-06-28 17:40:46 +00:00
|
|
|
try {
|
2013-03-05 17:45:10 +00:00
|
|
|
final Resources res = context.getResources();
|
|
|
|
final XmlResourceParser xml = res.getXml(R.xml.services);
|
2012-06-28 17:40:46 +00:00
|
|
|
int xmlEventType;
|
|
|
|
// walk through senders.xml file.
|
|
|
|
while ((xmlEventType = xml.next()) != XmlResourceParser.END_DOCUMENT) {
|
|
|
|
if (xmlEventType == XmlResourceParser.START_TAG &&
|
|
|
|
"emailservice".equals(xml.getName())) {
|
2013-03-05 17:45:10 +00:00
|
|
|
final EmailServiceInfo info = new EmailServiceInfo();
|
|
|
|
final TypedArray ta = res.obtainAttributes(xml, R.styleable.EmailServiceInfo);
|
2012-06-28 17:40:46 +00:00
|
|
|
info.protocol = ta.getString(R.styleable.EmailServiceInfo_protocol);
|
2012-08-21 22:21:40 +00:00
|
|
|
info.accountType = ta.getString(R.styleable.EmailServiceInfo_accountType);
|
|
|
|
// Handle upgrade of one protocol to another (e.g. imap to imap2)
|
2013-03-05 17:45:10 +00:00
|
|
|
final String newProtocol =
|
|
|
|
ta.getString(R.styleable.EmailServiceInfo_replaceWith);
|
|
|
|
|
2012-08-21 22:21:40 +00:00
|
|
|
if (newProtocol != null) {
|
2013-03-05 17:45:10 +00:00
|
|
|
final EmailServiceInfo newInfo = getServiceInfo(context, newProtocol);
|
2012-08-21 22:21:40 +00:00
|
|
|
if (newInfo == null) {
|
|
|
|
throw new IllegalStateException(
|
|
|
|
"Replacement service not found: " + newProtocol);
|
|
|
|
}
|
2013-03-05 17:45:10 +00:00
|
|
|
sServiceMap.put(info.protocol, newInfo);
|
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
info.requiresAccountUpdate = ta.getBoolean(
|
|
|
|
R.styleable.EmailServiceInfo_requiresAccountUpdate, false);
|
2013-03-05 17:45:10 +00:00
|
|
|
final AccountManager am = AccountManager.get(context);
|
|
|
|
final android.accounts.Account[] amAccounts =
|
2012-08-21 22:21:40 +00:00
|
|
|
am.getAccountsByType(info.accountType);
|
|
|
|
for (android.accounts.Account amAccount: amAccounts) {
|
2012-08-23 05:25:42 +00:00
|
|
|
new UpdateAccountManagerTask(context, amAccount, info, newInfo)
|
2013-03-05 17:45:10 +00:00
|
|
|
.executeOnExecutor(AsyncTask.SERIAL_EXECUTOR);
|
2012-08-21 22:21:40 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2012-06-28 17:40:46 +00:00
|
|
|
info.name = ta.getString(R.styleable.EmailServiceInfo_name);
|
2012-09-20 20:34:13 +00:00
|
|
|
info.hide = ta.getBoolean(R.styleable.EmailServiceInfo_hide, false);
|
2013-03-05 17:45:10 +00:00
|
|
|
final String klass = ta.getString(R.styleable.EmailServiceInfo_serviceClass);
|
2012-06-28 17:40:46 +00:00
|
|
|
info.intentAction = ta.getString(R.styleable.EmailServiceInfo_intent);
|
|
|
|
info.defaultSsl = ta.getBoolean(R.styleable.EmailServiceInfo_defaultSsl, false);
|
|
|
|
info.port = ta.getInteger(R.styleable.EmailServiceInfo_port, 0);
|
|
|
|
info.portSsl = ta.getInteger(R.styleable.EmailServiceInfo_portSsl, 0);
|
|
|
|
info.offerTls = ta.getBoolean(R.styleable.EmailServiceInfo_offerTls, false);
|
|
|
|
info.offerCerts = ta.getBoolean(R.styleable.EmailServiceInfo_offerCerts, false);
|
|
|
|
info.offerLocalDeletes =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_offerLocalDeletes, false);
|
|
|
|
info.defaultLocalDeletes =
|
|
|
|
ta.getInteger(R.styleable.EmailServiceInfo_defaultLocalDeletes,
|
|
|
|
Account.DELETE_POLICY_ON_DELETE);
|
|
|
|
info.offerPrefix =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_offerPrefix, false);
|
|
|
|
info.usesSmtp = ta.getBoolean(R.styleable.EmailServiceInfo_usesSmtp, false);
|
|
|
|
info.usesAutodiscover =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_usesAutodiscover, false);
|
|
|
|
info.offerLookback =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_offerLookback, false);
|
|
|
|
info.defaultLookback =
|
|
|
|
ta.getInteger(R.styleable.EmailServiceInfo_defaultLookback,
|
|
|
|
SyncWindow.SYNC_WINDOW_3_DAYS);
|
|
|
|
info.syncChanges =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_syncChanges, false);
|
|
|
|
info.syncContacts =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_syncContacts, false);
|
|
|
|
info.syncCalendar =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_syncCalendar, false);
|
|
|
|
info.offerAttachmentPreload =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_offerAttachmentPreload, false);
|
|
|
|
info.syncIntervalStrings =
|
|
|
|
ta.getTextArray(R.styleable.EmailServiceInfo_syncIntervalStrings);
|
|
|
|
info.syncIntervals =
|
|
|
|
ta.getTextArray(R.styleable.EmailServiceInfo_syncIntervals);
|
|
|
|
info.defaultSyncInterval =
|
|
|
|
ta.getInteger(R.styleable.EmailServiceInfo_defaultSyncInterval, 15);
|
2012-08-16 21:11:11 +00:00
|
|
|
info.inferPrefix = ta.getString(R.styleable.EmailServiceInfo_inferPrefix);
|
2012-08-24 02:40:47 +00:00
|
|
|
info.offerLoadMore =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_offerLoadMore, false);
|
2012-09-08 20:08:50 +00:00
|
|
|
info.requiresSetup =
|
|
|
|
ta.getBoolean(R.styleable.EmailServiceInfo_requiresSetup, false);
|
2012-04-25 17:26:46 +00:00
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
// Must have either "class" (local) or "intent" (remote)
|
|
|
|
if (klass != null) {
|
|
|
|
try {
|
|
|
|
info.klass = (Class<? extends Service>) Class.forName(klass);
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new IllegalStateException(
|
|
|
|
"Class not found in service descriptor: " + klass);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (info.klass == null && info.intentAction == null) {
|
|
|
|
throw new IllegalStateException(
|
|
|
|
"No class or intent action specified in service descriptor");
|
|
|
|
}
|
|
|
|
if (info.klass != null && info.intentAction != null) {
|
|
|
|
throw new IllegalStateException(
|
|
|
|
"Both class and intent action specified in service descriptor");
|
|
|
|
}
|
|
|
|
sServiceList.add(info);
|
2013-03-05 17:45:10 +00:00
|
|
|
sServiceMap.put(info.protocol, info);
|
2012-06-28 17:40:46 +00:00
|
|
|
}
|
|
|
|
}
|
2012-08-23 05:25:42 +00:00
|
|
|
// Disable our legacy components
|
|
|
|
new DisableComponentsTask(context).executeOnExecutor(AsyncTask.SERIAL_EXECUTOR);
|
2012-06-28 17:40:46 +00:00
|
|
|
} catch (XmlPullParserException e) {
|
|
|
|
// ignore
|
|
|
|
} catch (IOException e) {
|
|
|
|
// ignore
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
2012-06-28 17:40:46 +00:00
|
|
|
}
|
2012-04-25 17:26:46 +00:00
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
private static Uri asCalendarSyncAdapter(Uri uri, String account, String accountType) {
|
|
|
|
return uri.buildUpon().appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true")
|
|
|
|
.appendQueryParameter(Calendars.ACCOUNT_NAME, account)
|
|
|
|
.appendQueryParameter(Calendars.ACCOUNT_TYPE, accountType).build();
|
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
/**
|
|
|
|
* A no-op service that can be returned for non-existent/null protocols
|
|
|
|
*/
|
|
|
|
class NullService implements IEmailService {
|
|
|
|
@Override
|
|
|
|
public IBinder asBinder() {
|
|
|
|
return null;
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public Bundle validate(HostAuth hostauth) throws RemoteException {
|
|
|
|
return null;
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
2013-04-05 01:30:39 +00:00
|
|
|
public void startSync(long mailboxId, boolean userRequest, int deltaMessageCount)
|
|
|
|
throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void stopSync(long mailboxId) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void loadMore(long messageId) throws RemoteException {
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2013-07-15 21:44:54 +00:00
|
|
|
public void loadAttachment(final IEmailServiceCallback cb, final long attachmentId,
|
|
|
|
final boolean background) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void updateFolderList(long accountId) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public boolean createFolder(long accountId, String name) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public boolean deleteFolder(long accountId, String name) throws RemoteException {
|
|
|
|
return false;
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public boolean renameFolder(long accountId, String oldName, String newName)
|
|
|
|
throws RemoteException {
|
|
|
|
return false;
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void setCallback(IEmailServiceCallback cb) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void setLogging(int on) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void hostChanged(long accountId) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public Bundle autoDiscover(String userName, String password) throws RemoteException {
|
|
|
|
return null;
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void sendMeetingResponse(long messageId, int response) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
2012-04-25 17:26:46 +00:00
|
|
|
public void deleteAccountPIMData(long accountId) throws RemoteException {
|
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public int getApiLevel() throws RemoteException {
|
|
|
|
return Api.LEVEL;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int searchMessages(long accountId, SearchParams params, long destMailboxId)
|
|
|
|
throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-06-28 17:40:46 +00:00
|
|
|
@Override
|
|
|
|
public void sendMail(long accountId) throws RemoteException {
|
2012-04-25 17:26:46 +00:00
|
|
|
}
|
|
|
|
|
2012-08-23 05:25:42 +00:00
|
|
|
@Override
|
|
|
|
public void serviceUpdated(String emailAddress) throws RemoteException {
|
|
|
|
}
|
|
|
|
|
2012-04-25 17:26:46 +00:00
|
|
|
@Override
|
2012-07-25 19:23:01 +00:00
|
|
|
public int getCapabilities(Account acct) throws RemoteException {
|
2012-06-28 17:40:46 +00:00
|
|
|
return 0;
|
2010-08-20 16:53:46 +00:00
|
|
|
}
|
2010-02-02 19:17:48 +00:00
|
|
|
}
|
|
|
|
}
|